--- rxvt-unicode/src/perl/selection 2006/01/17 16:22:41 1.29 +++ rxvt-unicode/src/perl/selection 2012/06/10 17:31:53 1.55 @@ -1,105 +1,121 @@ #! perl -use Digest::MD5 qw/md5_hex/; -my $timers = {}; -my $pastebin_cmd; -my $pastebin_url; +#:META:X_RESOURCE:%.pattern-0:string:first selection pattern -sub on_start { - my ($self) = @_; - $pastebin_cmd = $self->x_resource ("selection-pastebin-cmd") - or "scp -p % ruth:/var/www/www.ta-sa.org/files/txt/"; +=head1 NAME - $pastebin_url = $self->x_resource ("selection-pastebin-url") - or "http://www.ta-sa.org/files/txt/"; - (); -} + selection - more intelligent selection (enabled by default) -sub upload_paste { - my ($self) = @_; +=head1 DESCRIPTION - my $txt = $self->selection; - my $filename = md5_hex ($txt) . ".txt"; +This extension tries to be more intelligent when the user extends +selections (double-click and further clicks). Right now, it tries to +select words, urls and complete shell-quoted arguments, which is very +convenient, too, if your F supports C<--quoting-style=shell>. - my $tmpfile = "/tmp/$filename"; +A double-click usually selects the word under the cursor, further clicks +will enlarge the selection. - my $msg = "uploaded $filename"; +The selection works by trying to match a number of regexes and displaying +them in increasing order of length. You can add your own regexes by +specifying resources of the form: - if (open my $o, ">" . $tmpfile) { - print $o $txt; - close $o; - } else { - $msg = "couldn't write $tmpfile: $!"; - } + URxvt.selection.pattern-0: perl-regex + URxvt.selection.pattern-1: perl-regex + ... - my $cmd = $pastebin_cmd; - $cmd =~ s/%/$tmpfile/; +The index number (0, 1...) must not have any holes, and each regex must +contain at least one pair of capturing parentheses, which will be used for +the match. For example, the following adds a regex that matches everything +between two vertical bars: - if (system ($cmd) == 0) { + URxvt.selection.pattern-0: \\|([^|]+)\\| - my $url = $pastebin_url; - $url =~ s/%/$filename/; +Another example: Programs I use often output "absolute path: " at the +beginning of a line when they process multiple files. The following +pattern matches the filename (note, there is a single space at the very +end): - $self->selection ($url); - } else { - $msg = "couldn't upload, '$cmd' failed"; - } + URxvt.selection.pattern-0: ^(/[^:]+):\ - my $ov = $timers->{ov} = $self->overlay (-1, 0, length ($msg), 1, urxvt::OVERLAY_RSTYLE, 0); - $ov->set (0, 0, $msg); +You can look at the source of the selection extension to see more +interesting uses, such as parsing a line from beginning to end. - $timers->{t1} = - urxvt::timer - ->new - ->start ((int urxvt::NOW) + 5) # make sure we update "on" the second - ->interval (1) - ->cb (sub { delete $timers->{ov}; delete $timers->{t1}; }); -} +This extension also offers following bindable keyboard commands: + +=over 4 + +=item rot13 + +Rot-13 the selection when activated. Used via keyboard trigger: + + URxvt.keysym.C-M-r: perl:selection:rot13 + +=back -sub on_keyboard_command { +=cut + +sub on_user_command { my ($self, $cmd) = @_; $cmd eq "selection:rot13" and $self->selection (map { y/A-Za-z/N-ZA-Mn-za-m/; $_ } $self->selection); - $cmd eq "selection:remote-pastebin" - and upload_paste ($self); - () } sub on_init { my ($self) = @_; + if (defined (my $res = $self->resource ("cutchars"))) { + $res = $self->locale_decode ($res); + push @{ $self->{patterns} }, qr{\G [\Q$res\E[:space:]]* ([^\Q$res\E[:space:]]+) }x; + } + for (my $idx = 0; defined (my $res = $self->x_resource ("selection.pattern-$idx")); $idx++) { - no re 'eval'; # just to be sure - $res = utf8::encode $self->locale_decode ($res); + $res = $self->locale_decode ($res); push @{ $self->{patterns} }, qr/$res/; } + $self->{enabled} = 1; + + push @{ $self->{term}{option_popup_hook} }, sub { + ("new selection" => $self->{enabled}, sub { $self->{enabled} = shift }) + }; + () } -# "find interetsing things"-patterns +# "find interesting things"-patterns my @mark_patterns = ( +# qr{ ([[:word:]]+) }x, + qr{ ([^[:space:]]+) }x, + # common types of "parentheses" - qr{ (?[:space:]]+) \> }x, - qr{ \{ ([^{}[:space:]]+) \} }x, - qr{ \[ ([^{}[:space:]]+) \] }x, - qr{ \( ([^()[:space:]]+) \) }x, + qr{ (?]+) \> }x, # urls, just a heuristic qr{( - (?:https?|ftp|news|mailto|file)://[ab-zA-Z0-9\-\@;\/?:&=%\$_.+!*\x27(),~]+ - [ab-zA-Z0-9\-\@;\/?:&=%\$_+!*\x27()~] # exclude some trailing characters (heuristic) + (?:https?://|ftp://|news://|mailto:|file://|\bwww\.)[ab-zA-Z0-9\-\@;\/?:&=%\$_.+!*\x27(),~#]+ + [ab-zA-Z0-9\-\@;\/?:&=%\$_+*()~] # exclude some trailing characters (heuristic) )}x, # shell-like argument quoting, basically always matches - qr{\G [\ \t|&;<>()] *( + qr{\G [\ \t|&;<>()]* ( (?: [^\\"'\ \t|&;<>()]+ | \\. @@ -118,6 +134,9 @@ sub on_sel_extend { my ($self, $time) = @_; + $self->{enabled} + or return; + my ($row, $col) = $self->selection_mark; my $line = $self->line ($row); my $text = $line->t; @@ -127,26 +146,24 @@ my @matches; - # not doing matches in unicode mode helps speed - # enourmously here. working in utf-8 should be - # equivalent due to the magic of utf-8 encoding. - utf8::encode $text; - study $text; # _really_ helps, too :) - - for my $regex (@mark_patterns, @{ $self->{patterns} }) { - while ($text =~ /$regex/g) { - if ($-[1] <= $markofs and $markofs <= $+[1]) { - my $ofs = $-[1]; - my $match = $1; - - for my $regex (@simplify_patterns) { - if ($match =~ $regex) { - $match = $1; - $ofs += $-[1]; + if ($markofs < $line->l) { + study $text; # _really_ helps, too :) + + for my $regex (@mark_patterns, @{ $self->{patterns} }) { + while ($text =~ /$regex/g) { + if ($-[1] <= $markofs and $markofs <= $+[1]) { + my $ofs = $-[1]; + my $match = $1; + + for my $regex (@simplify_patterns) { + if ($match =~ $regex) { + $match = $1; + $ofs += $-[1]; + } } - } - push @matches, [$ofs, length $match]; + push @matches, [$ofs, length $match]; + } } } }