App-ClipboardUtils
view release on metacpan or search on metacpan
lib/App/ClipboardUtils.pm view on Meta::CPAN
package App::ClipboardUtils;
use strict;
use warnings;
use Log::ger;
our $AUTHORITY = 'cpan:PERLANCAR'; # AUTHORITY
our $DATE = '2025-10-16'; # DATE
our $DIST = 'App-ClipboardUtils'; # DIST
our $VERSION = '0.014'; # VERSION
use Clipboard::Any ();
use Clone::PP qw(clone);
our %SPEC;
{
$SPEC{add_clipboard_content} = clone $Clipboard::Any::SPEC{add_clipboard_content};
# because we also have --command and do our own while(<>) { ... }
delete $SPEC{add_clipboard_content}{args}{content}{cmdline_src};
$SPEC{add_clipboard_content}{args}{split_by} = {
summary => 'Split content by specified string/regex, add the split content as multiple clipboard entries',
schema => ['str_or_re*'],
description => <<'MARKDOWN',
Cannot be used together with `--fragments` or `--command-line` option.
Note that if you supply a regex, you should not have any capture groups in the
regex.
MARKDOWN
cmdline_aliases => {s=>{}},
};
$SPEC{add_clipboard_content}{args}{fragments} = {
summary => 'Only add text fragments inside content',
schema => ['bool*'],
description => <<'MARKDOWN',
Cannot be used together with `--split-by` or `--command-line` option.
Example content:
FAQS
====
Q: Will the Thingamagic explode under extreme pressure?
A: No doubt.
# BEGIN clipadd id=2
Q: How much pressure should I apply to the Thingamagic initially?
A: Not that much, really. It will pressurize itself eventually.
# END clipadd id=2
# BEGIN clipadd id=1
Q: What is the appropriate age to start using the Thingamagic?
A: It depends on whether your kid is smart (or stupid) enough.
# END clipadd id=1
Q: Another question?
A: Another half-assed answer.
Command:
% cat faq.txt | clipadd --fragments
Read <pm:Text::Fragment> for more details on text fragments.
MARKDOWN
#cmdline_aliases => {f=>{}},
};
$SPEC{add_clipboard_content}{args}{tee} = {
summary => 'Pass stdin to stdout',
schema => ['true*'],
description => <<'MARKDOWN',
MARKDOWN
cmdline_aliases => {t=>{}},
};
$SPEC{add_clipboard_content}{args}{command_line} = {
summary => 'For every line of input in *stdin*, execute a command, feed it the input line, and add the output to clipboard',
schema => ['str*'],
description => <<'MARKDOWN',
Note that when you use this option, the `--content` argument is ignored. Input
is taken from stdin. With `--tee`, each output will be printed to stdout. After
eof, the utility will return empty result.
An example for using this option (<prog:safer> is a utility from <pm:App::safer>):
% clipadd -c safer --tee
Foo Bar, Co., Ltd.
foo-bar-co-ltd
BaZZ, Co., Ltd.
bazz-co-ltd
_
MARKDOWN
cmdline_aliases => {c=>{}},
};
$SPEC{add_clipboard_content}{args_rels}{"choose_one&"} = [
[qw/command_line split_by fragments/],
];
}
sub add_clipboard_content {
my %args = @_;
my $split_by = delete $args{split_by};
my $tee = delete $args{tee};
my $command_line = $args{command_line};
if (defined $command_line) {
require IPC::System::Options;
while (defined(my $input_line = <>)) {
my $stdout;
IPC::System::Options::run({log=>1, die=>1, stdin => $input_line, capture_stdout => \$stdout}, $command_line);
if (defined $split_by) {
my $content = delete $args{content};
my @split_parts = split /($split_by)/, $content;
log_trace "split_by=%s, split_contents=%s", $split_by, \@split_parts;
my $i = 0;
while (my ($part, $separator) = splice @split_parts, 0, 2) {
if ($tee) {
print $part;
print $separator if defined $separator;
}
# do not add empty part to clipboard
if (length $part) {
my $res = Clipboard::Any::add_clipboard_content(
%args, content => $part,
);
return $res unless $res->[0] == 200;
}
}
} else {
print $stdout if $tee;
my $res = Clipboard::Any::add_clipboard_content(%args, content => $stdout);
return $res unless $res->[0] == 200;
}
} # while input
return [200, "OK"];
} else {
my $content = $args{content};
$content = do { local $/; scalar <> } unless defined $content;
$args{content} = $content;
if (defined $split_by) {
my @split_parts = split /($split_by)/, $content;
log_trace "split_by=%s, split_contents=%s", $split_by, \@split_parts;
my $res = [204, "OK (no content)"];
my $i = 0;
while (my ($part, $separator) = splice @split_parts, 0, 2) {
if ($tee) {
print $part;
print $separator if defined $separator;
}
# do not add empty part to clipboard
if (length $part) {
$res = Clipboard::Any::add_clipboard_content(
%args, content => $part,
); # currently we use the last add_clipboard_content status
}
}
$res->[3]{'func.parts'} = @split_parts;
$res;
} elsif ($args{fragments}) {
require Text::Fragment;
my $lfres = Text::Fragment::list_fragments(text => $content, label => "clipadd");
return [500, "Can't list fragments in content: $lfres->[0] - $lfres->[1]"]
unless $lfres->[0] == 200;
my @parts = map { $_->{payload} } sort { $a->{id} <=> $b->{id} } @{ $lfres->[2] };
my $res = [204, "OK (no content)"];
for my $part (@parts) {
if ($tee) {
print $part;
}
# do not add empty part to clipboard
if (length $part) {
$res = Clipboard::Any::add_clipboard_content(
%args, content => $part,
); # currently we use the last add_clipboard_content status
}
}
$res->[3]{'func.parts'} = @parts;
$res;
} else {
print $content if $tee;
Clipboard::Any::add_clipboard_content(%args);
}
} # if command_line
}
$SPEC{tee_clipboard_content} = clone $Clipboard::Any::SPEC{add_clipboard_content};
$SPEC{tee_clipboard_content}{summary} = 'Shortcut for add-clipboard-content --tee';
$SPEC{tee_clipboard_content}{description} = '';
delete $SPEC{tee_clipboard_content}{args}{tee};
sub tee_clipboard_content {
add_clipboard_content(@_, tee => 1);
}
1;
# ABSTRACT: CLI utilities related to clipboard
__END__
=pod
=encoding UTF-8
=head1 NAME
App::ClipboardUtils - CLI utilities related to clipboard
=head1 VERSION
This document describes version 0.014 of App::ClipboardUtils (from Perl distribution App-ClipboardUtils), released on 2025-10-16.
=head1 DESCRIPTION
This distribution contains the following CLI utilities related to clipboard:
=over
=item 1. L<add-clipboard-content>
=item 2. L<ca>
=item 3. L<cg>
=item 4. L<clear-clipboard-content>
=item 5. L<clear-clipboard-history>
=item 6. L<clipadd>
=item 7. L<clipget>
=item 8. L<cliptee>
=item 9. L<ct>
=item 10. L<detect-clipboard-manager>
=item 11. L<get-clipboard-content>
=item 12. L<get-clipboard-history-item>
=item 13. L<list-clipboard-history>
=item 14. L<tee-clipboard-content>
=back
=head1 FUNCTIONS
=head2 add_clipboard_content
Usage:
add_clipboard_content(%args) -> [$status_code, $reason, $payload, \%result_meta]
Add a new content to the clipboard.
For C<xclip>: when adding content, the primary selection is set. The clipboard
content is unchanged.
This function is not exported.
Arguments ('*' denotes required arguments):
=over 4
=item * B<chomp_newline> => I<bool>
Remove trailing newlines before adding item to clipboard.
=item * B<clipboard_manager> => I<str>
Explicitly set clipboard manager to use.
The default, when left undef, is to detect what clipboard manager is running.
=item * B<command_line> => I<str>
For every line of input in *stdin*, execute a command, feed it the input line, and add the output to clipboard.
Note that when you use this option, the C<--content> argument is ignored. Input
is taken from stdin. With C<--tee>, each output will be printed to stdout. After
eof, the utility will return empty result.
An example for using this option (L<safer> is a utility from L<App::safer>):
% clipadd -c safer --tee
Foo Bar, Co., Ltd.
foo-bar-co-ltd
BaZZ, Co., Ltd.
bazz-co-ltd
_
=item * B<content> => I<str>
(No description)
=item * B<fragments> => I<bool>
Only add text fragments inside content.
Cannot be used together with C<--split-by> or C<--command-line> option.
Example content:
FAQS
====
Q: Will the Thingamagic explode under extreme pressure?
A: No doubt.
# BEGIN clipadd id=2
Q: How much pressure should I apply to the Thingamagic initially?
A: Not that much, really. It will pressurize itself eventually.
# END clipadd id=2
# BEGIN clipadd id=1
Q: What is the appropriate age to start using the Thingamagic?
A: It depends on whether your kid is smart (or stupid) enough.
# END clipadd id=1
Q: Another question?
A: Another half-assed answer.
Command:
% cat faq.txt | clipadd --fragments
Read L<Text::Fragment> for more details on text fragments.
=item * B<split_by> => I<str_or_re>
Split content by specified stringE<sol>regex, add the split content as multiple clipboard entries.
Cannot be used together with C<--fragments> or C<--command-line> option.
Note that if you supply a regex, you should not have any capture groups in the
regex.
=item * B<tee> => I<true>
Pass stdin to stdout.
=back
Returns an enveloped result (an array).
First element ($status_code) is an integer containing HTTP-like status code
(200 means OK, 4xx caller error, 5xx function error). Second element
($reason) is a string containing error message, or something like "OK" if status is
200. Third element ($payload) is the actual result, but usually not present when enveloped result is an error response ($status_code is not 2xx). Fourth
element (%result_meta) is called result metadata and is optional, a hash
that contains extra information, much like how HTTP response headers provide additional metadata.
Return value: (any)
=head2 tee_clipboard_content
Usage:
tee_clipboard_content(%args) -> [$status_code, $reason, $payload, \%result_meta]
Shortcut for add-clipboard-content --tee.
This function is not exported.
Arguments ('*' denotes required arguments):
=over 4
=item * B<chomp_newline> => I<bool>
Remove trailing newlines before adding item to clipboard.
=item * B<clipboard_manager> => I<str>
Explicitly set clipboard manager to use.
The default, when left undef, is to detect what clipboard manager is running.
=item * B<content> => I<str>
(No description)
=back
Returns an enveloped result (an array).
First element ($status_code) is an integer containing HTTP-like status code
(200 means OK, 4xx caller error, 5xx function error). Second element
($reason) is a string containing error message, or something like "OK" if status is
200. Third element ($payload) is the actual result, but usually not present when enveloped result is an error response ($status_code is not 2xx). Fourth
element (%result_meta) is called result metadata and is optional, a hash
that contains extra information, much like how HTTP response headers provide additional metadata.
Return value: (any)
=head1 HOMEPAGE
Please visit the project's homepage at L<https://metacpan.org/release/App-ClipboardUtils>.
=head1 SOURCE
Source repository is at L<https://github.com/perlancar/perl-App-ClipboardUtils>.
=head1 AUTHOR
perlancar <perlancar@cpan.org>
=head1 CONTRIBUTING
To contribute, you can send patches by email/via RT, or send pull requests on
GitHub.
Most of the time, you don't need to build the distribution yourself. You can
simply modify the code, then test via:
% prove -l
If you want to build the distribution (e.g. to try to install it locally on your
system), you can install L<Dist::Zilla>,
L<Dist::Zilla::PluginBundle::Author::PERLANCAR>,
L<Pod::Weaver::PluginBundle::Author::PERLANCAR>, and sometimes one or two other
Dist::Zilla- and/or Pod::Weaver plugins. Any additional steps required beyond
that are considered a bug and can be reported to me.
=head1 COPYRIGHT AND LICENSE
This software is copyright (c) 2025 by perlancar <perlancar@cpan.org>.
This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.
=head1 BUGS
Please report any bugs or feature requests on the bugtracker website L<https://rt.cpan.org/Public/Dist/Display.html?Name=App-ClipboardUtils>
( run in 3.843 seconds using v1.01-cache-2.11-cpan-df04353d9ac )