# main.py # # Copyright 2024 Lucas Fryzek # # 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 . # # SPDX-License-Identifier: GPL-3.0-or-later import sys import os import gi import traceback gi.require_version('Gtk', '4.0') gi.require_version('Adw', '1') from gi.repository import Gtk, Gio, Adw from .window import WeegtkWindow from weegtk import network from weegtk import protocol from weegtk import config from .chat import WeegtkChat from .preferences import WeegtkPreferences from .welcome import WeegtkWelcome class WeegtkApplication(Adw.Application): """The main application singleton class.""" def __init__(self): super().__init__(application_id='com.fryzekconcepts.weegtk', flags=Gio.ApplicationFlags.DEFAULT_FLAGS) self.create_action('quit', lambda *_: self.quit(), ['q']) self.create_action('about', self.on_about_action) self.create_action('preferences', self.on_preferences_action) self.create_action('disconnect', self.disconnect) self.network = network.Network() self.network.connect("status_changed", self._network_status_changed) self.network.connect("message_from_weechat", self._network_weechat_msg) self.buffers = [] # We need to keep track of buffer pages because some versions of Gtk # crash. I believe its related to this: # https://gitlab.gnome.org/GNOME/gtk/-/issues/5917 self.pages = [] conf = config.read() if config.str_to_bool(conf["relay"]["autoconnect"]): self.connect_to_weechat() def connect_to_weechat(self, *args): conf = config.read() self.network.connect_weechat_ssh(conf["ssh"]["host"], conf["ssh"]["port"], conf["ssh"]["username"], conf["ssh"]["key"], conf["relay"]["hostname"], conf["relay"]["port"], conf["relay"]["password"]) def clear_buffers(self): for buffer in self.buffers: res = self.props.active_window.stack.remove(buffer) self.buffers = [] self.pages = [] def remove_buffer(self, index): buf = self.buffers[index] self.props.active_window.stack.remove(buf) self.buffers.pop(index) self.pages.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: bufs = [i for i, b in enumerate(self.buffers) if b.pointer() == next_buffer] if bufs: index = bufs[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 disconnect(self, *args): self.network.disconnect_weechat() def handle_disconnect(self): self.props.active_window.set_page(self.welcome_page) self.clear_buffers() def _network_status_changed(self, source_object, status, extra): """Called when the network status has changed.""" if status == network.STATUS_DISCONNECTED: self.handle_disconnect() def _network_weechat_msg(self, source_object, message): """Called when a message is received from WeeChat.""" try: proto = protocol.Protocol() message = proto.decode(message.get_data()) # TODO figure this out #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) #print(f"parsed message is f{message}") self.parse_message(message) except: print(f"Error while decoding message from WeeChat:\n{traceback.format_exc()}") self.disconnect() def do_activate(self): """Called when the application is activated. We raise the application's main window, creating it if necessary. """ win = self.props.active_window if not win: win = WeegtkWindow(application=self) welcome = WeegtkWelcome() welcome.connect("connect-network", self.connect_to_weechat) self.welcome_page = win.stack.add_named(welcome, "welcome") win.set_page(self.welcome_page) win.present() def on_about_action(self, *args): """Callback for the app.about action.""" about = Adw.AboutDialog(application_name='weegtk', application_icon='com.fryzekconcepts.weegtk', developer_name='Lucas Fryzek', version='0.1.0', developers=['Lucas Fryzek '], copyright='© 2024 Lucas Fryzek') # Translators: Replace "translator-credits" with your name/username, and optionally an email or URL. about.set_translator_credits(_('translator-credits')) about.add_credit_section("QWeechat", ["Sébastien Helleu "]) about.present(self.props.active_window) def on_preferences_action(self, widget, _): """Callback for the app.preferences action.""" pref = WeegtkPreferences() pref.present(self.props.active_window) def create_action(self, name, callback, shortcuts=None): """Add an application action. Args: name: the name of the action callback: the function to be called when the action is activated shortcuts: an optional list of accelerators """ action = Gio.SimpleAction.new(name, None) action.connect("activate", callback) self.add_action(action) if shortcuts: self.set_accels_for_action(f"app.{name}", shortcuts) 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.""" # TODO handle gui for this for obj in message.objects: if obj.objtype != 'hda' or obj.value['path'][-1] != 'buffer': continue self.clear_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]].display(*line[1]) def _parse_nicklist(self, message): """Parse a WeeChat message with a buffer nicklist.""" # TODO implement nicklist for chat # 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': # TODO figure out what to do when we get a title change # as grep plugin seems to override title with results # which can break UI self.buffers[index].data['title'] = item['title'] ##self.pages[index].set_title(item['title']) elif message.msgid == '_buffer_cleared': self.buffers[index].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() elif message.msgid == "_pong": # For now don't do anything with pong messages pass else: print(f"Unknown message with id {message.msgid}") def create_buffer(self, item): """Create a new buffer.""" buf = WeegtkChat(data=item) buf.connect("buffer_input", self.buffer_input) return buf def insert_buffer(self, index, buf): """Insert a buffer in list.""" self.buffers.insert(index, buf) buf_name = buf.data['short_name'] # Only make chats visible on main screen if True or buf.is_chat(): page = self.props.active_window.stack.add_titled(buf, name=buf_name, title=buf_name) if buf.data["full_name"] == "core.weechat": self.props.active_window.set_page(page) self.pages.insert(index, page) def buffer_input(self, source_object, full_name, text): message = f"input {full_name} {text}\n" self.network.send_to_weechat(message) def main(version): """The application's entry point.""" app = WeegtkApplication() return app.run(sys.argv)