# See http://wouter.coekaerts.be/site/irssi/mouse
# based on irssi mouse patch by mirage: http://darksun.com.pt/mirage/irssi/
# Copyright (C) 2005-2009 Wouter Coekaerts
#
# 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 qw(signal_emit settings_get_str active_win signal_stop settings_add_str settings_add_bool settings_get_bool signal_add signal_add_first);
use Math::Trig;
use vars qw($VERSION %IRSSI);
$VERSION = '1.1.2';
%IRSSI = (
authors => 'Wouter Coekaerts',
contact => '[email protected]',
name => 'mouse',
description => 'control irssi using mouse clicks and gestures',
license => 'GPLv2 or later',
url => 'http://wouter.coekaerts.be/irssi/',
changed => '2021-03-05',
);
my @BUTTONS = ('', '_middle', '_right');
my $mouse_xterm_status = -1; # -1:off 0,1,2:filling mouse_xterm_combo
my @mouse_xterm_combo = (3, 0, 0); # 0:button 1:x 2:y
my @mouse_xterm_previous; # previous contents of mouse_xterm_combo
sub mouse_enable {
print STDERR "\e[?1000h"; # start tracking
}
sub mouse_disable {
print STDERR "\e[?1000l"; # stop tracking
}
# Handle mouse event (button press or release)
sub mouse_event {
my ($b, $x, $y, $oldb, $oldx, $oldy) = @_;
my ($xd, $yd);
my ($distance, $angle);
# uhm, in the patch the scrollwheel didn't work for me, but this does:
if ($b == 64) {
cmd("mouse_scroll_up");
} elsif ($b == 65) {
cmd("mouse_scroll_down")
}
# proceed only if a button is being released
return if ($b != 3);
return unless (0 <= $oldb && $oldb <= 2);
my $button = $BUTTONS[$oldb];
# if it was a mouse click of the left button (press and release in the same position)
if ($x == $oldx && $y == $oldy) {
cmd("mouse" . $button . "_click");
return;
}
# otherwise, find mouse gestures
$xd = $x - $oldx;
$yd = -1 * ($y - $oldy);
$distance = sqrt($xd*$xd + $yd*$yd);
# ignore small gestures
if ($distance < 3) {
return;
}
$angle = asin($yd/$distance) * 180 / 3.14159265358979;
if ($angle < 20 && $angle > -20 && $xd > 0) {
if ($distance <= 40) {
cmd("mouse" . $button . "_gesture_right");
} else {
cmd("mouse" . $button . "_gesture_bigright");
}
} elsif ($angle < 20 && $angle > -20 && $xd < 0) {
if ($distance <= 40) {
cmd("mouse" . $button . "_gesture_left");
} else {
cmd("mouse" . $button . "_gesture_bigleft");
}
} elsif ($angle > 40) {
cmd("mouse" . $button . "_gesture_up");
} elsif ($angle < -40) {
cmd("mouse" . $button . "_gesture_down");
}
}
# executes the command configured in the given setting
sub cmd
{
my ($setting) = @_;
signal_emit("send command", settings_get_str($setting), active_win->{'active_server'}, active_win->{'active'});
}
signal_add_first("gui key pressed", sub {
my ($key) = @_;
if ($mouse_xterm_status != -1) {
if ($mouse_xterm_status == 0 && ($mouse_xterm_previous[0] != $mouse_xterm_combo[0])) { # if combo is starting, and previous what not a move (button not changed)
@mouse_xterm_previous = @mouse_xterm_combo;
}
$mouse_xterm_combo[$mouse_xterm_status] = $key-32;
$mouse_xterm_status++;
if ($mouse_xterm_status == 3) {
$mouse_xterm_status = -1;
# match screen coordinates
$mouse_xterm_combo[1]--;
$mouse_xterm_combo[2]--;
mouse_event($mouse_xterm_combo[0], $mouse_xterm_combo[1], $mouse_xterm_combo[2], $mouse_xterm_previous[0], $mouse_xterm_previous[1], $mouse_xterm_previous[2]);
}
signal_stop();
}
});
sub UNLOAD {
mouse_disable();
}
if ($ENV{"TERM"} !~ /^rxvt|screen|xterm|tmux(-(256)?(color|kitty))?$/) {
die "Your terminal doesn't seem to support this.";
}
mouse_enable();
Irssi::command("/^bind meta-[M /mouse_xterm"); # FIXME evil
Irssi::command_bind("mouse_xterm", sub {$mouse_xterm_status = 0;});
Irssi::command_bind 'mouse' => sub {
my ($data, $server, $item) = @_;
$data =~ s/\s+$//g;
Irssi::command_runsub('mouse', $data, $server, $item);
};
# temporarily disable mouse handling. Useful for copy-pasting without touching the keyboard (pressing shift)
Irssi::command_bind 'mouse tempdisable' => sub {
my ($data, $server, $item) = @_;
my $seconds = ($data eq '') ? 5 : $data; # optional argument saying how many seconds, defaulting to 5
mouse_disable();
Irssi::timeout_add_once($seconds * 1000, 'mouse_enable', undef); # turn back on after $second seconds
};
for my $button (@BUTTONS) {
settings_add_str("lookandfeel", "mouse" . $button . "_click", "/mouse tempdisable 5");
settings_add_str("lookandfeel", "mouse" . $button . "_gesture_up", "/window last");
settings_add_str("lookandfeel", "mouse" . $button . "_gesture_down", "/window goto active");
settings_add_str("lookandfeel", "mouse" . $button . "_gesture_left", "/window prev");
settings_add_str("lookandfeel", "mouse" . $button . "_gesture_bigleft", "/eval window prev;window prev");
settings_add_str("lookandfeel", "mouse" . $button . "_gesture_right", "/window next");
settings_add_str("lookandfeel", "mouse" . $button . "_gesture_bigright", "/eval window next;window next");
}
settings_add_str("lookandfeel", "mouse_scroll_up", "/scrollback goto -10");
settings_add_str("lookandfeel", "mouse_scroll_down", "/scrollback goto +10");