summaryrefslogtreecommitdiff
path: root/.local
diff options
context:
space:
mode:
authorA Farzat <a@farzat.xyz>2024-10-20 19:54:46 +0900
committerA Farzat <a@farzat.xyz>2024-11-02 19:54:46 +0900
commita9a8ba18260383725ff30c1e4e9e8c5f8591e5ad (patch)
treedf2a45f0e7c4456a302dbceb258c27b98a86c7a3 /.local
parent6273242562b467a64ef04088809f03d9ab77f2a6 (diff)
downloaddotfiles-a9a8ba18260383725ff30c1e4e9e8c5f8591e5ad.tar.gz
dotfiles-a9a8ba18260383725ff30c1e4e9e8c5f8591e5ad.zip
Add weechat plugins
Diffstat (limited to '.local')
-rw-r--r--.local/share/weechat/.gitignore2
l---------.local/share/weechat/python/autoload/bufsize.py1
l---------.local/share/weechat/python/autoload/edit.py1
l---------.local/share/weechat/python/autoload/go.py1
l---------.local/share/weechat/python/autoload/notify_send.py1
-rw-r--r--.local/share/weechat/python/bufsize.py247
-rw-r--r--.local/share/weechat/python/go.py572
7 files changed, 825 insertions, 0 deletions
diff --git a/.local/share/weechat/.gitignore b/.local/share/weechat/.gitignore
new file mode 100644
index 0000000..0308619
--- /dev/null
+++ b/.local/share/weechat/.gitignore
@@ -0,0 +1,2 @@
+matrix/
+matrix-rust/
diff --git a/.local/share/weechat/python/autoload/bufsize.py b/.local/share/weechat/python/autoload/bufsize.py
new file mode 120000
index 0000000..ff72a18
--- /dev/null
+++ b/.local/share/weechat/python/autoload/bufsize.py
@@ -0,0 +1 @@
+../bufsize.py \ No newline at end of file
diff --git a/.local/share/weechat/python/autoload/edit.py b/.local/share/weechat/python/autoload/edit.py
new file mode 120000
index 0000000..fcdd325
--- /dev/null
+++ b/.local/share/weechat/python/autoload/edit.py
@@ -0,0 +1 @@
+/usr/share/weechat/python/edit.py \ No newline at end of file
diff --git a/.local/share/weechat/python/autoload/go.py b/.local/share/weechat/python/autoload/go.py
new file mode 120000
index 0000000..bdfb7dd
--- /dev/null
+++ b/.local/share/weechat/python/autoload/go.py
@@ -0,0 +1 @@
+../go.py \ No newline at end of file
diff --git a/.local/share/weechat/python/autoload/notify_send.py b/.local/share/weechat/python/autoload/notify_send.py
new file mode 120000
index 0000000..0d7dd7b
--- /dev/null
+++ b/.local/share/weechat/python/autoload/notify_send.py
@@ -0,0 +1 @@
+/usr/share/weechat/python/notify_send.py \ No newline at end of file
diff --git a/.local/share/weechat/python/bufsize.py b/.local/share/weechat/python/bufsize.py
new file mode 100644
index 0000000..153c9bc
--- /dev/null
+++ b/.local/share/weechat/python/bufsize.py
@@ -0,0 +1,247 @@
+# -*- coding: utf-8 -*-
+#
+# Copyright (c) 2012-2017 by nils_2 <weechatter@arcor.de>
+# and nesthib <nesthib@gmail.com>
+#
+# scroll indicator; displaying number of lines below last line, overall lines in buffer, number of current line and percent displayed
+#
+# 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 3 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, see <http://www.gnu.org/licenses/>.
+#
+# 2017-08-17: nils_2 (freenode.#weechat)
+# 0.8: add support for buffer_filters_enabled and buffer_filters_disabled (WeeChat ≥ 2.0)
+# 2016-12-16: nils_2 (freenode.#weechat)
+# 0.7: add option show_scroll (idea by earnestly)
+# 2016-04-23: wdbw <tuturu@tutanota.com>
+# 0.6.2 : fix: type of filters_enabled
+# 2014-02-24: nesthib (freenode.#weechat)
+# 0.6.1 : fix: color tags for default format
+# 2013-11-19: nils_2 (freenode.#weechat)
+# 0.6 : fix: stdout/stderr warning
+# 2013-11-02: nils_2 (freenode.#weechat)
+# 0.5 : fix refresh on (un)zoomed buffer
+# : add option 'count_filtered_lines' and format item "%F"
+# 2013-10-15: nils_2 (freenode.#weechat)
+# 0.4 : fix bug with root-bar
+# : add support of eval_expression (weechat >= 0.4.2)
+# 2013-01-25: nils_2 (freenode.#weechat)
+# 0.3 : make script compatible with Python 3.x
+# : internal changes
+# 2012-07-09: nils_2 (freenode.#weechat)
+# 0.2 : fix: display bug with more than one window
+# : hide item when buffer empty
+# 2012-07-08: obiwahn
+# 0.1.1 : add hook for switch_buffer
+# 2012-01-11: nils_2, nesthib (freenode.#weechat)
+# 0.1 : initial release
+#
+# Development is currently hosted at
+# https://github.com/weechatter/weechat-scripts
+
+try:
+ import weechat, re
+
+except Exception:
+ print("This script must be run under WeeChat.")
+ print("Get WeeChat now at: http://www.weechat.org/")
+ quit()
+
+SCRIPT_NAME = "bufsize"
+SCRIPT_AUTHOR = "nils_2 <weechatter@arcor.de>"
+SCRIPT_VERSION = "0.8"
+SCRIPT_LICENSE = "GPL"
+SCRIPT_DESC = "scroll indicator; displaying number of lines below last line, overall lines in buffer, number of current line and percent displayed"
+
+OPTIONS = { 'format' : ('${color:yellow}%P${color:default}⋅%{${color:yellow}%A${color:default}⇵${color:yellow}%C${color:default}/}${color:yellow}%L',
+ 'format for items to display in bar, possible items: %P = percent indicator, %A = number of lines below last line, %L = lines counter, %C = current line %F = number of filtered lines (note: using WeeChat >= 0.4.2, content is evaluated, so you can use colors with format \"${color:xxx}\", see /help eval)'),
+ 'count_filtered_lines': ('on',
+ 'filtered lines will be count in item.'),
+ 'show_scroll':('on','always show the scroll indicator number,even if its 0 (item %A), if option is off the scroll indicator will be hidden like the item "scroll"'),
+ }
+# ================================[ weechat item ]===============================
+# regexp to match ${color} tags
+regex_color=re.compile('\$\{([^\{\}]+)\}')
+
+# regexp to match ${optional string} tags
+regex_optional_tags=re.compile('%\{[^\{\}]+\}')
+
+filter_status = 0
+
+def show_item (data, item, window):
+ # check for root input bar!
+ if not window:
+ window = weechat.current_window()
+
+ ptr_buffer = weechat.window_get_pointer(window,'buffer')
+ if ptr_buffer == '':
+ return ''
+
+ if weechat.buffer_get_string(ptr_buffer,'name') != 'weechat': # not weechat core buffer
+ if weechat.buffer_get_string(ptr_buffer,'localvar_type') == '': # buffer with free content?
+ return ''
+
+ lines_after, lines_count, percent, current_line, filtered, filtered_before, filtered_after = count_lines(window,ptr_buffer)
+ lines_after_bak = lines_after
+
+ if lines_count == 0: # buffer empty?
+ return ''
+
+ if filtered == 0:
+ filtered = ''
+
+ if lines_after == 0 and (OPTIONS['show_scroll'].lower() == 'off'):
+ lines_after = ''
+
+ tags = {'%C': str(current_line),
+ '%A': str(lines_after),
+ '%F': str(filtered),
+ '%L': str(lines_count),
+ '%P': str(percent)+'%'}
+
+ bufsize_item = substitute_colors(OPTIONS['format'])
+
+ # replace mandatory tags
+ for tag in list(tags.keys()):
+# for tag in tags.keys():
+ bufsize_item = bufsize_item.replace(tag, tags[tag])
+
+ # replace optional tags
+ # %{…} only if lines after (e.g. %A > 0)
+ if lines_after_bak > 0:
+ for regex_tag in regex_optional_tags.findall(bufsize_item):
+ bufsize_item = bufsize_item.replace(regex_tag, regex_tag.lstrip('%{').rstrip('}'))
+ else:
+ bufsize_item = regex_optional_tags.sub('', bufsize_item)
+
+ return bufsize_item
+
+def substitute_colors(text):
+ if int(version) >= 0x00040200:
+ return weechat.string_eval_expression(text,{},{},{})
+ # substitute colors in output
+ return re.sub(regex_color, lambda match: weechat.color(match.group(1)), text)
+
+def count_lines(ptr_window,ptr_buffer):
+ global filter_status
+
+ hdata_buf = weechat.hdata_get('buffer')
+ hdata_lines = weechat.hdata_get('lines')
+ lines = weechat.hdata_pointer(hdata_buf, ptr_buffer, 'lines') # own_lines, mixed_lines
+ lines_count = weechat.hdata_integer(hdata_lines, lines, 'lines_count')
+
+ hdata_window = weechat.hdata_get('window')
+ hdata_winscroll = weechat.hdata_get('window_scroll')
+ window_scroll = weechat.hdata_pointer(hdata_window, ptr_window, 'scroll')
+ lines_after = weechat.hdata_integer(hdata_winscroll, window_scroll, 'lines_after')
+ window_height = weechat.window_get_integer(weechat.current_window(), 'win_chat_height')
+
+ filtered = 0
+ filtered_before = 0
+ filtered_after = 0
+ # if filter is disabled, don't count.
+ if (OPTIONS['count_filtered_lines'].lower() == 'off') and filter_status == 1:
+ filtered, filtered_before,filtered_after = count_filtered_lines(ptr_buffer,lines_count,lines_after)
+ lines_count = lines_count - filtered
+# lines_after = lines_after - filtered_after
+
+ if lines_count > window_height:
+ differential = lines_count - window_height
+ percent = max(int(round(100. * (differential - lines_after) / differential)), 0)
+ else:
+ percent = 100
+
+ # get current position
+ current_line = lines_count - lines_after
+
+ return lines_after,lines_count,percent,current_line, filtered, filtered_before, filtered_after
+
+def count_filtered_lines(ptr_buffer,lines_count,lines_after):
+ filtered_before = 0
+ filtered_after = 0
+ filtered = 0
+
+ lines = weechat.hdata_pointer(weechat.hdata_get('buffer'), ptr_buffer, 'own_lines')
+ counter = 0
+ current_position = lines_count - lines_after
+
+ if lines:
+ line = weechat.hdata_pointer(weechat.hdata_get('lines'), lines, 'first_line')
+ hdata_line = weechat.hdata_get('line')
+ hdata_line_data = weechat.hdata_get('line_data')
+
+ while line:
+ data = weechat.hdata_pointer(hdata_line, line, 'data')
+ if data:
+# message = weechat.hdata_string(hdata_line_data, data, 'message')
+ displayed = weechat.hdata_char(hdata_line_data, data, 'displayed')
+ if displayed == 0:
+# weechat.prnt('','%d - %s - %s' % (counter, displayed, message))
+ if counter < current_position:
+ filtered_before += 1
+ else:
+ filtered_after += 1
+ counter += 1
+ line = weechat.hdata_move(hdata_line, line, 1)
+
+ filtered = filtered_before + filtered_after
+ return filtered,filtered_before,filtered_after
+
+def update_cb(data, signal, signal_data):
+ weechat.bar_item_update(SCRIPT_NAME)
+ return weechat.WEECHAT_RC_OK
+
+def filtered_update_cb(data, signal, signal_data):
+ global filter_status
+ if signal == 'filters_disabled':
+ filter_status = 0
+ if signal == 'filters_enabled':
+ filter_status = 1
+ weechat.bar_item_update(SCRIPT_NAME)
+ return weechat.WEECHAT_RC_OK
+# ================================[ weechat options and description ]===============================
+def init_options():
+ for option,value in OPTIONS.items():
+ if not weechat.config_is_set_plugin(option):
+ weechat.config_set_plugin(option, value[0])
+ OPTIONS[option] = value[0]
+ else:
+ OPTIONS[option] = weechat.config_get_plugin(option)
+ weechat.config_set_desc_plugin(option, "%s (default: '%s')" % (value[1], value[0]))
+
+def toggle_refresh(pointer, name, value):
+ global OPTIONS
+ option = name[len('plugins.var.python.' + SCRIPT_NAME + '.'):] # get optionname
+ OPTIONS[option] = value # save new value
+ weechat.bar_item_update(SCRIPT_NAME)
+ return weechat.WEECHAT_RC_OK
+# ================================[ main ]===============================
+if __name__ == "__main__":
+# global filter_status
+ if weechat.register(SCRIPT_NAME, SCRIPT_AUTHOR, SCRIPT_VERSION, SCRIPT_LICENSE, SCRIPT_DESC, '', ''):
+ version = weechat.info_get("version_number", "") or 0
+
+ if int(version) >= 0x00030600:
+ filter_status = int(weechat.info_get('filters_enabled',''))
+ bar_item = weechat.bar_item_new(SCRIPT_NAME, 'show_item','')
+ weechat.bar_item_update(SCRIPT_NAME)
+ weechat.hook_signal('buffer_line_added','update_cb','')
+ weechat.hook_signal('window_scrolled','update_cb','')
+ weechat.hook_signal('buffer_switch','update_cb','')
+ weechat.hook_signal('*filters*','filtered_update_cb','')
+ weechat.hook_command_run('/buffer clear*','update_cb','')
+ weechat.hook_command_run('/window page*','update_cb','')
+ weechat.hook_command_run('/input zoom_merged_buffer','update_cb','')
+ weechat.hook_config( 'plugins.var.python.' + SCRIPT_NAME + '.*', 'toggle_refresh', '' )
+ init_options()
+ else:
+ weechat.prnt('','%s%s %s' % (weechat.prefix('error'),SCRIPT_NAME,': needs version 0.3.6 or higher'))
diff --git a/.local/share/weechat/python/go.py b/.local/share/weechat/python/go.py
new file mode 100644
index 0000000..77c5774
--- /dev/null
+++ b/.local/share/weechat/python/go.py
@@ -0,0 +1,572 @@
+# -*- coding: utf-8 -*-
+#
+# Copyright (C) 2009-2014 Sébastien Helleu <flashcode@flashtux.org>
+# Copyright (C) 2010 m4v <lambdae2@gmail.com>
+# Copyright (C) 2011 stfn <stfnmd@googlemail.com>
+#
+# 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 3 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, see <http://www.gnu.org/licenses/>.
+#
+
+#
+# History:
+#
+# 2021-05-25, Tomáš Janoušek <tomi@nomi.cz>:
+# version 2.7: add new option to prefix short names with server names
+# 2019-07-11, Simmo Saan <simmo.saan@gmail.com>
+# version 2.6: fix detection of "/input search_text_here"
+# 2017-04-01, Sébastien Helleu <flashcode@flashtux.org>:
+# version 2.5: add option "buffer_number"
+# 2017-03-02, Sébastien Helleu <flashcode@flashtux.org>:
+# version 2.4: fix syntax and indentation error
+# 2017-02-25, Simmo Saan <simmo.saan@gmail.com>
+# version 2.3: fix fuzzy search breaking buffer number search display
+# 2016-01-28, ylambda <ylambda@koalabeast.com>
+# version 2.2: add option "fuzzy_search"
+# 2015-11-12, nils_2 <weechatter@arcor.de>
+# version 2.1: fix problem with buffer short_name "weechat", using option
+# "use_core_instead_weechat", see:
+# https://github.com/weechat/weechat/issues/574
+# 2014-05-12, Sébastien Helleu <flashcode@flashtux.org>:
+# version 2.0: add help on options, replace option "sort_by_activity" by
+# "sort" (add sort by name and first match at beginning of
+# name and by number), PEP8 compliance
+# 2012-11-26, Nei <anti.teamidiot.de>
+# version 1.9: add auto_jump option to automatically go to buffer when it
+# is uniquely selected
+# 2012-09-17, Sébastien Helleu <flashcode@flashtux.org>:
+# version 1.8: fix jump to non-active merged buffers (jump with buffer name
+# instead of number)
+# 2012-01-03 nils_2 <weechatter@arcor.de>
+# version 1.7: add option "use_core_instead_weechat"
+# 2012-01-03, Sébastien Helleu <flashcode@flashtux.org>:
+# version 1.6: make script compatible with Python 3.x
+# 2011-08-24, stfn <stfnmd@googlemail.com>:
+# version 1.5: /go with name argument jumps directly to buffer
+# Remember cursor position in buffer input
+# 2011-05-31, Elián Hanisch <lambdae2@gmail.com>:
+# version 1.4: Sort list of buffers by activity.
+# 2011-04-25, Sébastien Helleu <flashcode@flashtux.org>:
+# version 1.3: add info "go_running" (used by script input_lock.rb)
+# 2010-11-01, Sébastien Helleu <flashcode@flashtux.org>:
+# version 1.2: use high priority for hooks to prevent conflict with other
+# plugins/scripts (WeeChat >= 0.3.4 only)
+# 2010-03-25, Elián Hanisch <lambdae2@gmail.com>:
+# version 1.1: use a space to match the end of a string
+# 2009-11-16, Sébastien Helleu <flashcode@flashtux.org>:
+# version 1.0: add new option to display short names
+# 2009-06-15, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.9: fix typo in /help go with command /key
+# 2009-05-16, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.8: search buffer by number, fix bug when window is split
+# 2009-05-03, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.7: eat tab key (do not complete input, just move buffer
+# pointer)
+# 2009-05-02, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.6: sync with last API changes
+# 2009-03-22, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.5: update modifier signal name for input text display,
+# fix arguments for function string_remove_color
+# 2009-02-18, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.4: do not hook command and init options if register failed
+# 2009-02-08, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.3: case insensitive search for buffers names
+# 2009-02-08, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.2: add help about Tab key
+# 2009-02-08, Sébastien Helleu <flashcode@flashtux.org>:
+# version 0.1: initial release
+#
+
+"""
+Quick jump to buffers.
+(this script requires WeeChat 0.3.0 or newer)
+"""
+
+from __future__ import print_function
+
+SCRIPT_NAME = 'go'
+SCRIPT_AUTHOR = 'Sébastien Helleu <flashcode@flashtux.org>'
+SCRIPT_VERSION = '2.7'
+SCRIPT_LICENSE = 'GPL3'
+SCRIPT_DESC = 'Quick jump to buffers'
+
+SCRIPT_COMMAND = 'go'
+
+IMPORT_OK = True
+
+try:
+ import weechat
+except ImportError:
+ print('This script must be run under WeeChat.')
+ print('Get WeeChat now at: http://www.weechat.org/')
+ IMPORT_OK = False
+
+import re
+
+# script options
+SETTINGS = {
+ 'color_number': (
+ 'yellow,magenta',
+ 'color for buffer number (not selected)'),
+ 'color_number_selected': (
+ 'yellow,red',
+ 'color for selected buffer number'),
+ 'color_name': (
+ 'black,cyan',
+ 'color for buffer name (not selected)'),
+ 'color_name_selected': (
+ 'black,brown',
+ 'color for a selected buffer name'),
+ 'color_name_highlight': (
+ 'red,cyan',
+ 'color for highlight in buffer name (not selected)'),
+ 'color_name_highlight_selected': (
+ 'red,brown',
+ 'color for highlight in a selected buffer name'),
+ 'message': (
+ 'Go to: ',
+ 'message to display before list of buffers'),
+ 'short_name': (
+ 'off',
+ 'display and search in short names instead of buffer name'),
+ 'short_name_server': (
+ 'off',
+ 'prefix short names with server names for search and display'),
+ 'sort': (
+ 'number,beginning',
+ 'comma-separated list of keys to sort buffers '
+ '(the order is important, sorts are performed in the given order): '
+ 'name = sort by name (or short name), ',
+ 'hotlist = sort by hotlist order, '
+ 'number = first match a buffer number before digits in name, '
+ 'beginning = first match at beginning of names (or short names); '
+ 'the default sort of buffers is by numbers'),
+ 'use_core_instead_weechat': (
+ 'off',
+ 'use name "core" instead of "weechat" for core buffer'),
+ 'auto_jump': (
+ 'off',
+ 'automatically jump to buffer when it is uniquely selected'),
+ 'fuzzy_search': (
+ 'off',
+ 'search buffer matches using approximation'),
+ 'buffer_number': (
+ 'on',
+ 'display buffer number'),
+}
+
+# hooks management
+HOOK_COMMAND_RUN = {
+ 'input': ('/input *', 'go_command_run_input'),
+ 'buffer': ('/buffer *', 'go_command_run_buffer'),
+ 'window': ('/window *', 'go_command_run_window'),
+}
+hooks = {}
+
+# input before command /go (we'll restore it later)
+saved_input = ''
+saved_input_pos = 0
+
+# last user input (if changed, we'll update list of matching buffers)
+old_input = None
+
+# matching buffers
+buffers = []
+buffers_pos = 0
+
+
+def go_option_enabled(option):
+ """Checks if a boolean script option is enabled or not."""
+ return weechat.config_string_to_boolean(weechat.config_get_plugin(option))
+
+
+def go_info_running(data, info_name, arguments):
+ """Returns "1" if go is running, otherwise "0"."""
+ return '1' if 'modifier' in hooks else '0'
+
+
+def go_unhook_one(hook):
+ """Unhook something hooked by this script."""
+ global hooks
+ if hook in hooks:
+ weechat.unhook(hooks[hook])
+ del hooks[hook]
+
+
+def go_unhook_all():
+ """Unhook all."""
+ go_unhook_one('modifier')
+ for hook in HOOK_COMMAND_RUN:
+ go_unhook_one(hook)
+
+
+def go_hook_all():
+ """Hook command_run and modifier."""
+ global hooks
+ priority = ''
+ version = weechat.info_get('version_number', '') or 0
+ # use high priority for hook to prevent conflict with other plugins/scripts
+ # (WeeChat >= 0.3.4 only)
+ if int(version) >= 0x00030400:
+ priority = '2000|'
+ for hook, value in HOOK_COMMAND_RUN.items():
+ if hook not in hooks:
+ hooks[hook] = weechat.hook_command_run(
+ '%s%s' % (priority, value[0]),
+ value[1], '')
+ if 'modifier' not in hooks:
+ hooks['modifier'] = weechat.hook_modifier(
+ 'input_text_display_with_cursor', 'go_input_modifier', '')
+
+
+def go_start(buf):
+ """Start go on buffer."""
+ global saved_input, saved_input_pos, old_input, buffers_pos
+ go_hook_all()
+ saved_input = weechat.buffer_get_string(buf, 'input')
+ saved_input_pos = weechat.buffer_get_integer(buf, 'input_pos')
+ weechat.buffer_set(buf, 'input', '')
+ old_input = None
+ buffers_pos = 0
+
+
+def go_end(buf):
+ """End go on buffer."""
+ global saved_input, saved_input_pos, old_input
+ go_unhook_all()
+ weechat.buffer_set(buf, 'input', saved_input)
+ weechat.buffer_set(buf, 'input_pos', str(saved_input_pos))
+ old_input = None
+
+
+def go_match_beginning(buf, string):
+ """Check if a string matches the beginning of buffer name/short name."""
+ if not string:
+ return False
+ esc_str = re.escape(string)
+ if re.search(r'^#?' + esc_str, buf['name']) \
+ or re.search(r'^#?' + esc_str, buf['short_name']):
+ return True
+ return False
+
+
+def go_match_fuzzy(name, string):
+ """Check if string matches name using approximation."""
+ if not string:
+ return False
+
+ name_len = len(name)
+ string_len = len(string)
+
+ if string_len > name_len:
+ return False
+ if name_len == string_len:
+ return name == string
+
+ # Attempt to match all chars somewhere in name
+ prev_index = -1
+ for i, char in enumerate(string):
+ index = name.find(char, prev_index+1)
+ if index == -1:
+ return False
+ prev_index = index
+ return True
+
+
+def go_now(buf, args):
+ """Go to buffer specified by args."""
+ listbuf = go_matching_buffers(args)
+ if not listbuf:
+ return
+
+ # prefer buffer that matches at beginning (if option is enabled)
+ if 'beginning' in weechat.config_get_plugin('sort').split(','):
+ for index in range(len(listbuf)):
+ if go_match_beginning(listbuf[index], args):
+ weechat.command(buf,
+ '/buffer ' + str(listbuf[index]['full_name']))
+ return
+
+ # jump to first buffer in matching buffers by default
+ weechat.command(buf, '/buffer ' + str(listbuf[0]['full_name']))
+
+
+def go_cmd(data, buf, args):
+ """Command "/go": just hook what we need."""
+ global hooks
+ if args:
+ go_now(buf, args)
+ elif 'modifier' in hooks:
+ go_end(buf)
+ else:
+ go_start(buf)
+ return weechat.WEECHAT_RC_OK
+
+
+def go_matching_buffers(strinput):
+ """Return a list with buffers matching user input."""
+ global buffers_pos
+ listbuf = []
+ if len(strinput) == 0:
+ buffers_pos = 0
+ strinput = strinput.lower()
+ infolist = weechat.infolist_get('buffer', '', '')
+ while weechat.infolist_next(infolist):
+ pointer = weechat.infolist_pointer(infolist, 'pointer')
+ short_name = weechat.infolist_string(infolist, 'short_name')
+ server = weechat.buffer_get_string(pointer, 'localvar_server')
+ if go_option_enabled('short_name'):
+ if go_option_enabled('short_name_server') and server:
+ name = server + '.' + short_name
+ else:
+ name = short_name
+ else:
+ name = weechat.infolist_string(infolist, 'name')
+ if name == 'weechat' \
+ and go_option_enabled('use_core_instead_weechat') \
+ and weechat.infolist_string(infolist, 'plugin_name') == 'core':
+ name = 'core'
+ number = weechat.infolist_integer(infolist, 'number')
+ full_name = weechat.infolist_string(infolist, 'full_name')
+ if not full_name:
+ full_name = '%s.%s' % (
+ weechat.infolist_string(infolist, 'plugin_name'),
+ weechat.infolist_string(infolist, 'name'))
+ matching = name.lower().find(strinput) >= 0
+ if not matching and strinput[-1] == ' ':
+ matching = name.lower().endswith(strinput.strip())
+ if not matching and go_option_enabled('fuzzy_search'):
+ matching = go_match_fuzzy(name.lower(), strinput)
+ if not matching and strinput.isdigit():
+ matching = str(number).startswith(strinput)
+ if len(strinput) == 0 or matching:
+ listbuf.append({
+ 'number': number,
+ 'short_name': short_name,
+ 'name': name,
+ 'full_name': full_name,
+ 'pointer': pointer,
+ })
+ weechat.infolist_free(infolist)
+
+ # sort buffers
+ hotlist = []
+ infolist = weechat.infolist_get('hotlist', '', '')
+ while weechat.infolist_next(infolist):
+ hotlist.append(
+ weechat.infolist_pointer(infolist, 'buffer_pointer'))
+ weechat.infolist_free(infolist)
+ last_index_hotlist = len(hotlist)
+
+ def _sort_name(buf):
+ """Sort buffers by name (or short name)."""
+ return buf['name']
+
+ def _sort_hotlist(buf):
+ """Sort buffers by hotlist order."""
+ try:
+ return hotlist.index(buf['pointer'])
+ except ValueError:
+ # not in hotlist, always last.
+ return last_index_hotlist
+
+ def _sort_match_number(buf):
+ """Sort buffers by match on number."""
+ return 0 if str(buf['number']) == strinput else 1
+
+ def _sort_match_beginning(buf):
+ """Sort buffers by match at beginning."""
+ return 0 if go_match_beginning(buf, strinput) else 1
+
+ funcs = {
+ 'name': _sort_name,
+ 'hotlist': _sort_hotlist,
+ 'number': _sort_match_number,
+ 'beginning': _sort_match_beginning,
+ }
+
+ for key in weechat.config_get_plugin('sort').split(','):
+ if key in funcs:
+ listbuf = sorted(listbuf, key=funcs[key])
+
+ if not strinput:
+ index = [i for i, buf in enumerate(listbuf)
+ if buf['pointer'] == weechat.current_buffer()]
+ if index:
+ buffers_pos = index[0]
+
+ return listbuf
+
+
+def go_buffers_to_string(listbuf, pos, strinput):
+ """Return string built with list of buffers found (matching user input)."""
+ string = ''
+ strinput = strinput.lower()
+ for i in range(len(listbuf)):
+ selected = '_selected' if i == pos else ''
+ buffer_name = listbuf[i]['name']
+ index = buffer_name.lower().find(strinput)
+ if index >= 0:
+ index2 = index + len(strinput)
+ name = '%s%s%s%s%s' % (
+ buffer_name[:index],
+ weechat.color(weechat.config_get_plugin(
+ 'color_name_highlight' + selected)),
+ buffer_name[index:index2],
+ weechat.color(weechat.config_get_plugin(
+ 'color_name' + selected)),
+ buffer_name[index2:])
+ elif go_option_enabled("fuzzy_search") and \
+ go_match_fuzzy(buffer_name.lower(), strinput):
+ name = ""
+ prev_index = -1
+ for char in strinput.lower():
+ index = buffer_name.lower().find(char, prev_index+1)
+ if prev_index < 0:
+ name += buffer_name[:index]
+ name += weechat.color(weechat.config_get_plugin(
+ 'color_name_highlight' + selected))
+ if prev_index >= 0 and index > prev_index+1:
+ name += weechat.color(weechat.config_get_plugin(
+ 'color_name' + selected))
+ name += buffer_name[prev_index+1:index]
+ name += weechat.color(weechat.config_get_plugin(
+ 'color_name_highlight' + selected))
+ name += buffer_name[index]
+ prev_index = index
+
+ name += weechat.color(weechat.config_get_plugin(
+ 'color_name' + selected))
+ name += buffer_name[prev_index+1:]
+ else:
+ name = buffer_name
+ string += ' '
+ if go_option_enabled('buffer_number'):
+ string += '%s%s' % (
+ weechat.color(weechat.config_get_plugin(
+ 'color_number' + selected)),
+ str(listbuf[i]['number']))
+ string += '%s%s%s' % (
+ weechat.color(weechat.config_get_plugin(
+ 'color_name' + selected)),
+ name,
+ weechat.color('reset'))
+ return ' ' + string if string else ''
+
+
+def go_input_modifier(data, modifier, modifier_data, string):
+ """This modifier is called when input text item is built by WeeChat.
+
+ This is commonly called after changes in input or cursor move: it builds
+ a new input with prefix ("Go to:"), and suffix (list of buffers found).
+ """
+ global old_input, buffers, buffers_pos
+ if modifier_data != weechat.current_buffer():
+ return ''
+ names = ''
+ new_input = weechat.string_remove_color(string, '')
+ new_input = new_input.lstrip()
+ if old_input is None or new_input != old_input:
+ old_buffers = buffers
+ buffers = go_matching_buffers(new_input)
+ if buffers != old_buffers and len(new_input) > 0:
+ if len(buffers) == 1 and go_option_enabled('auto_jump'):
+ weechat.command(modifier_data, '/wait 1ms /input return')
+ buffers_pos = 0
+ old_input = new_input
+ names = go_buffers_to_string(buffers, buffers_pos, new_input.strip())
+ return weechat.config_get_plugin('message') + string + names
+
+
+def go_command_run_input(data, buf, command):
+ """Function called when a command "/input xxx" is run."""
+ global buffers, buffers_pos
+ if command.startswith('/input search_text') or command.startswith('/input jump'):
+ # search text or jump to another buffer is forbidden now
+ return weechat.WEECHAT_RC_OK_EAT
+ elif command == '/input complete_next':
+ # choose next buffer in list
+ buffers_pos += 1
+ if buffers_pos >= len(buffers):
+ buffers_pos = 0
+ weechat.hook_signal_send('input_text_changed',
+ weechat.WEECHAT_HOOK_SIGNAL_STRING, '')
+ return weechat.WEECHAT_RC_OK_EAT
+ elif command == '/input complete_previous':
+ # choose previous buffer in list
+ buffers_pos -= 1
+ if buffers_pos < 0:
+ buffers_pos = len(buffers) - 1
+ weechat.hook_signal_send('input_text_changed',
+ weechat.WEECHAT_HOOK_SIGNAL_STRING, '')
+ return weechat.WEECHAT_RC_OK_EAT
+ elif command == '/input return':
+ # switch to selected buffer (if any)
+ go_end(buf)
+ if len(buffers) > 0:
+ weechat.command(
+ buf, '/buffer ' + str(buffers[buffers_pos]['full_name']))
+ return weechat.WEECHAT_RC_OK_EAT
+ return weechat.WEECHAT_RC_OK
+
+
+def go_command_run_buffer(data, buf, command):
+ """Function called when a command "/buffer xxx" is run."""
+ return weechat.WEECHAT_RC_OK_EAT
+
+
+def go_command_run_window(data, buf, command):
+ """Function called when a command "/window xxx" is run."""
+ return weechat.WEECHAT_RC_OK_EAT
+
+
+def go_unload_script():
+ """Function called when script is unloaded."""
+ go_unhook_all()
+ return weechat.WEECHAT_RC_OK
+
+
+def go_main():
+ """Entry point."""
+ if not weechat.register(SCRIPT_NAME, SCRIPT_AUTHOR, SCRIPT_VERSION,
+ SCRIPT_LICENSE, SCRIPT_DESC,
+ 'go_unload_script', ''):
+ return
+ weechat.hook_command(
+ SCRIPT_COMMAND,
+ 'Quick jump to buffers', '[name]',
+ 'name: directly jump to buffer by name (without argument, list is '
+ 'displayed)\n\n'
+ 'You can bind command to a key, for example:\n'
+ ' /key bind meta-g /go\n\n'
+ 'You can use completion key (commonly Tab and shift-Tab) to select '
+ 'next/previous buffer in list.',
+ '%(buffers_names)',
+ 'go_cmd', '')
+
+ # set default settings
+ version = weechat.info_get('version_number', '') or 0
+ for option, value in SETTINGS.items():
+ if not weechat.config_is_set_plugin(option):
+ weechat.config_set_plugin(option, value[0])
+ if int(version) >= 0x00030500:
+ weechat.config_set_desc_plugin(
+ option, '%s (default: "%s")' % (value[1], value[0]))
+ weechat.hook_info('go_running',
+ 'Return "1" if go is running, otherwise "0"',
+ '',
+ 'go_info_running', '')
+
+
+if __name__ == "__main__" and IMPORT_OK:
+ go_main()