scripts/history_search.pl
# Search within your typed history as you type (like ctrl-R in bash)
# Usage:
# * First do: /bind ^R /history_search
# * Then type ctrl-R and type what you're searching for
# * Optionally, you can bind something to "/history_search -forward" to go forward in the results
# Copyright 2007-2009 Wouter Coekaerts <coekie@irssi.org>
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
use strict;
use Irssi 20070804;
use Irssi::TextUI;
use vars qw($VERSION %IRSSI);
$VERSION = '2.0';
%IRSSI = (
authors => 'Wouter Coekaerts',
contact => 'coekie@irssi.org',
name => 'history_search',
description => 'Search within your typed history as you type (like ctrl-R in bash)',
license => 'GPLv2 or later',
url => 'http://wouter.coekaerts.be/irssi/',
changed => '17/01/09'
);
# is the searching enabled?
my $enabled = 0;
# the typed text (the query) last time a key was pressed
my $prev_typed;
# the position in the input of where the typed text started.
# everything before it is not typed by the user but added by this script as part of the result
my $prev_startpos;
# the current list of matches
my @matches;
# at what place are we in @matches?
my $current_match_index;
Irssi::command_bind('history_search', sub {
my ($data, $server, $item) = @_;
if ($data !~ /^ *(-forward)? *$/) {
Irssi::print("history_search: Unknown arguments: $data");
return;
}
my $forward = $1 eq '-forward';
if (! $enabled) {
$enabled = 1;
$prev_typed = '';
$prev_startpos = 0;
@matches = ();
$current_match_index = -1;
} else {
if ($forward) {
if ($current_match_index + 1 < scalar(@matches)) {
$current_match_index++;
}
} else { # backwards
if ($current_match_index > 0) {
$current_match_index--;
}
}
}
});
Irssi::signal_add_last 'gui key pressed' => sub {
my ($key) = @_;
if ($key == 10 || $key == 27) { # enter or escape
$enabled = 0;
}
return unless $enabled;
# get the content of the input line
my $prompt = Irssi::parse_special('$L');
my $pos = Irssi::gui_input_get_pos();
# stop if the cursor is before the position where the typing started (e.g. if user pressed backspace more than he typed characters)
if ($pos < $prev_startpos) {
$enabled = 0;
return;
}
# get the part of the input line that the user typed (strip the part before and after which this script added)
my $typed = substr($prompt, $prev_startpos, ($pos-$prev_startpos));
if ($typed ne $prev_typed) { # something changed
# find matches
find_matches($typed);
# start searching from the end again
$current_match_index = scalar(@matches) - 1;
}
# if nothing was found, just show what the user typed
# else, show the current match
my $result = ($current_match_index == -1) ? $typed : $matches[$current_match_index];
# update the input line
my $startpos = index(lc($result), lc($typed));
Irssi::gui_input_set($result);
Irssi::gui_input_set_pos($startpos + length($typed));
# remember for next time
$prev_typed = $typed;
$prev_startpos = $startpos;
};
# find matches for the given user-typed text, and put it in @matches
sub find_matches($) {
my ($typed) = @_;
if (Irssi::version() > 20090117) {
$typed = lc($typed);
my @history;
if ($prev_typed ne '' && index($typed, lc($prev_typed)) != -1) { # previous typed plus more
@history = @matches; # only search in previous results
} else {
@history = Irssi::active_win->get_history_lines();
}
@matches = ();
for my $history_line (@history) {
my $startpos = index(lc($history_line), $typed);
if ($startpos != -1) {
push @matches, $history_line;
}
}
} else { # older irssi version, can only get the last match
@matches = ();
my $last_match = Irssi::parse_special('$!' . $typed . '!');
if ($last_match ne '') {
push @matches, $last_match;
}
}
}