add third_party
This commit is contained in:
parent
db37d29dc8
commit
424e15b31c
27
toxygen/third_party/qweechat/about.py.diff
vendored
27
toxygen/third_party/qweechat/about.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** about.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- about.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 20,30 ****
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""About dialog box."""
|
|
||||||
|
|
||||||
! from PySide6 import QtCore, QtWidgets as QtGui
|
|
||||||
|
|
||||||
from qweechat.version import qweechat_version
|
|
||||||
|
|
||||||
|
|
||||||
class AboutDialog(QtGui.QDialog):
|
|
||||||
--- 20,30 ----
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""About dialog box."""
|
|
||||||
|
|
||||||
! from PyQt5 import QtCore, QtWidgets as QtGui
|
|
||||||
|
|
||||||
from qweechat.version import qweechat_version
|
|
||||||
|
|
||||||
|
|
||||||
class AboutDialog(QtGui.QDialog):
|
|
61
toxygen/third_party/qweechat/about.py.dst
vendored
61
toxygen/third_party/qweechat/about.py.dst
vendored
@ -1,61 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# about.py - about dialog box
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""About dialog box."""
|
|
||||||
|
|
||||||
from PySide6 import QtCore, QtWidgets as QtGui
|
|
||||||
|
|
||||||
from qweechat.version import qweechat_version
|
|
||||||
|
|
||||||
|
|
||||||
class AboutDialog(QtGui.QDialog):
|
|
||||||
"""About dialog."""
|
|
||||||
|
|
||||||
def __init__(self, app_name, author, weechat_site, *args):
|
|
||||||
QtGui.QDialog.__init__(*(self,) + args)
|
|
||||||
self.setModal(True)
|
|
||||||
self.setWindowTitle('About')
|
|
||||||
|
|
||||||
close_button = QtGui.QPushButton('Close')
|
|
||||||
close_button.pressed.connect(self.close)
|
|
||||||
|
|
||||||
hbox = QtGui.QHBoxLayout()
|
|
||||||
hbox.addStretch(1)
|
|
||||||
hbox.addWidget(close_button)
|
|
||||||
hbox.addStretch(1)
|
|
||||||
|
|
||||||
vbox = QtGui.QVBoxLayout()
|
|
||||||
messages = [
|
|
||||||
f'<b>{app_name}</b> {qweechat_version()}',
|
|
||||||
f'© 2011-2022 {author}',
|
|
||||||
'',
|
|
||||||
f'<a href="{weechat_site}">{weechat_site}</a>',
|
|
||||||
'',
|
|
||||||
]
|
|
||||||
for msg in messages:
|
|
||||||
label = QtGui.QLabel(msg)
|
|
||||||
label.setAlignment(QtCore.Qt.AlignHCenter)
|
|
||||||
vbox.addWidget(label)
|
|
||||||
vbox.addLayout(hbox)
|
|
||||||
|
|
||||||
self.setLayout(vbox)
|
|
||||||
self.show()
|
|
27
toxygen/third_party/qweechat/buffer.py.diff
vendored
27
toxygen/third_party/qweechat/buffer.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** buffer.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- buffer.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 22,32 ****
|
|
||||||
|
|
||||||
"""Management of WeeChat buffers/nicklist."""
|
|
||||||
|
|
||||||
from pkg_resources import resource_filename
|
|
||||||
|
|
||||||
! from PySide6 import QtCore, QtGui, QtWidgets
|
|
||||||
|
|
||||||
from qweechat.chat import ChatTextEdit
|
|
||||||
from qweechat.input import InputLineEdit
|
|
||||||
from qweechat.weechat import color
|
|
||||||
|
|
||||||
--- 22,32 ----
|
|
||||||
|
|
||||||
"""Management of WeeChat buffers/nicklist."""
|
|
||||||
|
|
||||||
from pkg_resources import resource_filename
|
|
||||||
|
|
||||||
! from PyQt5 import QtCore, QtGui, QtWidgets
|
|
||||||
|
|
||||||
from qweechat.chat import ChatTextEdit
|
|
||||||
from qweechat.input import InputLineEdit
|
|
||||||
from qweechat.weechat import color
|
|
||||||
|
|
248
toxygen/third_party/qweechat/buffer.py.dst
vendored
248
toxygen/third_party/qweechat/buffer.py.dst
vendored
@ -1,248 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# buffer.py - management of WeeChat buffers/nicklist
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Management of WeeChat buffers/nicklist."""
|
|
||||||
|
|
||||||
from pkg_resources import resource_filename
|
|
||||||
|
|
||||||
from PySide6 import QtCore, QtGui, QtWidgets
|
|
||||||
|
|
||||||
from qweechat.chat import ChatTextEdit
|
|
||||||
from qweechat.input import InputLineEdit
|
|
||||||
from qweechat.weechat import color
|
|
||||||
|
|
||||||
|
|
||||||
class GenericListWidget(QtWidgets.QListWidget):
|
|
||||||
"""Generic QListWidget with dynamic size."""
|
|
||||||
|
|
||||||
def __init__(self, *args):
|
|
||||||
super().__init__(*args)
|
|
||||||
self.setMaximumWidth(100)
|
|
||||||
self.setTextElideMode(QtCore.Qt.ElideNone)
|
|
||||||
self.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOff)
|
|
||||||
self.setFocusPolicy(QtCore.Qt.NoFocus)
|
|
||||||
pal = self.palette()
|
|
||||||
pal.setColor(QtGui.QPalette.Highlight, QtGui.QColor('#ddddff'))
|
|
||||||
pal.setColor(QtGui.QPalette.HighlightedText, QtGui.QColor('black'))
|
|
||||||
self.setPalette(pal)
|
|
||||||
|
|
||||||
def auto_resize(self):
|
|
||||||
size = self.sizeHintForColumn(0)
|
|
||||||
if size > 0:
|
|
||||||
size += 4
|
|
||||||
self.setMaximumWidth(size)
|
|
||||||
|
|
||||||
def clear(self, *args):
|
|
||||||
"""Re-implement clear to set dynamic size after clear."""
|
|
||||||
QtWidgets.QListWidget.clear(*(self,) + args)
|
|
||||||
self.auto_resize()
|
|
||||||
|
|
||||||
def addItem(self, *args):
|
|
||||||
"""Re-implement addItem to set dynamic size after add."""
|
|
||||||
QtWidgets.QListWidget.addItem(*(self,) + args)
|
|
||||||
self.auto_resize()
|
|
||||||
|
|
||||||
def insertItem(self, *args):
|
|
||||||
"""Re-implement insertItem to set dynamic size after insert."""
|
|
||||||
QtWidgets.QListWidget.insertItem(*(self,) + args)
|
|
||||||
self.auto_resize()
|
|
||||||
|
|
||||||
|
|
||||||
class BufferListWidget(GenericListWidget):
|
|
||||||
"""Widget with list of buffers."""
|
|
||||||
|
|
||||||
def switch_prev_buffer(self):
|
|
||||||
if self.currentRow() > 0:
|
|
||||||
self.setCurrentRow(self.currentRow() - 1)
|
|
||||||
else:
|
|
||||||
self.setCurrentRow(self.count() - 1)
|
|
||||||
|
|
||||||
def switch_next_buffer(self):
|
|
||||||
if self.currentRow() < self.count() - 1:
|
|
||||||
self.setCurrentRow(self.currentRow() + 1)
|
|
||||||
else:
|
|
||||||
self.setCurrentRow(0)
|
|
||||||
|
|
||||||
|
|
||||||
class BufferWidget(QtWidgets.QWidget):
|
|
||||||
"""
|
|
||||||
Widget with (from top to bottom):
|
|
||||||
title, chat + nicklist (optional) + prompt/input.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def __init__(self, display_nicklist=False):
|
|
||||||
super().__init__()
|
|
||||||
|
|
||||||
# title
|
|
||||||
self.title = QtWidgets.QLineEdit()
|
|
||||||
self.title.setFocusPolicy(QtCore.Qt.NoFocus)
|
|
||||||
|
|
||||||
# splitter with chat + nicklist
|
|
||||||
self.chat_nicklist = QtWidgets.QSplitter()
|
|
||||||
self.chat_nicklist.setSizePolicy(QtWidgets.QSizePolicy.Expanding,
|
|
||||||
QtWidgets.QSizePolicy.Expanding)
|
|
||||||
self.chat = ChatTextEdit(debug=False)
|
|
||||||
self.chat_nicklist.addWidget(self.chat)
|
|
||||||
self.nicklist = GenericListWidget()
|
|
||||||
if not display_nicklist:
|
|
||||||
self.nicklist.setVisible(False)
|
|
||||||
self.chat_nicklist.addWidget(self.nicklist)
|
|
||||||
|
|
||||||
# prompt + input
|
|
||||||
self.hbox_edit = QtWidgets.QHBoxLayout()
|
|
||||||
self.hbox_edit.setContentsMargins(0, 0, 0, 0)
|
|
||||||
self.hbox_edit.setSpacing(0)
|
|
||||||
self.input = InputLineEdit(self.chat)
|
|
||||||
self.hbox_edit.addWidget(self.input)
|
|
||||||
prompt_input = QtWidgets.QWidget()
|
|
||||||
prompt_input.setLayout(self.hbox_edit)
|
|
||||||
|
|
||||||
# vbox with title + chat/nicklist + prompt/input
|
|
||||||
vbox = QtWidgets.QVBoxLayout()
|
|
||||||
vbox.setContentsMargins(0, 0, 0, 0)
|
|
||||||
vbox.setSpacing(0)
|
|
||||||
vbox.addWidget(self.title)
|
|
||||||
vbox.addWidget(self.chat_nicklist)
|
|
||||||
vbox.addWidget(prompt_input)
|
|
||||||
|
|
||||||
self.setLayout(vbox)
|
|
||||||
|
|
||||||
def set_title(self, title):
|
|
||||||
"""Set buffer title."""
|
|
||||||
self.title.clear()
|
|
||||||
if title is not None:
|
|
||||||
self.title.setText(title)
|
|
||||||
|
|
||||||
def set_prompt(self, prompt):
|
|
||||||
"""Set prompt."""
|
|
||||||
if self.hbox_edit.count() > 1:
|
|
||||||
self.hbox_edit.takeAt(0)
|
|
||||||
if prompt is not None:
|
|
||||||
label = QtWidgets.QLabel(prompt)
|
|
||||||
label.setContentsMargins(0, 0, 5, 0)
|
|
||||||
self.hbox_edit.insertWidget(0, label)
|
|
||||||
|
|
||||||
|
|
||||||
class Buffer(QtCore.QObject):
|
|
||||||
"""A WeeChat buffer."""
|
|
||||||
|
|
||||||
bufferInput = QtCore.Signal(str, str)
|
|
||||||
|
|
||||||
def __init__(self, data=None):
|
|
||||||
QtCore.QObject.__init__(self)
|
|
||||||
self.data = data or {}
|
|
||||||
self.nicklist = {}
|
|
||||||
self.widget = BufferWidget(display_nicklist=self.data.get('nicklist',
|
|
||||||
0))
|
|
||||||
self.update_title()
|
|
||||||
self.update_prompt()
|
|
||||||
self.widget.input.textSent.connect(self.input_text_sent)
|
|
||||||
|
|
||||||
def pointer(self):
|
|
||||||
"""Return pointer on buffer."""
|
|
||||||
return self.data.get('__path', [''])[0]
|
|
||||||
|
|
||||||
def update_title(self):
|
|
||||||
"""Update title."""
|
|
||||||
try:
|
|
||||||
self.widget.set_title(
|
|
||||||
color.remove(self.data['title']))
|
|
||||||
except Exception: # noqa: E722
|
|
||||||
# TODO: Debug print the exception to be fixed.
|
|
||||||
# traceback.print_exc()
|
|
||||||
self.widget.set_title(None)
|
|
||||||
|
|
||||||
def update_prompt(self):
|
|
||||||
"""Update prompt."""
|
|
||||||
try:
|
|
||||||
self.widget.set_prompt(self.data['local_variables']['nick'])
|
|
||||||
except Exception: # noqa: E722
|
|
||||||
self.widget.set_prompt(None)
|
|
||||||
|
|
||||||
def input_text_sent(self, text):
|
|
||||||
"""Called when text has to be sent to buffer."""
|
|
||||||
if self.data:
|
|
||||||
self.bufferInput.emit(self.data['full_name'], text)
|
|
||||||
|
|
||||||
def nicklist_add_item(self, parent, group, prefix, name, visible):
|
|
||||||
"""Add a group/nick in nicklist."""
|
|
||||||
if group:
|
|
||||||
self.nicklist[name] = {
|
|
||||||
'visible': visible,
|
|
||||||
'nicks': []
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
self.nicklist[parent]['nicks'].append({
|
|
||||||
'prefix': prefix,
|
|
||||||
'name': name,
|
|
||||||
'visible': visible,
|
|
||||||
})
|
|
||||||
|
|
||||||
def nicklist_remove_item(self, parent, group, name):
|
|
||||||
"""Remove a group/nick from nicklist."""
|
|
||||||
if group:
|
|
||||||
if name in self.nicklist:
|
|
||||||
del self.nicklist[name]
|
|
||||||
else:
|
|
||||||
if parent in self.nicklist:
|
|
||||||
self.nicklist[parent]['nicks'] = [
|
|
||||||
nick for nick in self.nicklist[parent]['nicks']
|
|
||||||
if nick['name'] != name
|
|
||||||
]
|
|
||||||
|
|
||||||
def nicklist_update_item(self, parent, group, prefix, name, visible):
|
|
||||||
"""Update a group/nick in nicklist."""
|
|
||||||
if group:
|
|
||||||
if name in self.nicklist:
|
|
||||||
self.nicklist[name]['visible'] = visible
|
|
||||||
else:
|
|
||||||
if parent in self.nicklist:
|
|
||||||
for nick in self.nicklist[parent]['nicks']:
|
|
||||||
if nick['name'] == name:
|
|
||||||
nick['prefix'] = prefix
|
|
||||||
nick['visible'] = visible
|
|
||||||
break
|
|
||||||
|
|
||||||
def nicklist_refresh(self):
|
|
||||||
"""Refresh nicklist."""
|
|
||||||
self.widget.nicklist.clear()
|
|
||||||
for group in sorted(self.nicklist):
|
|
||||||
for nick in sorted(self.nicklist[group]['nicks'],
|
|
||||||
key=lambda n: n['name']):
|
|
||||||
prefix_color = {
|
|
||||||
'': '',
|
|
||||||
' ': '',
|
|
||||||
'+': 'yellow',
|
|
||||||
}
|
|
||||||
col = prefix_color.get(nick['prefix'], 'green')
|
|
||||||
if col:
|
|
||||||
icon = QtGui.QIcon(
|
|
||||||
resource_filename(__name__,
|
|
||||||
'data/icons/bullet_%s_8x8.png' %
|
|
||||||
col))
|
|
||||||
else:
|
|
||||||
pixmap = QtGui.QPixmap(8, 8)
|
|
||||||
pixmap.fill()
|
|
||||||
icon = QtGui.QIcon(pixmap)
|
|
||||||
item = QtWidgets.QListWidgetItem(icon, nick['name'])
|
|
||||||
self.widget.nicklist.addItem(item)
|
|
||||||
self.widget.nicklist.setVisible(True)
|
|
27
toxygen/third_party/qweechat/chat.py.diff
vendored
27
toxygen/third_party/qweechat/chat.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** chat.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- chat.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 22,32 ****
|
|
||||||
|
|
||||||
"""Chat area."""
|
|
||||||
|
|
||||||
import datetime
|
|
||||||
|
|
||||||
! from PySide6 import QtCore, QtWidgets, QtGui
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.weechat import color
|
|
||||||
|
|
||||||
|
|
||||||
--- 22,32 ----
|
|
||||||
|
|
||||||
"""Chat area."""
|
|
||||||
|
|
||||||
import datetime
|
|
||||||
|
|
||||||
! from PyQt5 import QtCore, QtWidgets, QtGui
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.weechat import color
|
|
||||||
|
|
||||||
|
|
142
toxygen/third_party/qweechat/chat.py.dst
vendored
142
toxygen/third_party/qweechat/chat.py.dst
vendored
@ -1,142 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# chat.py - chat area
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Chat area."""
|
|
||||||
|
|
||||||
import datetime
|
|
||||||
|
|
||||||
from PySide6 import QtCore, QtWidgets, QtGui
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.weechat import color
|
|
||||||
|
|
||||||
|
|
||||||
class ChatTextEdit(QtWidgets.QTextEdit):
|
|
||||||
"""Chat area."""
|
|
||||||
|
|
||||||
def __init__(self, debug, *args):
|
|
||||||
QtWidgets.QTextEdit.__init__(*(self,) + args)
|
|
||||||
self.debug = debug
|
|
||||||
self.readOnly = True
|
|
||||||
self.setFocusPolicy(QtCore.Qt.NoFocus)
|
|
||||||
self.setFontFamily('monospace')
|
|
||||||
self._textcolor = self.textColor()
|
|
||||||
self._bgcolor = QtGui.QColor('#FFFFFF')
|
|
||||||
self._setcolorcode = {
|
|
||||||
'F': (self.setTextColor, self._textcolor),
|
|
||||||
'B': (self.setTextBackgroundColor, self._bgcolor)
|
|
||||||
}
|
|
||||||
self._setfont = {
|
|
||||||
'*': self.setFontWeight,
|
|
||||||
'_': self.setFontUnderline,
|
|
||||||
'/': self.setFontItalic
|
|
||||||
}
|
|
||||||
self._fontvalues = {
|
|
||||||
False: {
|
|
||||||
'*': QtGui.QFont.Normal,
|
|
||||||
'_': False,
|
|
||||||
'/': False
|
|
||||||
},
|
|
||||||
True: {
|
|
||||||
'*': QtGui.QFont.Bold,
|
|
||||||
'_': True,
|
|
||||||
'/': True
|
|
||||||
}
|
|
||||||
}
|
|
||||||
self._color = color.Color(config.color_options(), self.debug)
|
|
||||||
|
|
||||||
def display(self, time, prefix, text, forcecolor=None):
|
|
||||||
if time == 0:
|
|
||||||
now = datetime.datetime.now()
|
|
||||||
else:
|
|
||||||
now = datetime.datetime.fromtimestamp(float(time))
|
|
||||||
self.setTextColor(QtGui.QColor('#999999'))
|
|
||||||
self.insertPlainText(now.strftime('%H:%M '))
|
|
||||||
prefix = self._color.convert(prefix)
|
|
||||||
text = self._color.convert(text)
|
|
||||||
if forcecolor:
|
|
||||||
if prefix:
|
|
||||||
prefix = '\x01(F%s)%s' % (forcecolor, prefix)
|
|
||||||
text = '\x01(F%s)%s' % (forcecolor, text)
|
|
||||||
if prefix:
|
|
||||||
self._display_with_colors(prefix + ' ')
|
|
||||||
if text:
|
|
||||||
self._display_with_colors(text)
|
|
||||||
if text[-1:] != '\n':
|
|
||||||
self.insertPlainText('\n')
|
|
||||||
else:
|
|
||||||
self.insertPlainText('\n')
|
|
||||||
self.scroll_bottom()
|
|
||||||
|
|
||||||
def _display_with_colors(self, string):
|
|
||||||
self.setTextColor(self._textcolor)
|
|
||||||
self.setTextBackgroundColor(self._bgcolor)
|
|
||||||
self._reset_attributes()
|
|
||||||
items = string.split('\x01')
|
|
||||||
for i, item in enumerate(items):
|
|
||||||
if i > 0 and item.startswith('('):
|
|
||||||
pos = item.find(')')
|
|
||||||
if pos >= 2:
|
|
||||||
action = item[1]
|
|
||||||
code = item[2:pos]
|
|
||||||
if action == '+':
|
|
||||||
# set attribute
|
|
||||||
self._set_attribute(code[0], True)
|
|
||||||
elif action == '-':
|
|
||||||
# remove attribute
|
|
||||||
self._set_attribute(code[0], False)
|
|
||||||
else:
|
|
||||||
# reset attributes and color
|
|
||||||
if code == 'r':
|
|
||||||
self._reset_attributes()
|
|
||||||
self._setcolorcode[action][0](
|
|
||||||
self._setcolorcode[action][1])
|
|
||||||
else:
|
|
||||||
# set attributes + color
|
|
||||||
while code.startswith(('*', '!', '/', '_', '|',
|
|
||||||
'r')):
|
|
||||||
if code[0] == 'r':
|
|
||||||
self._reset_attributes()
|
|
||||||
elif code[0] in self._setfont:
|
|
||||||
self._set_attribute(
|
|
||||||
code[0],
|
|
||||||
not self._font[code[0]])
|
|
||||||
code = code[1:]
|
|
||||||
if code:
|
|
||||||
self._setcolorcode[action][0](
|
|
||||||
QtGui.QColor(code))
|
|
||||||
item = item[pos+1:]
|
|
||||||
if len(item) > 0:
|
|
||||||
self.insertPlainText(item)
|
|
||||||
|
|
||||||
def _reset_attributes(self):
|
|
||||||
self._font = {}
|
|
||||||
for attr in self._setfont:
|
|
||||||
self._set_attribute(attr, False)
|
|
||||||
|
|
||||||
def _set_attribute(self, attr, value):
|
|
||||||
self._font[attr] = value
|
|
||||||
self._setfont[attr](self._fontvalues[self._font[attr]][attr])
|
|
||||||
|
|
||||||
def scroll_bottom(self):
|
|
||||||
scroll = self.verticalScrollBar()
|
|
||||||
scroll.setValue(scroll.maximum())
|
|
136
toxygen/third_party/qweechat/config.py.dst
vendored
136
toxygen/third_party/qweechat/config.py.dst
vendored
@ -1,136 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# config.py - configuration for QWeeChat
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Configuration for QWeeChat."""
|
|
||||||
|
|
||||||
import configparser
|
|
||||||
import os
|
|
||||||
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
CONFIG_DIR = '%s/.config/qweechat' % os.getenv('HOME')
|
|
||||||
CONFIG_FILENAME = '%s/qweechat.conf' % CONFIG_DIR
|
|
||||||
|
|
||||||
CONFIG_DEFAULT_RELAY_LINES = 50
|
|
||||||
|
|
||||||
CONFIG_DEFAULT_SECTIONS = ('relay', 'look', 'color')
|
|
||||||
CONFIG_DEFAULT_OPTIONS = (('relay.hostname', ''),
|
|
||||||
('relay.port', ''),
|
|
||||||
('relay.ssl', 'off'),
|
|
||||||
('relay.password', ''),
|
|
||||||
('relay.autoconnect', 'off'),
|
|
||||||
('relay.lines', str(CONFIG_DEFAULT_RELAY_LINES)),
|
|
||||||
('look.debug', 'off'),
|
|
||||||
('look.statusbar', 'off'))
|
|
||||||
|
|
||||||
# Default colors for WeeChat color options (option name, #rgb value)
|
|
||||||
CONFIG_DEFAULT_COLOR_OPTIONS = (
|
|
||||||
('separator', '#000066'), # 0
|
|
||||||
('chat', '#000000'), # 1
|
|
||||||
('chat_time', '#999999'), # 2
|
|
||||||
('chat_time_delimiters', '#000000'), # 3
|
|
||||||
('chat_prefix_error', '#FF6633'), # 4
|
|
||||||
('chat_prefix_network', '#990099'), # 5
|
|
||||||
('chat_prefix_action', '#000000'), # 6
|
|
||||||
('chat_prefix_join', '#00CC00'), # 7
|
|
||||||
('chat_prefix_quit', '#CC0000'), # 8
|
|
||||||
('chat_prefix_more', '#CC00FF'), # 9
|
|
||||||
('chat_prefix_suffix', '#330099'), # 10
|
|
||||||
('chat_buffer', '#000000'), # 11
|
|
||||||
('chat_server', '#000000'), # 12
|
|
||||||
('chat_channel', '#000000'), # 13
|
|
||||||
('chat_nick', '#000000'), # 14
|
|
||||||
('chat_nick_self', '*#000000'), # 15
|
|
||||||
('chat_nick_other', '#000000'), # 16
|
|
||||||
('', '#000000'), # 17 (nick1 -- obsolete)
|
|
||||||
('', '#000000'), # 18 (nick2 -- obsolete)
|
|
||||||
('', '#000000'), # 19 (nick3 -- obsolete)
|
|
||||||
('', '#000000'), # 20 (nick4 -- obsolete)
|
|
||||||
('', '#000000'), # 21 (nick5 -- obsolete)
|
|
||||||
('', '#000000'), # 22 (nick6 -- obsolete)
|
|
||||||
('', '#000000'), # 23 (nick7 -- obsolete)
|
|
||||||
('', '#000000'), # 24 (nick8 -- obsolete)
|
|
||||||
('', '#000000'), # 25 (nick9 -- obsolete)
|
|
||||||
('', '#000000'), # 26 (nick10 -- obsolete)
|
|
||||||
('chat_host', '#666666'), # 27
|
|
||||||
('chat_delimiters', '#9999FF'), # 28
|
|
||||||
('chat_highlight', '#3399CC'), # 29
|
|
||||||
('chat_read_marker', '#000000'), # 30
|
|
||||||
('chat_text_found', '#000000'), # 31
|
|
||||||
('chat_value', '#000000'), # 32
|
|
||||||
('chat_prefix_buffer', '#000000'), # 33
|
|
||||||
('chat_tags', '#000000'), # 34
|
|
||||||
('chat_inactive_window', '#000000'), # 35
|
|
||||||
('chat_inactive_buffer', '#000000'), # 36
|
|
||||||
('chat_prefix_buffer_inactive_buffer', '#000000'), # 37
|
|
||||||
('chat_nick_offline', '#000000'), # 38
|
|
||||||
('chat_nick_offline_highlight', '#000000'), # 39
|
|
||||||
('chat_nick_prefix', '#000000'), # 40
|
|
||||||
('chat_nick_suffix', '#000000'), # 41
|
|
||||||
('emphasis', '#000000'), # 42
|
|
||||||
('chat_day_change', '#000000'), # 43
|
|
||||||
)
|
|
||||||
config_color_options = []
|
|
||||||
|
|
||||||
|
|
||||||
def read():
|
|
||||||
"""Read config file."""
|
|
||||||
global config_color_options
|
|
||||||
config = configparser.RawConfigParser()
|
|
||||||
if os.path.isfile(CONFIG_FILENAME):
|
|
||||||
config.read(CONFIG_FILENAME)
|
|
||||||
|
|
||||||
# add missing sections/options
|
|
||||||
for section in CONFIG_DEFAULT_SECTIONS:
|
|
||||||
if not config.has_section(section):
|
|
||||||
config.add_section(section)
|
|
||||||
for option in reversed(CONFIG_DEFAULT_OPTIONS):
|
|
||||||
section, name = option[0].split('.', 1)
|
|
||||||
if not config.has_option(section, name):
|
|
||||||
config.set(section, name, option[1])
|
|
||||||
section = 'color'
|
|
||||||
for option in reversed(CONFIG_DEFAULT_COLOR_OPTIONS):
|
|
||||||
if option[0] and not config.has_option(section, option[0]):
|
|
||||||
config.set(section, option[0], option[1])
|
|
||||||
|
|
||||||
# build list of color options
|
|
||||||
config_color_options = []
|
|
||||||
for option in CONFIG_DEFAULT_COLOR_OPTIONS:
|
|
||||||
if option[0]:
|
|
||||||
config_color_options.append(config.get('color', option[0]))
|
|
||||||
else:
|
|
||||||
config_color_options.append('#000000')
|
|
||||||
|
|
||||||
return config
|
|
||||||
|
|
||||||
|
|
||||||
def write(config):
|
|
||||||
"""Write config file."""
|
|
||||||
Path(CONFIG_DIR).mkdir(mode=0o0700, parents=True, exist_ok=True)
|
|
||||||
with open(CONFIG_FILENAME, 'w') as cfg:
|
|
||||||
config.write(cfg)
|
|
||||||
|
|
||||||
|
|
||||||
def color_options():
|
|
||||||
"""Return color options."""
|
|
||||||
global config_color_options
|
|
||||||
return config_color_options
|
|
27
toxygen/third_party/qweechat/connection.py.diff
vendored
27
toxygen/third_party/qweechat/connection.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** connection.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- connection.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 20,30 ****
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Connection window."""
|
|
||||||
|
|
||||||
! from PySide6 import QtGui, QtWidgets
|
|
||||||
|
|
||||||
|
|
||||||
class ConnectionDialog(QtWidgets.QDialog):
|
|
||||||
"""Connection window."""
|
|
||||||
|
|
||||||
--- 20,30 ----
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Connection window."""
|
|
||||||
|
|
||||||
! from PyQt5 import QtGui, QtWidgets
|
|
||||||
|
|
||||||
|
|
||||||
class ConnectionDialog(QtWidgets.QDialog):
|
|
||||||
"""Connection window."""
|
|
||||||
|
|
122
toxygen/third_party/qweechat/connection.py.dst
vendored
122
toxygen/third_party/qweechat/connection.py.dst
vendored
@ -1,122 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# connection.py - connection window
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Connection window."""
|
|
||||||
|
|
||||||
from PySide6 import QtGui, QtWidgets
|
|
||||||
|
|
||||||
|
|
||||||
class ConnectionDialog(QtWidgets.QDialog):
|
|
||||||
"""Connection window."""
|
|
||||||
|
|
||||||
def __init__(self, values, *args):
|
|
||||||
super().__init__(*args)
|
|
||||||
self.values = values
|
|
||||||
self.setModal(True)
|
|
||||||
self.setWindowTitle('Connect to WeeChat')
|
|
||||||
|
|
||||||
grid = QtWidgets.QGridLayout()
|
|
||||||
grid.setSpacing(10)
|
|
||||||
|
|
||||||
self.fields = {}
|
|
||||||
focus = None
|
|
||||||
|
|
||||||
# hostname
|
|
||||||
grid.addWidget(QtWidgets.QLabel('<b>Hostname</b>'), 0, 0)
|
|
||||||
line_edit = QtWidgets.QLineEdit()
|
|
||||||
line_edit.setFixedWidth(200)
|
|
||||||
value = self.values.get('hostname', '')
|
|
||||||
line_edit.insert(value)
|
|
||||||
grid.addWidget(line_edit, 0, 1)
|
|
||||||
self.fields['hostname'] = line_edit
|
|
||||||
if not focus and not value:
|
|
||||||
focus = 'hostname'
|
|
||||||
|
|
||||||
# port / SSL
|
|
||||||
grid.addWidget(QtWidgets.QLabel('<b>Port</b>'), 1, 0)
|
|
||||||
line_edit = QtWidgets.QLineEdit()
|
|
||||||
line_edit.setFixedWidth(200)
|
|
||||||
value = self.values.get('port', '')
|
|
||||||
line_edit.insert(value)
|
|
||||||
grid.addWidget(line_edit, 1, 1)
|
|
||||||
self.fields['port'] = line_edit
|
|
||||||
if not focus and not value:
|
|
||||||
focus = 'port'
|
|
||||||
|
|
||||||
ssl = QtWidgets.QCheckBox('SSL')
|
|
||||||
ssl.setChecked(self.values['ssl'] == 'on')
|
|
||||||
grid.addWidget(ssl, 1, 2)
|
|
||||||
self.fields['ssl'] = ssl
|
|
||||||
|
|
||||||
# password
|
|
||||||
grid.addWidget(QtWidgets.QLabel('<b>Password</b>'), 2, 0)
|
|
||||||
line_edit = QtWidgets.QLineEdit()
|
|
||||||
line_edit.setFixedWidth(200)
|
|
||||||
line_edit.setEchoMode(QtWidgets.QLineEdit.Password)
|
|
||||||
value = self.values.get('password', '')
|
|
||||||
line_edit.insert(value)
|
|
||||||
grid.addWidget(line_edit, 2, 1)
|
|
||||||
self.fields['password'] = line_edit
|
|
||||||
if not focus and not value:
|
|
||||||
focus = 'password'
|
|
||||||
|
|
||||||
# TOTP (Time-Based One-Time Password)
|
|
||||||
label = QtWidgets.QLabel('TOTP')
|
|
||||||
label.setToolTip('Time-Based One-Time Password (6 digits)')
|
|
||||||
grid.addWidget(label, 3, 0)
|
|
||||||
line_edit = QtWidgets.QLineEdit()
|
|
||||||
line_edit.setPlaceholderText('6 digits')
|
|
||||||
validator = QtGui.QIntValidator(0, 999999, self)
|
|
||||||
line_edit.setValidator(validator)
|
|
||||||
line_edit.setFixedWidth(80)
|
|
||||||
value = self.values.get('totp', '')
|
|
||||||
line_edit.insert(value)
|
|
||||||
grid.addWidget(line_edit, 3, 1)
|
|
||||||
self.fields['totp'] = line_edit
|
|
||||||
if not focus and not value:
|
|
||||||
focus = 'totp'
|
|
||||||
|
|
||||||
# lines
|
|
||||||
grid.addWidget(QtWidgets.QLabel('Lines'), 4, 0)
|
|
||||||
line_edit = QtWidgets.QLineEdit()
|
|
||||||
line_edit.setFixedWidth(200)
|
|
||||||
validator = QtGui.QIntValidator(0, 2147483647, self)
|
|
||||||
line_edit.setValidator(validator)
|
|
||||||
line_edit.setFixedWidth(80)
|
|
||||||
value = self.values.get('lines', '')
|
|
||||||
line_edit.insert(value)
|
|
||||||
grid.addWidget(line_edit, 4, 1)
|
|
||||||
self.fields['lines'] = line_edit
|
|
||||||
if not focus and not value:
|
|
||||||
focus = 'lines'
|
|
||||||
|
|
||||||
self.dialog_buttons = QtWidgets.QDialogButtonBox()
|
|
||||||
self.dialog_buttons.setStandardButtons(
|
|
||||||
QtWidgets.QDialogButtonBox.Ok | QtWidgets.QDialogButtonBox.Cancel)
|
|
||||||
self.dialog_buttons.rejected.connect(self.close)
|
|
||||||
|
|
||||||
grid.addWidget(self.dialog_buttons, 5, 0, 1, 2)
|
|
||||||
self.setLayout(grid)
|
|
||||||
self.show()
|
|
||||||
|
|
||||||
if focus:
|
|
||||||
self.fields[focus].setFocus()
|
|
27
toxygen/third_party/qweechat/debug.py.diff
vendored
27
toxygen/third_party/qweechat/debug.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** debug.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- debug.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 20,30 ****
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Debug window."""
|
|
||||||
|
|
||||||
! from PySide6 import QtWidgets
|
|
||||||
|
|
||||||
from qweechat.chat import ChatTextEdit
|
|
||||||
from qweechat.input import InputLineEdit
|
|
||||||
|
|
||||||
|
|
||||||
--- 20,30 ----
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Debug window."""
|
|
||||||
|
|
||||||
! from PyQt5 import QtWidgets
|
|
||||||
|
|
||||||
from qweechat.chat import ChatTextEdit
|
|
||||||
from qweechat.input import InputLineEdit
|
|
||||||
|
|
||||||
|
|
51
toxygen/third_party/qweechat/debug.py.dst
vendored
51
toxygen/third_party/qweechat/debug.py.dst
vendored
@ -1,51 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# debug.py - debug window
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Debug window."""
|
|
||||||
|
|
||||||
from PySide6 import QtWidgets
|
|
||||||
|
|
||||||
from qweechat.chat import ChatTextEdit
|
|
||||||
from qweechat.input import InputLineEdit
|
|
||||||
|
|
||||||
|
|
||||||
class DebugDialog(QtWidgets.QDialog):
|
|
||||||
"""Debug dialog."""
|
|
||||||
|
|
||||||
def __init__(self, *args):
|
|
||||||
QtWidgets.QDialog.__init__(*(self,) + args)
|
|
||||||
self.resize(1024, 768)
|
|
||||||
self.setWindowTitle('Debug console')
|
|
||||||
|
|
||||||
self.chat = ChatTextEdit(debug=True)
|
|
||||||
self.input = InputLineEdit(self.chat)
|
|
||||||
|
|
||||||
vbox = QtWidgets.QVBoxLayout()
|
|
||||||
vbox.addWidget(self.chat)
|
|
||||||
vbox.addWidget(self.input)
|
|
||||||
|
|
||||||
self.setLayout(vbox)
|
|
||||||
self.show()
|
|
||||||
|
|
||||||
def display_lines(self, lines):
|
|
||||||
for line in lines:
|
|
||||||
self.chat.display(*line[0], **line[1])
|
|
27
toxygen/third_party/qweechat/input.py.diff
vendored
27
toxygen/third_party/qweechat/input.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** input.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- input.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 20,30 ****
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Input line for chat and debug window."""
|
|
||||||
|
|
||||||
! from PySide6 import QtCore, QtWidgets
|
|
||||||
|
|
||||||
|
|
||||||
class InputLineEdit(QtWidgets.QLineEdit):
|
|
||||||
"""Input line."""
|
|
||||||
|
|
||||||
--- 20,30 ----
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Input line for chat and debug window."""
|
|
||||||
|
|
||||||
! from PyQt5 import QtCore, QtWidgets
|
|
||||||
|
|
||||||
|
|
||||||
class InputLineEdit(QtWidgets.QLineEdit):
|
|
||||||
"""Input line."""
|
|
||||||
|
|
95
toxygen/third_party/qweechat/input.py.dst
vendored
95
toxygen/third_party/qweechat/input.py.dst
vendored
@ -1,95 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# input.py - input line for chat and debug window
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Input line for chat and debug window."""
|
|
||||||
|
|
||||||
from PySide6 import QtCore, QtWidgets
|
|
||||||
|
|
||||||
|
|
||||||
class InputLineEdit(QtWidgets.QLineEdit):
|
|
||||||
"""Input line."""
|
|
||||||
|
|
||||||
bufferSwitchPrev = QtCore.Signal()
|
|
||||||
bufferSwitchNext = QtCore.Signal()
|
|
||||||
textSent = QtCore.Signal(str)
|
|
||||||
|
|
||||||
def __init__(self, scroll_widget):
|
|
||||||
super().__init__()
|
|
||||||
self.scroll_widget = scroll_widget
|
|
||||||
self._history = []
|
|
||||||
self._history_index = -1
|
|
||||||
self.returnPressed.connect(self._input_return_pressed)
|
|
||||||
|
|
||||||
def keyPressEvent(self, event):
|
|
||||||
key = event.key()
|
|
||||||
modifiers = event.modifiers()
|
|
||||||
scroll = self.scroll_widget.verticalScrollBar()
|
|
||||||
if modifiers == QtCore.Qt.ControlModifier:
|
|
||||||
if key == QtCore.Qt.Key_PageUp:
|
|
||||||
self.bufferSwitchPrev.emit()
|
|
||||||
elif key == QtCore.Qt.Key_PageDown:
|
|
||||||
self.bufferSwitchNext.emit()
|
|
||||||
else:
|
|
||||||
QtWidgets.QLineEdit.keyPressEvent(self, event)
|
|
||||||
elif modifiers == QtCore.Qt.AltModifier:
|
|
||||||
if key in (QtCore.Qt.Key_Left, QtCore.Qt.Key_Up):
|
|
||||||
self.bufferSwitchPrev.emit()
|
|
||||||
elif key in (QtCore.Qt.Key_Right, QtCore.Qt.Key_Down):
|
|
||||||
self.bufferSwitchNext.emit()
|
|
||||||
elif key == QtCore.Qt.Key_PageUp:
|
|
||||||
scroll.setValue(scroll.value() - (scroll.pageStep() / 10))
|
|
||||||
elif key == QtCore.Qt.Key_PageDown:
|
|
||||||
scroll.setValue(scroll.value() + (scroll.pageStep() / 10))
|
|
||||||
elif key == QtCore.Qt.Key_Home:
|
|
||||||
scroll.setValue(scroll.minimum())
|
|
||||||
elif key == QtCore.Qt.Key_End:
|
|
||||||
scroll.setValue(scroll.maximum())
|
|
||||||
else:
|
|
||||||
QtWidgets.QLineEdit.keyPressEvent(self, event)
|
|
||||||
elif key == QtCore.Qt.Key_PageUp:
|
|
||||||
scroll.setValue(scroll.value() - scroll.pageStep())
|
|
||||||
elif key == QtCore.Qt.Key_PageDown:
|
|
||||||
scroll.setValue(scroll.value() + scroll.pageStep())
|
|
||||||
elif key == QtCore.Qt.Key_Up:
|
|
||||||
self._history_navigate(-1)
|
|
||||||
elif key == QtCore.Qt.Key_Down:
|
|
||||||
self._history_navigate(1)
|
|
||||||
else:
|
|
||||||
QtWidgets.QLineEdit.keyPressEvent(self, event)
|
|
||||||
|
|
||||||
def _input_return_pressed(self):
|
|
||||||
self._history.append(self.text())
|
|
||||||
self._history_index = len(self._history)
|
|
||||||
self.textSent.emit(self.text())
|
|
||||||
self.clear()
|
|
||||||
|
|
||||||
def _history_navigate(self, direction):
|
|
||||||
if self._history:
|
|
||||||
self._history_index += direction
|
|
||||||
if self._history_index < 0:
|
|
||||||
self._history_index = 0
|
|
||||||
return
|
|
||||||
if self._history_index > len(self._history) - 1:
|
|
||||||
self._history_index = len(self._history)
|
|
||||||
self.clear()
|
|
||||||
return
|
|
||||||
self.setText(self._history[self._history_index])
|
|
27
toxygen/third_party/qweechat/network.py.diff
vendored
27
toxygen/third_party/qweechat/network.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** network.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- network.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 24,34 ****
|
|
||||||
|
|
||||||
import hashlib
|
|
||||||
import secrets
|
|
||||||
import struct
|
|
||||||
|
|
||||||
! from PySide6 import QtCore, QtNetwork
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.debug import DebugDialog
|
|
||||||
|
|
||||||
|
|
||||||
--- 24,34 ----
|
|
||||||
|
|
||||||
import hashlib
|
|
||||||
import secrets
|
|
||||||
import struct
|
|
||||||
|
|
||||||
! from PyQt5 import QtCore, QtNetwork
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.debug import DebugDialog
|
|
||||||
|
|
||||||
|
|
356
toxygen/third_party/qweechat/network.py.dst
vendored
356
toxygen/third_party/qweechat/network.py.dst
vendored
@ -1,356 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# network.py - I/O with WeeChat/relay
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""I/O with WeeChat/relay."""
|
|
||||||
|
|
||||||
import hashlib
|
|
||||||
import secrets
|
|
||||||
import struct
|
|
||||||
|
|
||||||
from PySide6 import QtCore, QtNetwork
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.debug import DebugDialog
|
|
||||||
|
|
||||||
|
|
||||||
# list of supported hash algorithms on our side
|
|
||||||
# (the hash algorithm will be negotiated with the remote WeeChat)
|
|
||||||
_HASH_ALGOS_LIST = [
|
|
||||||
'plain',
|
|
||||||
'sha256',
|
|
||||||
'sha512',
|
|
||||||
'pbkdf2+sha256',
|
|
||||||
'pbkdf2+sha512',
|
|
||||||
]
|
|
||||||
_HASH_ALGOS = ':'.join(_HASH_ALGOS_LIST)
|
|
||||||
|
|
||||||
# handshake with remote WeeChat (before init)
|
|
||||||
_PROTO_HANDSHAKE = f'(handshake) handshake password_hash_algo={_HASH_ALGOS}\n'
|
|
||||||
|
|
||||||
# initialize with the password (plain text)
|
|
||||||
_PROTO_INIT_PWD = 'init password=%(password)s%(totp)s\n' # nosec
|
|
||||||
|
|
||||||
# initialize with the hashed password
|
|
||||||
_PROTO_INIT_HASH = ('init password_hash='
|
|
||||||
'%(algo)s:%(salt)s%(iter)s:%(hash)s%(totp)s\n')
|
|
||||||
|
|
||||||
_PROTO_SYNC_CMDS = [
|
|
||||||
# get buffers
|
|
||||||
'(listbuffers) hdata buffer:gui_buffers(*) number,full_name,short_name,'
|
|
||||||
'type,nicklist,title,local_variables',
|
|
||||||
# get lines
|
|
||||||
'(listlines) hdata buffer:gui_buffers(*)/own_lines/last_line(-%(lines)d)/'
|
|
||||||
'data date,displayed,prefix,message',
|
|
||||||
# get nicklist for all buffers
|
|
||||||
'(nicklist) nicklist',
|
|
||||||
# enable synchronization
|
|
||||||
'sync',
|
|
||||||
]
|
|
||||||
|
|
||||||
STATUS_DISCONNECTED = 'disconnected'
|
|
||||||
STATUS_CONNECTING = 'connecting'
|
|
||||||
STATUS_AUTHENTICATING = 'authenticating'
|
|
||||||
STATUS_CONNECTED = 'connected'
|
|
||||||
|
|
||||||
NETWORK_STATUS = {
|
|
||||||
STATUS_DISCONNECTED: {
|
|
||||||
'label': 'Disconnected',
|
|
||||||
'color': '#aa0000',
|
|
||||||
'icon': 'dialog-close.png',
|
|
||||||
},
|
|
||||||
STATUS_CONNECTING: {
|
|
||||||
'label': 'Connecting…',
|
|
||||||
'color': '#dd5f00',
|
|
||||||
'icon': 'dialog-warning.png',
|
|
||||||
},
|
|
||||||
STATUS_AUTHENTICATING: {
|
|
||||||
'label': 'Authenticating…',
|
|
||||||
'color': '#007fff',
|
|
||||||
'icon': 'dialog-password.png',
|
|
||||||
},
|
|
||||||
STATUS_CONNECTED: {
|
|
||||||
'label': 'Connected',
|
|
||||||
'color': 'green',
|
|
||||||
'icon': 'dialog-ok-apply.png',
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class Network(QtCore.QObject):
|
|
||||||
"""I/O with WeeChat/relay."""
|
|
||||||
|
|
||||||
statusChanged = QtCore.Signal(str, str)
|
|
||||||
messageFromWeechat = QtCore.Signal(QtCore.QByteArray)
|
|
||||||
|
|
||||||
def __init__(self, *args):
|
|
||||||
super().__init__(*args)
|
|
||||||
self._init_connection()
|
|
||||||
self.debug_lines = []
|
|
||||||
self.debug_dialog = None
|
|
||||||
self._lines = config.CONFIG_DEFAULT_RELAY_LINES
|
|
||||||
self._buffer = QtCore.QByteArray()
|
|
||||||
self._socket = QtNetwork.QSslSocket()
|
|
||||||
self._socket.connected.connect(self._socket_connected)
|
|
||||||
self._socket.readyRead.connect(self._socket_read)
|
|
||||||
self._socket.disconnected.connect(self._socket_disconnected)
|
|
||||||
|
|
||||||
def _init_connection(self):
|
|
||||||
self.status = STATUS_DISCONNECTED
|
|
||||||
self._hostname = None
|
|
||||||
self._port = None
|
|
||||||
self._ssl = None
|
|
||||||
self._password = None
|
|
||||||
self._totp = None
|
|
||||||
self._handshake_received = False
|
|
||||||
self._handshake_timer = None
|
|
||||||
self._handshake_timer = False
|
|
||||||
self._pwd_hash_algo = None
|
|
||||||
self._pwd_hash_iter = 0
|
|
||||||
self._server_nonce = None
|
|
||||||
|
|
||||||
def set_status(self, status):
|
|
||||||
"""Set current status."""
|
|
||||||
self.status = status
|
|
||||||
self.statusChanged.emit(status, None)
|
|
||||||
|
|
||||||
def pbkdf2(self, hash_name, salt):
|
|
||||||
"""Return hashed password with PBKDF2-HMAC."""
|
|
||||||
return hashlib.pbkdf2_hmac(
|
|
||||||
hash_name,
|
|
||||||
password=self._password.encode('utf-8'),
|
|
||||||
salt=salt,
|
|
||||||
iterations=self._pwd_hash_iter,
|
|
||||||
).hex()
|
|
||||||
|
|
||||||
def _build_init_command(self):
|
|
||||||
"""Build the init command to send to WeeChat."""
|
|
||||||
totp = f',totp={self._totp}' if self._totp else ''
|
|
||||||
if self._pwd_hash_algo == 'plain':
|
|
||||||
cmd = _PROTO_INIT_PWD % {
|
|
||||||
'password': self._password,
|
|
||||||
'totp': totp,
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
client_nonce = secrets.token_bytes(16)
|
|
||||||
salt = self._server_nonce + client_nonce
|
|
||||||
pwd_hash = None
|
|
||||||
iterations = ''
|
|
||||||
if self._pwd_hash_algo == 'pbkdf2+sha512':
|
|
||||||
pwd_hash = self.pbkdf2('sha512', salt)
|
|
||||||
iterations = f':{self._pwd_hash_iter}'
|
|
||||||
elif self._pwd_hash_algo == 'pbkdf2+sha256':
|
|
||||||
pwd_hash = self.pbkdf2('sha256', salt)
|
|
||||||
iterations = f':{self._pwd_hash_iter}'
|
|
||||||
elif self._pwd_hash_algo == 'sha512':
|
|
||||||
pwd = salt + self._password.encode('utf-8')
|
|
||||||
pwd_hash = hashlib.sha512(pwd).hexdigest()
|
|
||||||
elif self._pwd_hash_algo == 'sha256':
|
|
||||||
pwd = salt + self._password.encode('utf-8')
|
|
||||||
pwd_hash = hashlib.sha256(pwd).hexdigest()
|
|
||||||
if not pwd_hash:
|
|
||||||
return None
|
|
||||||
cmd = _PROTO_INIT_HASH % {
|
|
||||||
'algo': self._pwd_hash_algo,
|
|
||||||
'salt': bytearray(salt).hex(),
|
|
||||||
'iter': iterations,
|
|
||||||
'hash': pwd_hash,
|
|
||||||
'totp': totp,
|
|
||||||
}
|
|
||||||
return cmd
|
|
||||||
|
|
||||||
def _build_sync_command(self):
|
|
||||||
"""Build the sync commands to send to WeeChat."""
|
|
||||||
cmd = '\n'.join(_PROTO_SYNC_CMDS) + '\n'
|
|
||||||
return cmd % {'lines': self._lines}
|
|
||||||
|
|
||||||
def handshake_timer_expired(self):
|
|
||||||
if self.status == STATUS_AUTHENTICATING:
|
|
||||||
self._pwd_hash_algo = 'plain'
|
|
||||||
self.send_to_weechat(self._build_init_command())
|
|
||||||
self.sync_weechat()
|
|
||||||
self.set_status(STATUS_CONNECTED)
|
|
||||||
|
|
||||||
def _socket_connected(self):
|
|
||||||
"""Slot: socket connected."""
|
|
||||||
self.set_status(STATUS_AUTHENTICATING)
|
|
||||||
self.send_to_weechat(_PROTO_HANDSHAKE)
|
|
||||||
self._handshake_timer = QtCore.QTimer()
|
|
||||||
self._handshake_timer.setSingleShot(True)
|
|
||||||
self._handshake_timer.setInterval(2000)
|
|
||||||
self._handshake_timer.timeout.connect(self.handshake_timer_expired)
|
|
||||||
self._handshake_timer.start()
|
|
||||||
|
|
||||||
def _socket_read(self):
|
|
||||||
"""Slot: data available on socket."""
|
|
||||||
data = self._socket.readAll()
|
|
||||||
self._buffer.append(data)
|
|
||||||
while len(self._buffer) >= 4:
|
|
||||||
remainder = None
|
|
||||||
length = struct.unpack('>i', self._buffer[0:4].data())[0]
|
|
||||||
if len(self._buffer) < length:
|
|
||||||
# partial message, just wait for end of message
|
|
||||||
break
|
|
||||||
# more than one message?
|
|
||||||
if length < len(self._buffer):
|
|
||||||
# save beginning of another message
|
|
||||||
remainder = self._buffer[length:]
|
|
||||||
self._buffer = self._buffer[0:length]
|
|
||||||
self.messageFromWeechat.emit(self._buffer)
|
|
||||||
if not self.is_connected():
|
|
||||||
return
|
|
||||||
self._buffer.clear()
|
|
||||||
if remainder:
|
|
||||||
self._buffer.append(remainder)
|
|
||||||
|
|
||||||
def _socket_disconnected(self):
|
|
||||||
"""Slot: socket disconnected."""
|
|
||||||
if self._handshake_timer:
|
|
||||||
self._handshake_timer.stop()
|
|
||||||
self._init_connection()
|
|
||||||
self.set_status(STATUS_DISCONNECTED)
|
|
||||||
|
|
||||||
def is_connected(self):
|
|
||||||
"""Return True if the socket is connected, False otherwise."""
|
|
||||||
return self._socket.state() == QtNetwork.QAbstractSocket.ConnectedState
|
|
||||||
|
|
||||||
def is_ssl(self):
|
|
||||||
"""Return True if SSL is used, False otherwise."""
|
|
||||||
return self._ssl
|
|
||||||
|
|
||||||
def connect_weechat(self, hostname, port, ssl, password, totp, lines):
|
|
||||||
"""Connect to WeeChat."""
|
|
||||||
self._hostname = hostname
|
|
||||||
try:
|
|
||||||
self._port = int(port)
|
|
||||||
except ValueError:
|
|
||||||
self._port = 0
|
|
||||||
self._ssl = ssl
|
|
||||||
self._password = password
|
|
||||||
self._totp = totp
|
|
||||||
try:
|
|
||||||
self._lines = int(lines)
|
|
||||||
except ValueError:
|
|
||||||
self._lines = config.CONFIG_DEFAULT_RELAY_LINES
|
|
||||||
if self._socket.state() == QtNetwork.QAbstractSocket.ConnectedState:
|
|
||||||
return
|
|
||||||
if self._socket.state() != QtNetwork.QAbstractSocket.UnconnectedState:
|
|
||||||
self._socket.abort()
|
|
||||||
if self._ssl:
|
|
||||||
self._socket.ignoreSslErrors()
|
|
||||||
self._socket.connectToHostEncrypted(self._hostname, self._port)
|
|
||||||
else:
|
|
||||||
self._socket.connectToHost(self._hostname, self._port)
|
|
||||||
self.set_status(STATUS_CONNECTING)
|
|
||||||
|
|
||||||
def disconnect_weechat(self):
|
|
||||||
"""Disconnect from WeeChat."""
|
|
||||||
if self._socket.state() == QtNetwork.QAbstractSocket.UnconnectedState:
|
|
||||||
self.set_status(STATUS_DISCONNECTED)
|
|
||||||
return
|
|
||||||
if self._socket.state() == QtNetwork.QAbstractSocket.ConnectedState:
|
|
||||||
self.send_to_weechat('quit\n')
|
|
||||||
self._socket.waitForBytesWritten(1000)
|
|
||||||
else:
|
|
||||||
self.set_status(STATUS_DISCONNECTED)
|
|
||||||
self._socket.abort()
|
|
||||||
|
|
||||||
def send_to_weechat(self, message):
|
|
||||||
"""Send a message to WeeChat."""
|
|
||||||
self.debug_print(0, '<==', message, forcecolor='#AA0000')
|
|
||||||
self._socket.write(message.encode('utf-8'))
|
|
||||||
|
|
||||||
def init_with_handshake(self, response):
|
|
||||||
"""Initialize with WeeChat using the handshake response."""
|
|
||||||
self._pwd_hash_algo = response['password_hash_algo']
|
|
||||||
self._pwd_hash_iter = int(response['password_hash_iterations'])
|
|
||||||
self._server_nonce = bytearray.fromhex(response['nonce'])
|
|
||||||
if self._pwd_hash_algo:
|
|
||||||
cmd = self._build_init_command()
|
|
||||||
if cmd:
|
|
||||||
self.send_to_weechat(cmd)
|
|
||||||
self.sync_weechat()
|
|
||||||
self.set_status(STATUS_CONNECTED)
|
|
||||||
return
|
|
||||||
# failed to initialize: disconnect
|
|
||||||
self.disconnect_weechat()
|
|
||||||
|
|
||||||
def desync_weechat(self):
|
|
||||||
"""Desynchronize from WeeChat."""
|
|
||||||
self.send_to_weechat('desync\n')
|
|
||||||
|
|
||||||
def sync_weechat(self):
|
|
||||||
"""Synchronize with WeeChat."""
|
|
||||||
self.send_to_weechat(self._build_sync_command())
|
|
||||||
|
|
||||||
def status_label(self, status):
|
|
||||||
"""Return the label for a given status."""
|
|
||||||
return NETWORK_STATUS.get(status, {}).get('label', '')
|
|
||||||
|
|
||||||
def status_color(self, status):
|
|
||||||
"""Return the color for a given status."""
|
|
||||||
return NETWORK_STATUS.get(status, {}).get('color', 'black')
|
|
||||||
|
|
||||||
def status_icon(self, status):
|
|
||||||
"""Return the name of icon for a given status."""
|
|
||||||
return NETWORK_STATUS.get(status, {}).get('icon', '')
|
|
||||||
|
|
||||||
def get_options(self):
|
|
||||||
"""Get connection options."""
|
|
||||||
return {
|
|
||||||
'hostname': self._hostname,
|
|
||||||
'port': self._port,
|
|
||||||
'ssl': 'on' if self._ssl else 'off',
|
|
||||||
'password': self._password,
|
|
||||||
'lines': str(self._lines),
|
|
||||||
}
|
|
||||||
|
|
||||||
def debug_print(self, *args, **kwargs):
|
|
||||||
"""Display a debug message."""
|
|
||||||
self.debug_lines.append((args, kwargs))
|
|
||||||
if self.debug_dialog:
|
|
||||||
self.debug_dialog.chat.display(*args, **kwargs)
|
|
||||||
|
|
||||||
def _debug_dialog_closed(self, result):
|
|
||||||
"""Called when debug dialog is closed."""
|
|
||||||
self.debug_dialog = None
|
|
||||||
|
|
||||||
def debug_input_text_sent(self, text):
|
|
||||||
"""Send debug buffer input to WeeChat."""
|
|
||||||
if self.network.is_connected():
|
|
||||||
text = str(text)
|
|
||||||
pos = text.find(')')
|
|
||||||
if text.startswith('(') and pos >= 0:
|
|
||||||
text = '(debug_%s)%s' % (text[1:pos], text[pos+1:])
|
|
||||||
else:
|
|
||||||
text = '(debug) %s' % text
|
|
||||||
self.network.debug_print(0, '<==', text, forcecolor='#AA0000')
|
|
||||||
self.network.send_to_weechat(text + '\n')
|
|
||||||
|
|
||||||
def open_debug_dialog(self):
|
|
||||||
"""Open a dialog with debug messages."""
|
|
||||||
if not self.debug_dialog:
|
|
||||||
self.debug_dialog = DebugDialog()
|
|
||||||
self.debug_dialog.input.textSent.connect(
|
|
||||||
self.debug_input_text_sent)
|
|
||||||
self.debug_dialog.finished.connect(self._debug_dialog_closed)
|
|
||||||
self.debug_dialog.display_lines(self.debug_lines)
|
|
||||||
self.debug_dialog.chat.scroll_bottom()
|
|
27
toxygen/third_party/qweechat/preferences.py.diff
vendored
27
toxygen/third_party/qweechat/preferences.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** preferences.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- preferences.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 20,30 ****
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Preferences dialog box."""
|
|
||||||
|
|
||||||
! from PySide6 import QtCore, QtWidgets as QtGui
|
|
||||||
|
|
||||||
|
|
||||||
class PreferencesDialog(QtGui.QDialog):
|
|
||||||
"""Preferences dialog."""
|
|
||||||
|
|
||||||
--- 20,30 ----
|
|
||||||
# along with QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Preferences dialog box."""
|
|
||||||
|
|
||||||
! from PyQt5 import QtCore, QtWidgets as QtGui
|
|
||||||
|
|
||||||
|
|
||||||
class PreferencesDialog(QtGui.QDialog):
|
|
||||||
"""Preferences dialog."""
|
|
||||||
|
|
57
toxygen/third_party/qweechat/preferences.py.dst
vendored
57
toxygen/third_party/qweechat/preferences.py.dst
vendored
@ -1,57 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# preferences.py - preferences dialog box
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""Preferences dialog box."""
|
|
||||||
|
|
||||||
from PySide6 import QtCore, QtWidgets as QtGui
|
|
||||||
|
|
||||||
|
|
||||||
class PreferencesDialog(QtGui.QDialog):
|
|
||||||
"""Preferences dialog."""
|
|
||||||
|
|
||||||
def __init__(self, *args):
|
|
||||||
QtGui.QDialog.__init__(*(self,) + args)
|
|
||||||
self.setModal(True)
|
|
||||||
self.setWindowTitle('Preferences')
|
|
||||||
|
|
||||||
close_button = QtGui.QPushButton('Close')
|
|
||||||
close_button.pressed.connect(self.close)
|
|
||||||
|
|
||||||
hbox = QtGui.QHBoxLayout()
|
|
||||||
hbox.addStretch(1)
|
|
||||||
hbox.addWidget(close_button)
|
|
||||||
hbox.addStretch(1)
|
|
||||||
|
|
||||||
vbox = QtGui.QVBoxLayout()
|
|
||||||
|
|
||||||
label = QtGui.QLabel('Not yet implemented!')
|
|
||||||
label.setAlignment(QtCore.Qt.AlignHCenter)
|
|
||||||
vbox.addWidget(label)
|
|
||||||
|
|
||||||
label = QtGui.QLabel('')
|
|
||||||
label.setAlignment(QtCore.Qt.AlignHCenter)
|
|
||||||
vbox.addWidget(label)
|
|
||||||
|
|
||||||
vbox.addLayout(hbox)
|
|
||||||
|
|
||||||
self.setLayout(vbox)
|
|
||||||
self.show()
|
|
27
toxygen/third_party/qweechat/qweechat.py.diff
vendored
27
toxygen/third_party/qweechat/qweechat.py.diff
vendored
@ -1,27 +0,0 @@
|
|||||||
*** qweechat.py.dst 2022-11-19 18:31:51.000000000 +0000
|
|
||||||
--- qweechat.py 2022-11-19 18:32:41.000000000 +0000
|
|
||||||
***************
|
|
||||||
*** 35,45 ****
|
|
||||||
|
|
||||||
import sys
|
|
||||||
import traceback
|
|
||||||
from pkg_resources import resource_filename
|
|
||||||
|
|
||||||
! from PySide6 import QtCore, QtGui, QtWidgets
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.about import AboutDialog
|
|
||||||
from qweechat.buffer import BufferListWidget, Buffer
|
|
||||||
from qweechat.connection import ConnectionDialog
|
|
||||||
--- 35,45 ----
|
|
||||||
|
|
||||||
import sys
|
|
||||||
import traceback
|
|
||||||
from pkg_resources import resource_filename
|
|
||||||
|
|
||||||
! from PyQt5 import QtCore, QtGui, QtWidgets
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.about import AboutDialog
|
|
||||||
from qweechat.buffer import BufferListWidget, Buffer
|
|
||||||
from qweechat.connection import ConnectionDialog
|
|
542
toxygen/third_party/qweechat/qweechat.py.dst
vendored
542
toxygen/third_party/qweechat/qweechat.py.dst
vendored
@ -1,542 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
#
|
|
||||||
# qweechat.py - WeeChat remote GUI using Qt toolkit
|
|
||||||
#
|
|
||||||
# Copyright (C) 2011-2022 Sébastien Helleu <flashcode@flashtux.org>
|
|
||||||
#
|
|
||||||
# This file is part of QWeeChat, a Qt remote GUI for WeeChat.
|
|
||||||
#
|
|
||||||
# QWeeChat 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.
|
|
||||||
#
|
|
||||||
# QWeeChat 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 QWeeChat. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
#
|
|
||||||
|
|
||||||
"""
|
|
||||||
QWeeChat is a WeeChat remote GUI using Qt toolkit.
|
|
||||||
|
|
||||||
It requires requires WeeChat 0.3.7 or newer, running on local or remote host.
|
|
||||||
"""
|
|
||||||
|
|
||||||
#
|
|
||||||
# History:
|
|
||||||
#
|
|
||||||
# 2011-05-27, Sébastien Helleu <flashcode@flashtux.org>:
|
|
||||||
# start dev
|
|
||||||
#
|
|
||||||
|
|
||||||
import sys
|
|
||||||
import traceback
|
|
||||||
from pkg_resources import resource_filename
|
|
||||||
|
|
||||||
from PySide6 import QtCore, QtGui, QtWidgets
|
|
||||||
|
|
||||||
from qweechat import config
|
|
||||||
from qweechat.about import AboutDialog
|
|
||||||
from qweechat.buffer import BufferListWidget, Buffer
|
|
||||||
from qweechat.connection import ConnectionDialog
|
|
||||||
from qweechat.network import Network, STATUS_DISCONNECTED
|
|
||||||
from qweechat.preferences import PreferencesDialog
|
|
||||||
from qweechat.weechat import protocol
|
|
||||||
|
|
||||||
|
|
||||||
APP_NAME = 'QWeeChat'
|
|
||||||
AUTHOR = 'Sébastien Helleu'
|
|
||||||
WEECHAT_SITE = 'https://weechat.org/'
|
|
||||||
|
|
||||||
|
|
||||||
class MainWindow(QtWidgets.QMainWindow):
|
|
||||||
"""Main window."""
|
|
||||||
|
|
||||||
def __init__(self, *args):
|
|
||||||
super().__init__(*args)
|
|
||||||
|
|
||||||
self.config = config.read()
|
|
||||||
|
|
||||||
self.resize(1000, 600)
|
|
||||||
self.setWindowTitle(APP_NAME)
|
|
||||||
|
|
||||||
self.about_dialog = None
|
|
||||||
self.connection_dialog = None
|
|
||||||
self.preferences_dialog = None
|
|
||||||
|
|
||||||
# network
|
|
||||||
self.network = Network()
|
|
||||||
self.network.statusChanged.connect(self._network_status_changed)
|
|
||||||
self.network.messageFromWeechat.connect(self._network_weechat_msg)
|
|
||||||
|
|
||||||
# list of buffers
|
|
||||||
self.list_buffers = BufferListWidget()
|
|
||||||
self.list_buffers.currentRowChanged.connect(self._buffer_switch)
|
|
||||||
|
|
||||||
# default buffer
|
|
||||||
self.buffers = [Buffer()]
|
|
||||||
self.stacked_buffers = QtWidgets.QStackedWidget()
|
|
||||||
self.stacked_buffers.addWidget(self.buffers[0].widget)
|
|
||||||
|
|
||||||
# splitter with buffers + chat/input
|
|
||||||
splitter = QtWidgets.QSplitter()
|
|
||||||
splitter.addWidget(self.list_buffers)
|
|
||||||
splitter.addWidget(self.stacked_buffers)
|
|
||||||
|
|
||||||
self.setCentralWidget(splitter)
|
|
||||||
|
|
||||||
if self.config.getboolean('look', 'statusbar'):
|
|
||||||
self.statusBar().visible = True
|
|
||||||
|
|
||||||
# actions for menu and toolbar
|
|
||||||
actions_def = {
|
|
||||||
'connect': [
|
|
||||||
'network-connect.png',
|
|
||||||
'Connect to WeeChat',
|
|
||||||
'Ctrl+O',
|
|
||||||
self.open_connection_dialog,
|
|
||||||
],
|
|
||||||
'disconnect': [
|
|
||||||
'network-disconnect.png',
|
|
||||||
'Disconnect from WeeChat',
|
|
||||||
'Ctrl+D',
|
|
||||||
self.network.disconnect_weechat,
|
|
||||||
],
|
|
||||||
'debug': [
|
|
||||||
'edit-find.png',
|
|
||||||
'Open debug console window',
|
|
||||||
'Ctrl+B',
|
|
||||||
self.network.open_debug_dialog,
|
|
||||||
],
|
|
||||||
'preferences': [
|
|
||||||
'preferences-other.png',
|
|
||||||
'Change preferences',
|
|
||||||
'Ctrl+P',
|
|
||||||
self.open_preferences_dialog,
|
|
||||||
],
|
|
||||||
'about': [
|
|
||||||
'help-about.png',
|
|
||||||
'About QWeeChat',
|
|
||||||
'Ctrl+H',
|
|
||||||
self.open_about_dialog,
|
|
||||||
],
|
|
||||||
'save connection': [
|
|
||||||
'document-save.png',
|
|
||||||
'Save connection configuration',
|
|
||||||
'Ctrl+S',
|
|
||||||
self.save_connection,
|
|
||||||
],
|
|
||||||
'quit': [
|
|
||||||
'application-exit.png',
|
|
||||||
'Quit application',
|
|
||||||
'Ctrl+Q',
|
|
||||||
self.close,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
self.actions = {}
|
|
||||||
for name, action in list(actions_def.items()):
|
|
||||||
self.actions[name] = QtGui.QAction(
|
|
||||||
QtGui.QIcon(
|
|
||||||
resource_filename(__name__, 'data/icons/%s' % action[0])),
|
|
||||||
name.capitalize(), self)
|
|
||||||
self.actions[name].setToolTip(f'{action[1]} ({action[2]})')
|
|
||||||
self.actions[name].setShortcut(action[2])
|
|
||||||
self.actions[name].triggered.connect(action[3])
|
|
||||||
|
|
||||||
# menu
|
|
||||||
self.menu = self.menuBar()
|
|
||||||
menu_file = self.menu.addMenu('&File')
|
|
||||||
menu_file.addActions([self.actions['connect'],
|
|
||||||
self.actions['disconnect'],
|
|
||||||
self.actions['preferences'],
|
|
||||||
self.actions['save connection'],
|
|
||||||
self.actions['quit']])
|
|
||||||
menu_window = self.menu.addMenu('&Window')
|
|
||||||
menu_window.addAction(self.actions['debug'])
|
|
||||||
menu_help = self.menu.addMenu('&Help')
|
|
||||||
menu_help.addAction(self.actions['about'])
|
|
||||||
self.network_status = QtWidgets.QLabel()
|
|
||||||
self.network_status.setFixedHeight(20)
|
|
||||||
self.network_status.setFixedWidth(200)
|
|
||||||
self.network_status.setContentsMargins(0, 0, 10, 0)
|
|
||||||
self.network_status.setAlignment(QtCore.Qt.AlignRight)
|
|
||||||
if hasattr(self.menu, 'setCornerWidget'):
|
|
||||||
self.menu.setCornerWidget(self.network_status,
|
|
||||||
QtCore.Qt.TopRightCorner)
|
|
||||||
self.network_status_set(STATUS_DISCONNECTED)
|
|
||||||
|
|
||||||
# toolbar
|
|
||||||
toolbar = self.addToolBar('toolBar')
|
|
||||||
toolbar.setToolButtonStyle(QtCore.Qt.ToolButtonTextUnderIcon)
|
|
||||||
toolbar.addActions([self.actions['connect'],
|
|
||||||
self.actions['disconnect'],
|
|
||||||
self.actions['debug'],
|
|
||||||
self.actions['preferences'],
|
|
||||||
self.actions['about'],
|
|
||||||
self.actions['quit']])
|
|
||||||
|
|
||||||
self.buffers[0].widget.input.setFocus()
|
|
||||||
|
|
||||||
# open debug dialog
|
|
||||||
if self.config.getboolean('look', 'debug'):
|
|
||||||
self.network.open_debug_dialog()
|
|
||||||
|
|
||||||
# auto-connect to relay
|
|
||||||
if self.config.getboolean('relay', 'autoconnect'):
|
|
||||||
self.network.connect_weechat(
|
|
||||||
hostname=self.config.get('relay', 'hostname', fallback=''),
|
|
||||||
port=self.config.get('relay', 'port', fallback=''),
|
|
||||||
ssl=self.config.getboolean('relay', 'ssl', fallback=''),
|
|
||||||
password=self.config.get('relay', 'password', fallback=''),
|
|
||||||
totp=None,
|
|
||||||
lines=self.config.get('relay', 'lines', fallback=''),
|
|
||||||
)
|
|
||||||
|
|
||||||
self.show()
|
|
||||||
|
|
||||||
def _buffer_switch(self, index):
|
|
||||||
"""Switch to a buffer."""
|
|
||||||
if index >= 0:
|
|
||||||
self.stacked_buffers.setCurrentIndex(index)
|
|
||||||
self.stacked_buffers.widget(index).input.setFocus()
|
|
||||||
|
|
||||||
def buffer_input(self, full_name, text):
|
|
||||||
"""Send buffer input to WeeChat."""
|
|
||||||
if self.network.is_connected():
|
|
||||||
message = 'input %s %s\n' % (full_name, text)
|
|
||||||
self.network.send_to_weechat(message)
|
|
||||||
self.network.debug_print(0, '<==', message, forcecolor='#AA0000')
|
|
||||||
|
|
||||||
def open_preferences_dialog(self):
|
|
||||||
"""Open a dialog with preferences."""
|
|
||||||
# TODO: implement the preferences dialog box
|
|
||||||
self.preferences_dialog = PreferencesDialog(self)
|
|
||||||
|
|
||||||
def save_connection(self):
|
|
||||||
"""Save connection configuration."""
|
|
||||||
if self.network:
|
|
||||||
options = self.network.get_options()
|
|
||||||
for option in options:
|
|
||||||
self.config.set('relay', option, options[option])
|
|
||||||
|
|
||||||
def open_about_dialog(self):
|
|
||||||
"""Open a dialog with info about QWeeChat."""
|
|
||||||
self.about_dialog = AboutDialog(APP_NAME, AUTHOR, WEECHAT_SITE, self)
|
|
||||||
|
|
||||||
def open_connection_dialog(self):
|
|
||||||
"""Open a dialog with connection settings."""
|
|
||||||
values = {}
|
|
||||||
for option in ('hostname', 'port', 'ssl', 'password', 'lines'):
|
|
||||||
values[option] = self.config.get('relay', option, fallback='')
|
|
||||||
self.connection_dialog = ConnectionDialog(values, self)
|
|
||||||
self.connection_dialog.dialog_buttons.accepted.connect(
|
|
||||||
self.connect_weechat)
|
|
||||||
|
|
||||||
def connect_weechat(self):
|
|
||||||
"""Connect to WeeChat."""
|
|
||||||
self.network.connect_weechat(
|
|
||||||
hostname=self.connection_dialog.fields['hostname'].text(),
|
|
||||||
port=self.connection_dialog.fields['port'].text(),
|
|
||||||
ssl=self.connection_dialog.fields['ssl'].isChecked(),
|
|
||||||
password=self.connection_dialog.fields['password'].text(),
|
|
||||||
totp=self.connection_dialog.fields['totp'].text(),
|
|
||||||
lines=int(self.connection_dialog.fields['lines'].text()),
|
|
||||||
)
|
|
||||||
self.connection_dialog.close()
|
|
||||||
|
|
||||||
def _network_status_changed(self, status, extra):
|
|
||||||
"""Called when the network status has changed."""
|
|
||||||
if self.config.getboolean('look', 'statusbar'):
|
|
||||||
self.statusBar().showMessage(status)
|
|
||||||
self.network.debug_print(0, '', status, forcecolor='#0000AA')
|
|
||||||
self.network_status_set(status)
|
|
||||||
|
|
||||||
def network_status_set(self, status):
|
|
||||||
"""Set the network status."""
|
|
||||||
pal = self.network_status.palette()
|
|
||||||
pal.setColor(self.network_status.foregroundRole(),
|
|
||||||
self.network.status_color(status))
|
|
||||||
ssl = ' (SSL)' if status != STATUS_DISCONNECTED \
|
|
||||||
and self.network.is_ssl() else ''
|
|
||||||
self.network_status.setPalette(pal)
|
|
||||||
icon = self.network.status_icon(status)
|
|
||||||
if icon:
|
|
||||||
self.network_status.setText(
|
|
||||||
'<img src="%s"> %s' %
|
|
||||||
(resource_filename(__name__, 'data/icons/%s' % icon),
|
|
||||||
self.network.status_label(status) + ssl))
|
|
||||||
else:
|
|
||||||
self.network_status.setText(status.capitalize())
|
|
||||||
if status == STATUS_DISCONNECTED:
|
|
||||||
self.actions['connect'].setEnabled(True)
|
|
||||||
self.actions['disconnect'].setEnabled(False)
|
|
||||||
else:
|
|
||||||
self.actions['connect'].setEnabled(False)
|
|
||||||
self.actions['disconnect'].setEnabled(True)
|
|
||||||
|
|
||||||
def _network_weechat_msg(self, message):
|
|
||||||
"""Called when a message is received from WeeChat."""
|
|
||||||
self.network.debug_print(
|
|
||||||
0, '==>',
|
|
||||||
'message (%d bytes):\n%s'
|
|
||||||
% (len(message),
|
|
||||||
protocol.hex_and_ascii(message.data(), 20)),
|
|
||||||
forcecolor='#008800',
|
|
||||||
)
|
|
||||||
try:
|
|
||||||
proto = protocol.Protocol()
|
|
||||||
message = proto.decode(message.data())
|
|
||||||
if message.uncompressed:
|
|
||||||
self.network.debug_print(
|
|
||||||
0, '==>',
|
|
||||||
'message uncompressed (%d bytes):\n%s'
|
|
||||||
% (message.size_uncompressed,
|
|
||||||
protocol.hex_and_ascii(message.uncompressed, 20)),
|
|
||||||
forcecolor='#008800')
|
|
||||||
self.network.debug_print(0, '', 'Message: %s' % message)
|
|
||||||
self.parse_message(message)
|
|
||||||
except Exception: # noqa: E722
|
|
||||||
print('Error while decoding message from WeeChat:\n%s'
|
|
||||||
% traceback.format_exc())
|
|
||||||
self.network.disconnect_weechat()
|
|
||||||
|
|
||||||
def _parse_handshake(self, message):
|
|
||||||
"""Parse a WeeChat message with handshake response."""
|
|
||||||
for obj in message.objects:
|
|
||||||
if obj.objtype != 'htb':
|
|
||||||
continue
|
|
||||||
self.network.init_with_handshake(obj.value)
|
|
||||||
break
|
|
||||||
|
|
||||||
def _parse_listbuffers(self, message):
|
|
||||||
"""Parse a WeeChat message with list of buffers."""
|
|
||||||
for obj in message.objects:
|
|
||||||
if obj.objtype != 'hda' or obj.value['path'][-1] != 'buffer':
|
|
||||||
continue
|
|
||||||
self.list_buffers.clear()
|
|
||||||
while self.stacked_buffers.count() > 0:
|
|
||||||
buf = self.stacked_buffers.widget(0)
|
|
||||||
self.stacked_buffers.removeWidget(buf)
|
|
||||||
self.buffers = []
|
|
||||||
for item in obj.value['items']:
|
|
||||||
buf = self.create_buffer(item)
|
|
||||||
self.insert_buffer(len(self.buffers), buf)
|
|
||||||
self.list_buffers.setCurrentRow(0)
|
|
||||||
self.buffers[0].widget.input.setFocus()
|
|
||||||
|
|
||||||
def _parse_line(self, message):
|
|
||||||
"""Parse a WeeChat message with a buffer line."""
|
|
||||||
for obj in message.objects:
|
|
||||||
lines = []
|
|
||||||
if obj.objtype != 'hda' or obj.value['path'][-1] != 'line_data':
|
|
||||||
continue
|
|
||||||
for item in obj.value['items']:
|
|
||||||
if message.msgid == 'listlines':
|
|
||||||
ptrbuf = item['__path'][0]
|
|
||||||
else:
|
|
||||||
ptrbuf = item['buffer']
|
|
||||||
index = [i for i, b in enumerate(self.buffers)
|
|
||||||
if b.pointer() == ptrbuf]
|
|
||||||
if index:
|
|
||||||
lines.append(
|
|
||||||
(index[0],
|
|
||||||
(item['date'], item['prefix'],
|
|
||||||
item['message']))
|
|
||||||
)
|
|
||||||
if message.msgid == 'listlines':
|
|
||||||
lines.reverse()
|
|
||||||
for line in lines:
|
|
||||||
self.buffers[line[0]].widget.chat.display(*line[1])
|
|
||||||
|
|
||||||
def _parse_nicklist(self, message):
|
|
||||||
"""Parse a WeeChat message with a buffer nicklist."""
|
|
||||||
buffer_refresh = {}
|
|
||||||
for obj in message.objects:
|
|
||||||
if obj.objtype != 'hda' or \
|
|
||||||
obj.value['path'][-1] != 'nicklist_item':
|
|
||||||
continue
|
|
||||||
group = '__root'
|
|
||||||
for item in obj.value['items']:
|
|
||||||
index = [i for i, b in enumerate(self.buffers)
|
|
||||||
if b.pointer() == item['__path'][0]]
|
|
||||||
if index:
|
|
||||||
if not index[0] in buffer_refresh:
|
|
||||||
self.buffers[index[0]].nicklist = {}
|
|
||||||
buffer_refresh[index[0]] = True
|
|
||||||
if item['group']:
|
|
||||||
group = item['name']
|
|
||||||
self.buffers[index[0]].nicklist_add_item(
|
|
||||||
group, item['group'], item['prefix'], item['name'],
|
|
||||||
item['visible'])
|
|
||||||
for index in buffer_refresh:
|
|
||||||
self.buffers[index].nicklist_refresh()
|
|
||||||
|
|
||||||
def _parse_nicklist_diff(self, message):
|
|
||||||
"""Parse a WeeChat message with a buffer nicklist diff."""
|
|
||||||
buffer_refresh = {}
|
|
||||||
for obj in message.objects:
|
|
||||||
if obj.objtype != 'hda' or \
|
|
||||||
obj.value['path'][-1] != 'nicklist_item':
|
|
||||||
continue
|
|
||||||
group = '__root'
|
|
||||||
for item in obj.value['items']:
|
|
||||||
index = [i for i, b in enumerate(self.buffers)
|
|
||||||
if b.pointer() == item['__path'][0]]
|
|
||||||
if not index:
|
|
||||||
continue
|
|
||||||
buffer_refresh[index[0]] = True
|
|
||||||
if item['_diff'] == ord('^'):
|
|
||||||
group = item['name']
|
|
||||||
elif item['_diff'] == ord('+'):
|
|
||||||
self.buffers[index[0]].nicklist_add_item(
|
|
||||||
group, item['group'], item['prefix'], item['name'],
|
|
||||||
item['visible'])
|
|
||||||
elif item['_diff'] == ord('-'):
|
|
||||||
self.buffers[index[0]].nicklist_remove_item(
|
|
||||||
group, item['group'], item['name'])
|
|
||||||
elif item['_diff'] == ord('*'):
|
|
||||||
self.buffers[index[0]].nicklist_update_item(
|
|
||||||
group, item['group'], item['prefix'], item['name'],
|
|
||||||
item['visible'])
|
|
||||||
for index in buffer_refresh:
|
|
||||||
self.buffers[index].nicklist_refresh()
|
|
||||||
|
|
||||||
def _parse_buffer_opened(self, message):
|
|
||||||
"""Parse a WeeChat message with a new buffer (opened)."""
|
|
||||||
for obj in message.objects:
|
|
||||||
if obj.objtype != 'hda' or obj.value['path'][-1] != 'buffer':
|
|
||||||
continue
|
|
||||||
for item in obj.value['items']:
|
|
||||||
buf = self.create_buffer(item)
|
|
||||||
index = self.find_buffer_index_for_insert(item['next_buffer'])
|
|
||||||
self.insert_buffer(index, buf)
|
|
||||||
|
|
||||||
def _parse_buffer(self, message):
|
|
||||||
"""Parse a WeeChat message with a buffer event
|
|
||||||
(anything except a new buffer).
|
|
||||||
"""
|
|
||||||
for obj in message.objects:
|
|
||||||
if obj.objtype != 'hda' or obj.value['path'][-1] != 'buffer':
|
|
||||||
continue
|
|
||||||
for item in obj.value['items']:
|
|
||||||
index = [i for i, b in enumerate(self.buffers)
|
|
||||||
if b.pointer() == item['__path'][0]]
|
|
||||||
if not index:
|
|
||||||
continue
|
|
||||||
index = index[0]
|
|
||||||
if message.msgid == '_buffer_type_changed':
|
|
||||||
self.buffers[index].data['type'] = item['type']
|
|
||||||
elif message.msgid in ('_buffer_moved', '_buffer_merged',
|
|
||||||
'_buffer_unmerged'):
|
|
||||||
buf = self.buffers[index]
|
|
||||||
buf.data['number'] = item['number']
|
|
||||||
self.remove_buffer(index)
|
|
||||||
index2 = self.find_buffer_index_for_insert(
|
|
||||||
item['next_buffer'])
|
|
||||||
self.insert_buffer(index2, buf)
|
|
||||||
elif message.msgid == '_buffer_renamed':
|
|
||||||
self.buffers[index].data['full_name'] = item['full_name']
|
|
||||||
self.buffers[index].data['short_name'] = item['short_name']
|
|
||||||
elif message.msgid == '_buffer_title_changed':
|
|
||||||
self.buffers[index].data['title'] = item['title']
|
|
||||||
self.buffers[index].update_title()
|
|
||||||
elif message.msgid == '_buffer_cleared':
|
|
||||||
self.buffers[index].widget.chat.clear()
|
|
||||||
elif message.msgid.startswith('_buffer_localvar_'):
|
|
||||||
self.buffers[index].data['local_variables'] = \
|
|
||||||
item['local_variables']
|
|
||||||
self.buffers[index].update_prompt()
|
|
||||||
elif message.msgid == '_buffer_closing':
|
|
||||||
self.remove_buffer(index)
|
|
||||||
|
|
||||||
def parse_message(self, message):
|
|
||||||
"""Parse a WeeChat message."""
|
|
||||||
if message.msgid.startswith('debug'):
|
|
||||||
self.network.debug_print(0, '', '(debug message, ignored)')
|
|
||||||
elif message.msgid == 'handshake':
|
|
||||||
self._parse_handshake(message)
|
|
||||||
elif message.msgid == 'listbuffers':
|
|
||||||
self._parse_listbuffers(message)
|
|
||||||
elif message.msgid in ('listlines', '_buffer_line_added'):
|
|
||||||
self._parse_line(message)
|
|
||||||
elif message.msgid in ('_nicklist', 'nicklist'):
|
|
||||||
self._parse_nicklist(message)
|
|
||||||
elif message.msgid == '_nicklist_diff':
|
|
||||||
self._parse_nicklist_diff(message)
|
|
||||||
elif message.msgid == '_buffer_opened':
|
|
||||||
self._parse_buffer_opened(message)
|
|
||||||
elif message.msgid.startswith('_buffer_'):
|
|
||||||
self._parse_buffer(message)
|
|
||||||
elif message.msgid == '_upgrade':
|
|
||||||
self.network.desync_weechat()
|
|
||||||
elif message.msgid == '_upgrade_ended':
|
|
||||||
self.network.sync_weechat()
|
|
||||||
else:
|
|
||||||
print(f"Unknown message with id {message.msgid}")
|
|
||||||
|
|
||||||
def create_buffer(self, item):
|
|
||||||
"""Create a new buffer."""
|
|
||||||
buf = Buffer(item)
|
|
||||||
buf.bufferInput.connect(self.buffer_input)
|
|
||||||
buf.widget.input.bufferSwitchPrev.connect(
|
|
||||||
self.list_buffers.switch_prev_buffer)
|
|
||||||
buf.widget.input.bufferSwitchNext.connect(
|
|
||||||
self.list_buffers.switch_next_buffer)
|
|
||||||
return buf
|
|
||||||
|
|
||||||
def insert_buffer(self, index, buf):
|
|
||||||
"""Insert a buffer in list."""
|
|
||||||
self.buffers.insert(index, buf)
|
|
||||||
self.list_buffers.insertItem(index, '%s'
|
|
||||||
% (buf.data['local_variables']['name']))
|
|
||||||
self.stacked_buffers.insertWidget(index, buf.widget)
|
|
||||||
|
|
||||||
def remove_buffer(self, index):
|
|
||||||
"""Remove a buffer."""
|
|
||||||
if self.list_buffers.currentRow == index and index > 0:
|
|
||||||
self.list_buffers.setCurrentRow(index - 1)
|
|
||||||
self.list_buffers.takeItem(index)
|
|
||||||
self.stacked_buffers.removeWidget(self.stacked_buffers.widget(index))
|
|
||||||
self.buffers.pop(index)
|
|
||||||
|
|
||||||
def find_buffer_index_for_insert(self, next_buffer):
|
|
||||||
"""Find position to insert a buffer in list."""
|
|
||||||
index = -1
|
|
||||||
if next_buffer == '0x0':
|
|
||||||
index = len(self.buffers)
|
|
||||||
else:
|
|
||||||
index = [i for i, b in enumerate(self.buffers)
|
|
||||||
if b.pointer() == next_buffer]
|
|
||||||
if index:
|
|
||||||
index = index[0]
|
|
||||||
if index < 0:
|
|
||||||
print('Warning: unable to find position for buffer, using end of '
|
|
||||||
'list by default')
|
|
||||||
index = len(self.buffers)
|
|
||||||
return index
|
|
||||||
|
|
||||||
def closeEvent(self, event):
|
|
||||||
"""Called when QWeeChat window is closed."""
|
|
||||||
self.network.disconnect_weechat()
|
|
||||||
if self.network.debug_dialog:
|
|
||||||
self.network.debug_dialog.close()
|
|
||||||
config.write(self.config)
|
|
||||||
QtWidgets.QMainWindow.closeEvent(self, event)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
app = QtWidgets.QApplication(sys.argv)
|
|
||||||
app.setStyle(QtWidgets.QStyleFactory.create('Cleanlooks'))
|
|
||||||
app.setWindowIcon(QtGui.QIcon(
|
|
||||||
resource_filename(__name__, 'data/icons/weechat.png')))
|
|
||||||
main_win = MainWindow()
|
|
||||||
main_win.show()
|
|
||||||
sys.exit(app.exec_())
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
main()
|
|
Loading…
Reference in New Issue
Block a user