use strict; use warnings; use Irssi; use Irssi::TextUI; # for sbar_items_redraw use Data::Dumper; our $VERSION = "0.1"; our %IRSSI = ( authors => "shabble", contact => 'shabble+irssi@metavore.org, shabble@#irssi/Freenode', name => "prompt_info", description => "Helper script for dynamically adding text " . "into the input-bar prompt.", license => "Public Domain", changed => "24/7/2010" ); sub DEBUG () { 1 } #sub DEBUG () { 0 } my $prompt_data = undef; my $prompt_item = undef; my $region_active = 0; my ($term_w, $term_h) = (0, 0); # visual region selected. my ($region_start, $region_end) = (0, 0); my $region_content = ''; my $prompt_format = ''; init(); sub update_terminal_size { my @stty_data = qx/stty -a/; my $line = $stty_data[0]; # linux # speed 38400 baud; rows 36; columns 126; line = 0; if ($line =~ m/rows (\d+); columns (\d+);/) { $term_h = $1; $term_w = $2; # osx # speed 9600 baud; 40 rows; 235 columns; } elsif ($line =~ m/(\d+) rows; (\d+) columns;/) { $term_h = $1; $term_w = $2; } else { # guess? $term_h = 24; $term_w = 80; } print "Terminal detected as $term_w cols by $term_h rows" if DEBUG; } sub prompt_subcmd_handler { my ($data, $server, $item) = @_; $data =~ s/\s+$//g; # strip trailing whitespace. Irssi::command_runsub('prompt', $data, $server, $item); } sub visual_subcmd_handler { my ($data, $server, $item) = @_; $data =~ s/\s+$//g; # strip trailing whitespace. Irssi::command_runsub('visual', $data, $server, $item); } sub init { Irssi::statusbar_item_register ('uberprompt', 0, 'uberprompt_draw'); Irssi::settings_add_str('uberprompt', 'uberprompt_format', '[$*] '); Irssi::command_bind("prompt", \&prompt_subcmd_handler); Irssi::command_bind('prompt on', \&replace_prompt_items); Irssi::command_bind('prompt off', \&restore_prompt_items); Irssi::command_bind('prompt set', sub { Irssi::signal_emit 'change prompt', shift; }); Irssi::command_bind('prompt clear', sub { Irssi::signal_emit 'change prompt', '$p'; }); # misc faff Irssi::command_bind('visual', \&visual_subcmd_handler); Irssi::command_bind('visual toggle', \&cmd_toggle_visual); Irssi::command_bind('visual clear', \&cmd_clear_visual); Irssi::command("^BIND ^F /visual toggle"); Irssi::command("^BIND ^G /visual clear"); Irssi::command_bind 'print_test', sub { Irssi::gui_printtext(0, 0, '%8hello there%n'); }; # redraw interception Irssi::signal_add_last('command redraw', \&augment_redraw); Irssi::signal_add_first('gui key pressed', \&ctrl_l_intercept); # for updating the overlay. Irssi::signal_add_last ('gui key pressed', \&key_pressed); # things to refresh the overlay for. Irssi::signal_add('window changed', \&uberprompt_refresh); Irssi::signal_add('window name changed', \&uberprompt_refresh); Irssi::signal_add('window changed automatic', \&uberprompt_refresh); Irssi::signal_add('window item changed', \&uberprompt_refresh); Irssi::signal_add('terminal resized', \&update_terminal_size); Irssi::signal_add('setup changed', \&reload_settings); # so we know where the bottom line is update_terminal_size(); # intialise the prompt format. reload_settings(); # install our statusbars. replace_prompt_items(); # the actual API signals. Irssi::signal_register({'change prompt' => [qw/string/]}); Irssi::signal_add('change prompt' => \&change_prompt_sig); Irssi::signal_register({'prompt changed' => [qw/string int/]}); } sub change_prompt_sig { my ($text) = @_; $text = '$p' . $text; print "Got prompt change sig with: $text" if DEBUG; my $changed; $changed = defined $prompt_data ? $prompt_data ne $text : 1; $prompt_data = $text; if ($changed) { print "Redrawing prompt" if DEBUG; uberprompt_refresh(); } } sub UNLOAD { # remove uberprompt and return the original ones. restore_prompt_items(); } sub reload_settings { my $new = Irssi::settings_get_str('uberprompt_format'); if ($prompt_format ne $new) { print "Updated prompt format" if DEBUG; $prompt_format = $new; Irssi::abstracts_register(['uberprompt', $prompt_format]); } } sub uberprompt_draw { my ($sb_item, $get_size_only) = @_; my $default_prompt = ''; my $window = Irssi::active_win; # hack to produce the same defaults as prompt/prompt_empty sbars. if (scalar( () = $window->items )) { $default_prompt = '{uberprompt $[.15]itemname}'; } else { $default_prompt = '{uberprompt $winname}'; } my $p_copy = $prompt_data; if (defined $prompt_data) { # replace the special marker '$p' with the original prompt. $p_copy =~ s/\$p/$default_prompt/; } else { $p_copy = $default_prompt; } print "Redrawing with: $p_copy, size-only: $get_size_only" if DEBUG; $prompt_item = $sb_item; my $ret = $sb_item->default_handler($get_size_only, $p_copy, '', 0); Irssi::signal_emit('prompt changed', $p_copy, $sb_item->{size}); return $ret; } sub augment_redraw { print "Redraw called" if DEBUG; uberprompt_refresh(); Irssi::timeout_add_once(10, \&refresh_visual_overlay, 0); } sub uberprompt_refresh { Irssi::statusbar_items_redraw('uberprompt'); } sub cmd_clear_visual { _clear_visual_region(); #refresh_visual_overlay(); Irssi::statusbar_items_redraw('input'); } sub cmd_toggle_visual { $region_active = not $region_active; if ($region_active) { $region_start = _pos(); $region_end = 0; # reset end marker. print "visual mode started at $region_start" if DEBUG; } else { $region_end = _pos(); print "Visual mode ended at $region_end" if DEBUG; if ($region_end > $region_start) { my $input = Irssi::parse_special('$L', 0, 0); my $str = substr($input, $region_start, $region_end - $region_start); print "Region selected: $str" if DEBUG; } else { print "Invalid region selection: [ $region_start - $region_end ]" if DEBUG; $region_start = $region_end = 0; } cmd_clear_visual(); } } sub ctrl_l_intercept { my $key = shift; if ($key == 12) { # C-l print "C-l pressed" if DEBUG; Irssi::command("redraw"); Irssi::signal_stop(); } elsif ($key == 10) { # RET _clear_visual_region(); } } sub key_pressed { # this handler needs to be last so the actual character is printed by irssi # before we overlay on it. Otherwise things are all a bit off-by-1 return unless $region_active; refresh_visual_overlay(); } sub _clear_visual_region { print "Clearing Region markers" if DEBUG; $region_end = 0; $region_start = 0; } sub refresh_visual_overlay { my $end_pos = $region_end; $end_pos ||= _pos(); # if not set, take current position as end. my $len = $end_pos - $region_start; return unless $len; # no point drawing an empty overlay my $input = Irssi::parse_special('$L'); my $offset = $prompt_item->{size} + $region_start; my $text = substr($input, $region_start, $len); print "printing '$text' at $offset [$region_start, $end_pos] ($len)" if DEBUG; $text = '%8' . $text . '%8'; _draw_overlay($offset, $text, $len); } sub _draw_overlay { my ($offset, $text, $len) = @_; Irssi::gui_printtext($offset, $term_h, $text); } sub replace_prompt_items { # remove existing ones. print "Removing original prompt" if DEBUG; _sbar_command('prompt', 'remove', 'prompt'); _sbar_command('prompt', 'remove', 'prompt_empty'); # add the new one. _sbar_command('prompt', 'add', 'uberprompt', qw/-alignment left -before input -priority '-1'/); _sbar_command('prompt', 'position', '100'); } sub restore_prompt_items { _sbar_command('prompt', 'remove', 'uberprompt'); print "Restoring original prompt" if DEBUG; _sbar_command('prompt', 'add', 'prompt', qw/-alignment left -before input -priority '-1'/); _sbar_command('prompt', 'add', 'prompt_empty', qw/-alignment left -after prompt -priority '-1'/); _sbar_command('prompt', 'position', '100'); } sub _sbar_command { my ($bar, $cmd, $item, @args) = @_; my $args_str = join ' ', @args; $args_str .= ' ' if length $args_str && defined $item; my $command = sprintf 'STATUSBAR %s %s %s%s', $bar, $cmd, $args_str, defined($item)?$item:''; print "Running command: $command" if DEBUG; Irssi::command($command); } sub _pos { return Irssi::gui_input_get_pos(); } # bit of fakery so things don't complain about the lack of prompt_info (hoepfully) %Irssi::Script::prompt_info:: = ();