--- rxvt-unicode/src/perl/background 2012/06/04 21:39:56 1.1 +++ rxvt-unicode/src/perl/background 2012/06/07 12:56:27 1.28 @@ -1,92 +1,311 @@ #! perl -our $EXPR = 'resize (blur (load "/root/pix/das_fette_schwein.jpg", 1 + (counter 1) % 10, 10), w, h)'; +#:META:X_RESOURCE:%.expr:string:background expression +#:META:X_RESOURCE:%.enable:boolean:some boolean +#:META:X_RESOURCE:%.extra.:value:extra config + +our $EXPR; +#$EXPR = 'move W * 0.1, -H * 0.1, resize W * 0.5, H * 0.5, repeat_none load "opensource.png"'; +$EXPR = 'border; move -X, -Y, load "argb.png"'; +#$EXPR = ' +# rotate W, H, 50, 50, counter 1/59.95, repeat_mirror, +# clip X, Y, W, H, repeat_mirror, +# load "/root/pix/das_fette_schwein.jpg" +#'; +#$EXPR = 'solid "red"'; #$EXPR = 'blur root, 10, 10' #$EXPR = 'blur move (root, -x, -y), 5, 5' #resize load "/root/pix/das_fette_schwein.jpg", w, h use Safe; -{ - package urxvt::bgdsl::vars; +our $border; +our ($bgdsl_self, $old, $new); +our ($l, $t, $w, $h); - our ($self, $old, $new); - our ($x, $y, $w, $h); +# enforce at least this interval between updates +our $MIN_INTERVAL = 1/100; +{ package urxvt::bgdsl; # background language - *repeat_black = \&urxvt::RepeatNone; #TODO wtf - *repeat_wrap = \&urxvt::RepeatNormal; - *repeat_pad = \&urxvt::RepeatPad; - *repeat_mirror = \&urxvt::RepeatReflect; +# *repeat_empty = \&urxvt::RepeatNone; +# *repeat_tile = \&urxvt::RepeatNormal; +# *repeat_pad = \&urxvt::RepeatPad; +# *repeat_mirror = \&urxvt::RepeatReflect; + +=head2 PROVIDERS/GENERATORS + +=over 4 - sub load { +=item load $path + +=cut + + sub load($) { my ($path) = @_; - $new->{load}{$path} = $old->{load}{$path} || $self->new_img_from_file ($path); + $new->{load}{$path} = $old->{load}{$path} || $bgdsl_self->new_img_from_file ($path); } - sub root { + sub root() { + $new->{rootpmap_sensitive} = 1; die "root op not supported, exg, we need you"; } - sub resize { - $_[0]->scale ($_[1], $_[2]) + sub solid($;$$) { + my $img = $bgdsl_self->new_img (urxvt::PictStandardARGB32, $_[1] || 1, $_[2] || 1); + $img->fill ($_[0]); + $img } - sub move { - # TODO: must be simpler - $_[0]->transform ($_[0]->w, $_[0]->h, $_[1], - 1, 0, -$_[2], - 0, 1, -$_[3], - 0, 0, 1, - ) +=back + +=head2 VARIABLES + +=over 4 + +=cut + + sub X() { $new->{position_sensitive} = 1; $l } + sub Y() { $new->{position_sensitive} = 1; $t } + sub W() { $new->{size_sensitive} = 1; $w } + sub H() { $new->{size_sensitive} = 1; $h } + + sub now() { urxvt::NOW } + + sub again($) { + $new->{again} = $_[0]; } - sub rotate { - $_[0]->rotate ($_[0], $_[1], $_[2], $_[3] * (3.14159265 / 180)) + sub counter($) { + $new->{again} = $_[0]; + $bgdsl_self->{counter} + 0 } - sub blur { - my ($img, $rh, $rv) = @_; +=back - $img = $img->clone; - $img->clone->blur ($rh, $rv); +=head2 TILING MODES + +The following operators modify the tiling mode of an image, that is, the +way that pixels outside the image area are painted when the image is used. + +=over 4 + +=item tile $img + +Tiles the whole plane with the image and returns this new image - or in +other words, it returns a copy of the image in plane tiling mode. + +=item mirror $img + +Similar to tile, but reflects the image each time it uses a new copy, so +that top edges always touch top edges, right edges always touch right +edges and so on (with normal tiling, left edges always touch right edges +and top always touch bottom edges). + +=item pad $img + +Takes an image and modifies it so that all pixels outside the image area +become transparent. This mode is most useful when you want to place an +image over another image or the background colour while leaving all +background pixels outside the image unchanged. + +=item extend $img + +Extends the image over the whole plane, using the closest pixel in the +area outside the image. This mode is mostly useful when you more complex +filtering operations and want the pixels outside the image to have the +same values as the pixels near the edge. + +=cut + + sub pad($) { + my $img = $_[0]->clone; + $img->repeat_mode (urxvt::RepeatNone); $img } - sub contrast { - my ($img, $r, $g, $b, $a) = @_; + sub tile($) { + my $img = $_[0]->clone; + $img->repeat_mode (urxvt::RepeatNormal); + $img + } + + sub mirror($) { + my $img = $_[0]->clone; + $img->repeat_mode (urxvt::RepeatReflect); + $img + } + + sub extend($) { + my $img = $_[0]->clone; + $img->repeat_mode (urxvt::RepeatPad); + $img + } + +=back + +=head2 PIXEL OPERATORS + +The following operators modify the image pixels in various ways. + +=over 4 + +=item clone $img + +Returns an exact copy of the image. + +=cut + + sub clone($) { + $_[0]->clone + } + +=item clip $img + +=item clip $width, $height, $img + +=item clip $x, $y, $width, $height, $img + +Clips an image to the given rectangle. If the rectangle is outside the +image area (e.g. when C<$x> or C<$y> are negative) or the rectangle is +larger than the image, then the tiling mode defines how the extra pixels +will be filled. + +If C<$x> an C<$y> are missing, then C<0> is assumed for both. + +If C<$width> and C<$height> are missing, then the window size will be +assumed. + +Example: load an image, blur it, and clip it to the window size to save +memory. + + clip blur 10, load "mybg.png" + +=cut + + sub clip($;$$;$$) { + my $img = pop; + my $h = pop || H; + my $w = pop || W; + $img->sub_rect ($_[0], $_[1], $w, $h) + } + +=item scale $img + +=item scale $size_percent, $img + +=item scale $width_percent, $height_percent, $img + +Scales the image by the given percentages in horizontal +(C<$width_percent>) and vertical (C<$height_percent>) direction. + +If only one percentage is give, it is used for both directions. + +If no percentages are given, scales the image to the window size without +keeping aspect. + +=item resize $width, $height, $img + +Resizes the image to exactly C<$width> times C<$height> pixels. + +=cut + +#TODO: maximise, maximise_fill? + + sub scale($$$) { + my $img = pop; + + @_ == 2 ? $img->scale ($_[0] * $img->w * 0.01, $_[1] * $img->h * 0.01) + : @_ ? $img->scale ($_[0] * $img->w * 0.01, $_[0] * $img->h * 0.01) + : $img->scale (W, H) + } + + sub resize($$$) { + my $img = pop; + $img->scale ($_[0], $_[1]) + } + + # TODO: ugly + sub move($$;$) { + my $img = pop->clone; + $img->move ($_[0], $_[1]); + $img +# my $img = pop; +# $img->sub_rect ( +# $_[0], $_[1], +# $img->w, $img->h, +# $_[2], +# ) + } + + sub rotate($$$$$$) { + my $img = pop; + $img->rotate ( + $_[0], + $_[1], + $_[2] * $img->w * .01, + $_[3] * $img->h * .01, + $_[4] * (3.14159265 / 180), + ) + } + + sub blur($$;$) { + my $img = pop; + + $img->blur ($_[0], @_ >= 2 ? $_[1] : $_[0]); + } + + sub contrast($$;$$;$) { + my $img = pop; + my ($r, $g, $b, $a) = @_; + ($g, $b) = ($r, $r) if @_ < 4; $a = 1 if @_ < 5; + $img = $img->clone; $img->contrast ($r, $g, $b, $a); $img } - sub brightness { - my ($img, $r, $g, $b, $a) = @_; + sub brightness($$;$$;$) { + my $img = pop; + my ($r, $g, $b, $a) = @_; + ($g, $b) = ($r, $r) if @_ < 4; $a = 1 if @_ < 5; + $img = $img->clone; $img->brightness ($r, $g, $b, $a); $img } - sub x { $new->{position_sensitive} = 1; $x } - sub y { $new->{position_sensitive} = 1; $y } - sub w { $new->{size_sensitive} = 1; $w } - sub h { $new->{size_sensitive} = 1; $h } - sub now { urxvt::NOW } +=back - sub again { - $new->{again} = $_[0]; - } +=head2 SETTINGS - sub counter { - $new->{again} = $_[0]; - $self->{counter}++ + 0 +=over 4 + +=item border $respect_border=1 + +Sets whether the image should respect the terminal border (argument true +or missing), or whether it should fill the whole window (the default). + +By default, the image will cover the whole toplevel window. If C +is enabled, then it will only fill the character area and leave a normal +border in the background colour around it and behind the scrollbar. + +=cut + + sub border { + $border = @_ ? $_[0] : 1; } + +=back + +=cut + } sub parse_expr { @@ -99,7 +318,6 @@ sub set_expr { my ($self, $expr) = @_; - local $Data::Dumper::Deparse=1; use Data::Dumper; warn Dumper $expr;#d# $self->{expr} = $expr; $self->recalculate; } @@ -108,26 +326,86 @@ sub recalculate { my ($self) = @_; - local $urxvt::bgdsl::vars::self = $self; + # rate limit evaluation + + if ($self->{next_refresh} > urxvt::NOW) { + $self->{next_refresh_timer} = urxvt::timer->new->after ($self->{next_refresh} - urxvt::NOW)->cb (sub { + $self->recalculate; + }); + return; + } + + $self->{next_refresh} = urxvt::NOW + $MIN_INTERVAL; - local $urxvt::bgdsl::vars::old = $self->{state}; - local $urxvt::bgdsl::vars::new = my $state = $self->{state} = {}; + # set environment to evaluate user expression - ($urxvt::bgdsl::vars::x, $urxvt::bgdsl::vars::y, $urxvt::bgdsl::vars::w, $urxvt::bgdsl::vars::h) = + local $bgdsl_self = $self; + local $border; + + local $old = $self->{state}; + local $new = my $state = $self->{state} = {}; + + ($l, $t, $w, $h) = $self->get_geometry; + warn "$l,$t,$w,$h\n";#d# + + # evaluate user expression + my $img = eval { $self->{expr}->() }; warn $@ if $@;#d# + die if !UNIVERSAL::isa $img, "urxvt::img"; + + # if the expression is sensitive to external events, prepare reevaluation then + + my $repeat; if (my $again = $state->{again}) { - warn $again;#d# - $state->{again} = urxvt::timer->new->after ($again)->cb (sub { $self->recalculate }); + $repeat = 1; + $state->{timer} = $again == $old->{again} + ? $old->{timer} + : urxvt::timer->new->after ($again)->interval ($again)->cb (sub { + ++$self->{counter}; + $self->recalculate + }); + } + + if (delete $state->{position_sensitive}) { + $repeat = 1; + $self->enable (position_change => sub { $_[0]->recalculate }); + } else { + $self->disable ("position_change"); + } + + if (delete $state->{size_sensitive}) { + $repeat = 1; + $self->enable (size_change => sub { $_[0]->recalculate }); + } else { + $self->disable ("size_change"); } - # TODO: install handlers for geometry changes &c + if (delete $state->{rootpmap_sensitive}) { + $repeat = 1; + $self->enable (rootpmap_change => sub { $_[0]->recalculate }); + } else { + $self->disable ("rootpmap_change"); + } + + # clear stuff we no longer need + + %$old = (); + + unless ($repeat) { + delete $self->{state}; + delete $self->{expr}; + } + + # prepare and set background pixmap + + $img = $img->sub_rect (0, 0, $w, $h) + if $img->w != $w || $img->h != $h; - warn $img; - $self->set_background ($img); + $self->set_background ($img, $border); $self->scr_recolour (0); $self->want_refresh; }