From b609e772d534642a86800107fb970cc7fbdd1822 Mon Sep 17 00:00:00 2001 From: sf-exg Date: Thu, 30 Oct 2014 09:44:58 +0000 Subject: Destroy overlay timer on terminal destruction in selection-{autotransform,popup}. --- src/perl/selection-autotransform | 101 +++++++++++++++++++++++++++ src/perl/selection-popup | 147 +++++++++++++++++++++++++++++++++++++++ 2 files changed, 248 insertions(+) create mode 100644 src/perl/selection-autotransform create mode 100644 src/perl/selection-popup diff --git a/src/perl/selection-autotransform b/src/perl/selection-autotransform new file mode 100644 index 0000000..044dec4 --- /dev/null +++ b/src/perl/selection-autotransform @@ -0,0 +1,101 @@ +#! perl + +#:META:RESOURCE:%.:string:autotransform expression + +=head1 NAME + +selection-autotransform - automatically transform select text + +=head1 DESCRIPTION + +This selection allows you to do automatic transforms on a selection +whenever a selection is made. + +It works by specifying perl snippets (most useful is a single C +operator) that modify C<$_> as resources: + + URxvt.selection-autotransform.0: transform + URxvt.selection-autotransform.1: transform + ... + +For example, the following will transform selections of the form +C, often seen in compiler messages, into C: + + URxvt.selection-autotransform.0: s/^([^:[:space:]]+):(\\d+):?$/vi +$2 \\Q$1\\E\\x0d/ + +And this example matches the same,but replaces it with vi-commands you can +paste directly into your (vi :) editor: + + URxvt.selection-autotransform.0: s/^([^:[:space:]]+(\\d+):?$/:e \\Q$1\\E\\x0d:$2\\x0d/ + +Of course, this can be modified to suit your needs and your editor :) + +To expand the example above to typical perl error messages ("XXX at +FILENAME line YYY."), you need a slightly more elaborate solution: + + URxvt.selection.pattern-0: ( at .*? line \\d+[,.]) + URxvt.selection-autotransform.0: s/^ at (.*?) line (\\d+)[,.]$/:e \\Q$1\E\\x0d:$2\\x0d/ + +The first line tells the selection code to treat the unchanging part of +every error message as a selection pattern, and the second line transforms +the message into vi commands to load the file. + +=cut + +sub msg { + my ($self, $msg) = @_; + + my $overlay = $self->overlay (0, 0, $self->strwidth ($msg), 1); + $overlay->set (0, 0, $msg); + $self->{timer} = urxvt::timer->new->after (2)->cb (sub { + delete $self->{timer}; + undef $overlay; + }); +} + +sub on_init { + my ($self) = @_; + + for (my $idx = 0; defined (my $res = $self->x_resource ("%.$idx")); $idx++) { + $res = $self->locale_decode ($res); + my $transform = eval "sub { $res }"; + + if ($transform) { + push @{ $self->{transforms} }, $transform; + } else { + warn "$res: $@"; + } + } + + $self->{enabled} = 1; + + push @{ $self->{term}{option_popup_hook} }, sub { + ("autotransform" => $self->{enabled}, sub { $self->{enabled} = shift }) + }; + + () +} + +sub on_sel_grab { + my ($self) = @_; + + $self->{enabled} + or return; + + my $text = $self->selection; + local $_ = $text; + + for my $transform (@{ $self->{transforms} }) { + $transform->(); + if ($text ne $_) { + $self->selection ($_); + s/[\x00-\x1f\x80-\x9f]/·/g; + $self->msg ($self->special_encode ("auto-transformed to $_")); + last; + } + } + + () +} + diff --git a/src/perl/selection-popup b/src/perl/selection-popup new file mode 100644 index 0000000..07811d8 --- /dev/null +++ b/src/perl/selection-popup @@ -0,0 +1,147 @@ +#! perl + +#:META:RESOURCE:url-launcher:string:shell command to use + +=head1 NAME + +selection-popup (enabled by default) + +=head1 DESCRIPTION + +Binds a popup menu to Ctrl-Button3 that lets you paste the X +selections and either modify or use the internal selection text in +various ways (such as uri unescaping, perl evaluation, web-browser +starting etc.), depending on content. + +Other extensions can extend this popup menu by pushing a code reference +onto C<< @{ $term->{selection_popup_hook} } >>, which gets called whenever +the popup is being displayed. + +Its sole argument is the popup menu, which can be modified. The selection +is in C<$_>, which can be used to decide whether to add something or not. +It should either return nothing or a string and a code reference. The +string will be used as button text and the code reference will be called +when the button gets activated and should transform C<$_>. + +The following will add an entry C that transforms all Cs in +the selection to Cs, but only if the selection currently contains any +Cs: + + push @{ $self->{term}{selection_popup_hook} }, sub { + /a/ ? ("a to b" => sub { s/a/b/g } + : () + }; + +=cut + +sub msg { + my ($self, $msg) = @_; + + my $overlay = $self->overlay (0, 0, $self->strwidth ($msg), 1); + $overlay->set (0, 0, $msg); + $self->{timer} = urxvt::timer->new->after (1)->cb (sub { + delete $self->{timer}; + undef $overlay; + }); +} + +sub on_start { + my ($self) = @_; + + $self->{browser} = $self->x_resource ("url-launcher") || "sensible-browser"; + + $self->grab_button (3, urxvt::ControlMask); + + () +} + +sub on_button_press { + my ($self, $event) = @_; + + if ($event->{button} == 3 && $event->{state} & urxvt::ControlMask) { + my $popup = $self->popup ($event) + or return 1; + + $popup->add_title ("Selection"); + + my $text = $self->selection; + + my $title = $text; + $title =~ s/[\x00-\x1f\x80-\x9f]/·/g; + substr $title, 40, -1, "..." if 40 < length $title; + $popup->add_title ($title); + $popup->add_separator; + + my $add_button = sub { + my ($title, $cb) = @_; + + $popup->add_button ($title => sub { + for ($text) { + my $orig = $_; + $cb->(); + + if ($orig ne $_) { + $self->selection ($_); + s/[\x00-\x1f\x80-\x9f]/·/g; + $self->msg ($self->special_encode ($_)); + } + } + }); + }; + + for ($text) { + /\n/ + and $add_button->("paste primary selection" => sub { $self->selection_request (urxvt::CurrentTime, 1) }); + + /./ + and $add_button->("paste clipboard selection" => sub { $self->selection_request (urxvt::CurrentTime, 3) }); + + /./ + and $add_button->("copy selection to clipboard" => sub { $self->selection ($self->selection, 1); + $self->selection_grab (urxvt::CurrentTime, 1) }); + + /./ + and $add_button->("newlines to spaces" => sub { y/\n/ / }); + + /./ + and $add_button->("rot13" => sub { y/A-Za-z/N-ZA-Mn-za-m/ }); + + /./ + and $add_button->("eval perl expression" => sub { my $self = $self; no warnings; $_ = eval $_; $_ = "$@" if $@ }); + + /./ + and $add_button->((sprintf "to unicode hex index (%x)", ord) => sub { $_ = sprintf "%x", ord }); + + /(\S+):(\d+):?/ + and $add_button->("vi-commands to load '$1'" => sub { s/^(\S+):(\d+):?$/\x1b:e $1\x0d:$2\x0d/ }); + + /%[0-9a-fA-F]{2}/ && !/%[^0-9a-fA-F]/ && !/%.[^0-9a-fA-F]/ + and $add_button->("uri unescape" => sub { s/%([0-9a-fA-F]{2})/chr hex $1/ge }); + + /[\\"'\ \t|&;<>()]/ + and $add_button->("shell quote" => sub { $_ = "\Q$_" }); + + /^(https?|ftp|telnet|irc|news):\// + and $add_button->("run $self->{browser}" => sub { $self->exec_async ($self->{browser}, $_) }); + + for my $hook (@{ $self->{term}{selection_popup_hook} || [] }) { + if (my ($title, $cb) = $hook->($popup)) { + $add_button->($title, $cb); + } + } + + if (/^\s*((?:0x)?\d+)\s*$/) { + $popup->add_title (sprintf "%20s", eval $1); + $popup->add_title (sprintf "%20s", sprintf "0x%x", eval $1); + $popup->add_title (sprintf "%20s", sprintf "0%o", eval $1); + } + } + + $popup->show; + + return 1; + } + + () +} + -- cgit v1.2.3