From d3d268004bc063e94eff67a55c0bc3827631a846 Mon Sep 17 00:00:00 2001 From: Storm Dragon Date: Tue, 25 Mar 2025 19:41:57 -0400 Subject: [PATCH] Start migration to pluggy for plugins. --- src/cthulhu/cthulhu.py | 5 +- src/cthulhu/cthulhuVersion.py | 2 +- src/cthulhu/plugin.py | 426 ++++++++-- src/cthulhu/plugin_system_manager.py | 1109 +++++++++++++++++++++----- 4 files changed, 1298 insertions(+), 244 deletions(-) diff --git a/src/cthulhu/cthulhu.py b/src/cthulhu/cthulhu.py index ba0a3b4..cf77704 100644 --- a/src/cthulhu/cthulhu.py +++ b/src/cthulhu/cthulhu.py @@ -74,7 +74,8 @@ from .ax_object import AXObject from .ax_utilities import AXUtilities from .input_event import BrailleEvent from . import cmdnames -from . import plugin_system_manager +from . import plugin_system_manager # This will now be your pluggy-based implementation +from .plugin_api_helper import APIHelper # New import for the separated APIHelper from . import guilabels from . import acss from . import text_attribute_names @@ -920,7 +921,7 @@ class Cthulhu(GObject.Object): GObject.Object.__init__(self) # add members self.resourceManager = resource_manager.ResourceManager(self) - self.APIHelper = plugin_system_manager.APIHelper(self) + self.APIHelper = APIHelper(self) self.eventManager = _eventManager self.settingsManager = _settingsManager self.scriptManager = _scriptManager diff --git a/src/cthulhu/cthulhuVersion.py b/src/cthulhu/cthulhuVersion.py index 6bf1831..07ebed7 100644 --- a/src/cthulhu/cthulhuVersion.py +++ b/src/cthulhu/cthulhuVersion.py @@ -23,5 +23,5 @@ # Fork of Orca Screen Reader (GNOME) # Original source: https://gitlab.gnome.org/GNOME/orca -version = "2024.12.23" +version = "2025.03.25" codeName = "testing" diff --git a/src/cthulhu/plugin.py b/src/cthulhu/plugin.py index b5d127b..3564e71 100644 --- a/src/cthulhu/plugin.py +++ b/src/cthulhu/plugin.py @@ -23,15 +23,32 @@ # Fork of Orca Screen Reader (GNOME) # Original source: https://gitlab.gnome.org/GNOME/orca -import os, inspect -import gi -from gi.repository import GObject +"""Base class for Cthulhu plugins using pluggy.""" -class Plugin(): - #__gtype_name__ = 'BasePlugin' +import os +import inspect +import logging +# Import pluggy for hook specifications +try: + import pluggy + cthulhu_hookimpl = pluggy.HookimplMarker("cthulhu") +except ImportError: + # Fallback if pluggy is not available + def cthulhu_hookimpl(func=None, **kwargs): + """Dummy decorator for systems without pluggy.""" + if func is None: + return lambda f: f + return func + +logger = logging.getLogger(__name__) + +class Plugin: + """Base class for Cthulhu plugins.""" + def __init__(self, *args, **kwargs): - self.API = None + """Initialize the plugin with default attributes.""" + self.app = None self.pluginInfo = None self.moduleDir = '' self.hidden = False @@ -48,19 +65,46 @@ class Plugin(): self.helpUri = '' self.dataDir = '' self.translationContext = None + self.dynamicApiManager = None + self.signalManager = None + def setApp(self, app): + """Set the application reference. + + Args: + app: The Cthulhu application instance. + """ self.app = app self.dynamicApiManager = app.getDynamicApiManager() self.signalManager = app.getSignalManager() def getApp(self): + """Get the application reference. + + Returns: + The Cthulhu application instance. + """ return self.app + def setPluginInfo(self, pluginInfo): + """Set the plugin information. + + Args: + pluginInfo: The plugin information object. + """ self.pluginInfo = pluginInfo self.updatePluginInfoAttributes() + def getPluginInfo(self): + """Get the plugin information. + + Returns: + The plugin information object. + """ return self.pluginInfo + def updatePluginInfoAttributes(self): + """Update plugin attributes from the plugin information.""" self.moduleDir = '' self.hidden = False self.moduleName = '' @@ -75,44 +119,82 @@ class Plugin(): self.dependencies = False self.helpUri = '' self.dataDir = '' + pluginInfo = self.getPluginInfo() - if pluginInfo == None: + if pluginInfo is None: return - self.moduleName = self.getApp().getPluginSystemManager().getPluginModuleName(pluginInfo) - self.name = self.getApp().getPluginSystemManager().getPluginName(pluginInfo) - self.version = self.getApp().getPluginSystemManager().getPluginVersion(pluginInfo) - self.moduleDir = self.getApp().getPluginSystemManager().getPluginModuleDir(pluginInfo) - self.buildIn = self.getApp().getPluginSystemManager().isPluginBuildIn(pluginInfo) - self.description = self.getApp().getPluginSystemManager().getPluginDescription(pluginInfo) - self.hidden = self.getApp().getPluginSystemManager().isPluginHidden(pluginInfo) - self.website = self.getApp().getPluginSystemManager().getPluginWebsite(pluginInfo) - self.authors = self.getApp().getPluginSystemManager().getPluginAuthors(pluginInfo) - self.iconName = self.getApp().getPluginSystemManager().getPluginIconName(pluginInfo) - self.copyright = self.getApp().getPluginSystemManager().getPluginCopyright(pluginInfo) - self.dependencies = self.getApp().getPluginSystemManager().getPluginDependencies(pluginInfo) - - #settings = self.getApp().getPluginSystemManager().getPluginSettings(pluginInfo) - #hasDependencies = self.getApp().getPluginSystemManager().hasPluginDependency(pluginInfo) - - #externalData = self.getApp().getPluginSystemManager().getPluginExternalData(pluginInfo) - self.helpUri = self.getApp().getPluginSystemManager().getPlugingetHelpUri(pluginInfo) - self.dataDir = self.getApp().getPluginSystemManager().getPluginDataDir(pluginInfo) + + if hasattr(self.app, 'getPluginSystemManager'): + plugin_system = self.app.getPluginSystemManager() + + self.moduleName = plugin_system.getPluginModuleName(pluginInfo) + self.name = plugin_system.getPluginName(pluginInfo) + self.version = plugin_system.getPluginVersion(pluginInfo) + self.moduleDir = plugin_system.getPluginModuleDir(pluginInfo) + self.buildIn = plugin_system.isPluginBuildIn(pluginInfo) + self.description = plugin_system.getPluginDescription(pluginInfo) + self.hidden = plugin_system.isPluginHidden(pluginInfo) + self.website = plugin_system.getPluginWebsite(pluginInfo) + self.authors = plugin_system.getPluginAuthors(pluginInfo) + self.iconName = plugin_system.getPluginIconName(pluginInfo) + self.copyright = plugin_system.getPluginCopyright(pluginInfo) + self.dependencies = plugin_system.getPluginDependencies(pluginInfo) + self.helpUri = plugin_system.getPlugingetHelpUri(pluginInfo) + self.dataDir = plugin_system.getPluginDataDir(pluginInfo) + else: + # Direct attribute access for pluggy-based plugins + self.moduleName = getattr(pluginInfo, 'module_name', '') + self.name = getattr(pluginInfo, 'name', '') + self.version = getattr(pluginInfo, 'version', '') + self.moduleDir = getattr(pluginInfo, 'module_dir', '') + self.buildIn = getattr(pluginInfo, 'builtin', False) + self.description = getattr(pluginInfo, 'description', '') + self.hidden = getattr(pluginInfo, 'hidden', False) + self.website = getattr(pluginInfo, 'website', '') + self.authors = getattr(pluginInfo, 'authors', []) + self.iconName = getattr(pluginInfo, 'icon_name', '') + self.copyright = getattr(pluginInfo, 'copyright', '') + self.dependencies = getattr(pluginInfo, 'dependencies', []) + self.helpUri = getattr(pluginInfo, 'help_uri', '') + self.dataDir = getattr(pluginInfo, 'data_dir', '') + self.updateTranslationContext() - - def updateTranslationContext(self, domain = None, localeDir = None, language = None, fallbackToCthulhuTranslation = True): + + def updateTranslationContext(self, domain=None, localeDir=None, language=None, fallbackToCthulhuTranslation=True): + """Update the translation context. + + Args: + domain: The translation domain. + localeDir: The locale directory. + language: The language to use. + fallbackToCthulhuTranslation: Whether to fall back to Cthulhu translations. + """ self.translationContext = None - useLocaleDir = '{}/locale/'.format(self.getModuleDir()) + useLocaleDir = f'{self.getModuleDir()}/locale/' + if localeDir: if os.path.isdir(localeDir): useLocaleDir = localeDir + useName = self.getModuleName() useDomain = useName + if domain: useDomain = domain + useLanguage = None if language: useLanguage = language - self.translationContext = self.getApp().getTranslationManager().initTranslation(useName, domain=useDomain, localeDir=useLocaleDir, language=useLanguage, fallbackToCthulhuTranslation=fallbackToCthulhuTranslation) + + translation_manager = self.getApp().getTranslationManager() + self.translationContext = translation_manager.initTranslation( + useName, + domain=useDomain, + localeDir=useLocaleDir, + language=useLanguage, + fallbackToCthulhuTranslation=fallbackToCthulhuTranslation + ) + # Point _ to the translation object in the globals namespace of the caller frame try: callerFrame = inspect.currentframe().f_back @@ -120,66 +202,306 @@ class Plugin(): callerFrame.f_globals['_'] = self.translationContext.gettext callerFrame.f_globals['ngettext'] = self.translationContext.ngettext finally: - del callerFrame # Avoid reference problems with frames (per python docs) + del callerFrame # Avoid reference problems with frames (per python docs) + def getTranslationContext(self): + """Get the translation context. + + Returns: + The translation context. + """ return self.translationContext + def isPluginBuildIn(self): + """Check if the plugin is built-in. + + Returns: + True if the plugin is built-in, False otherwise. + """ return self.buildIn + def isPluginHidden(self): + """Check if the plugin is hidden. + + Returns: + True if the plugin is hidden, False otherwise. + """ return self.hidden + def getAuthors(self): + """Get the plugin authors. + + Returns: + A list of plugin authors. + """ return self.authors + def getCopyright(self): + """Get the plugin copyright. + + Returns: + The plugin copyright. + """ return self.copyright + def getDataDir(self): + """Get the plugin data directory. + + Returns: + The plugin data directory. + """ return self.dataDir + def getDependencies(self): + """Get the plugin dependencies. + + Returns: + The plugin dependencies. + """ return self.dependencies + def getDescription(self): + """Get the plugin description. + + Returns: + The plugin description. + """ return self.description + def getgetHelpUri(self): + """Get the plugin help URI. + + Returns: + The plugin help URI. + """ return self.helpUri + def getIconName(self): + """Get the plugin icon name. + + Returns: + The plugin icon name. + """ return self.iconName + def getModuleDir(self): + """Get the plugin module directory. + + Returns: + The plugin module directory. + """ return self.moduleDir + def getModuleName(self): + """Get the plugin module name. + + Returns: + The plugin module name. + """ return self.moduleName + def getName(self): + """Get the plugin name. + + Returns: + The plugin name. + """ return self.name + def getVersion(self): + """Get the plugin version. + + Returns: + The plugin version. + """ return self.version + def getWebsite(self): + """Get the plugin website. + + Returns: + The plugin website. + """ return self.website - def getSetting(key): - #self.getModuleName()) + + def getSetting(self, key): + """Get a plugin setting. + + Args: + key: The setting key. + + Returns: + The setting value, or None if not found. + """ + # To be implemented return None - def setSetting(key, value): - #self.getModuleName()) + + def setSetting(self, key, value): + """Set a plugin setting. + + Args: + key: The setting key. + value: The setting value. + """ + # To be implemented pass - def registerGestureByString(self, function, name, gestureString, learnModeEnabled = True): - keybinding = self.getApp().getAPIHelper().registerGestureByString(function, name, gestureString, 'default', 'cthulhu', learnModeEnabled, contextName = self.getModuleName()) + + def registerGestureByString(self, function, name, gestureString, learnModeEnabled=True): + """Register a gesture by string. + + Args: + function: The function to call when the gesture is triggered. + name: The name of the gesture. + gestureString: The gesture string. + learnModeEnabled: Whether the gesture is enabled in learn mode. + + Returns: + The registered keybinding. + """ + keybinding = self.getApp().getAPIHelper().registerGestureByString( + function, + name, + gestureString, + 'default', + 'cthulhu', + learnModeEnabled, + contextName=self.getModuleName() + ) return keybinding - def unregisterShortcut(self, function, name, gestureString, learnModeEnabled = True): - ok = self.getApp().getAPIHelper().unregisterShortcut(keybinding, contextName = self.getModuleName()) + + def unregisterShortcut(self, keybinding, learnModeEnabled=True): + """Unregister a shortcut. + + Args: + keybinding: The keybinding to unregister. + learnModeEnabled: Whether the shortcut is enabled in learn mode. + + Returns: + True if the shortcut was unregistered, False otherwise. + """ + ok = self.getApp().getAPIHelper().unregisterShortcut( + keybinding, + contextName=self.getModuleName() + ) return ok - def registerSignal(self, signalName, signalFlag = GObject.SignalFlags.RUN_LAST, closure = GObject.TYPE_NONE, accumulator=()): - ok = self.signalManager.registerSignal(signalName, signalFlag, closure, accumulator, contextName = self.getModuleName()) + + def registerSignal(self, signalName, signalFlag=None, closure=None, accumulator=()): + """Register a signal. + + Args: + signalName: The signal name. + signalFlag: The signal flags. + closure: The closure. + accumulator: The accumulator. + + Returns: + True if the signal was registered, False otherwise. + """ + if signalFlag is None: + # Import GObject if available, otherwise use a dummy value + try: + from gi.repository import GObject + signalFlag = GObject.SignalFlags.RUN_LAST + except ImportError: + signalFlag = 1 # Default value + + if closure is None: + try: + from gi.repository import GObject + closure = GObject.TYPE_NONE + except ImportError: + closure = None # Default value + + ok = self.signalManager.registerSignal( + signalName, + signalFlag, + closure, + accumulator, + contextName=self.getModuleName() + ) return ok + def unregisterSignal(self, signalName): - # how to unregister? + """Unregister a signal. + + Args: + signalName: The signal name. + """ + # To be implemented pass - - def connectSignal(self, signalName, function, param = None): - signalID = self.signalManager.connectSignal(signalName, function, param, contextName = self.getModuleName()) + + def connectSignal(self, signalName, function, param=None): + """Connect to a signal. + + Args: + signalName: The signal name. + function: The function to call when the signal is triggered. + param: The parameter to pass to the function. + + Returns: + The signal ID. + """ + signalID = self.signalManager.connectSignal( + signalName, + function, + param, + contextName=self.getModuleName() + ) return signalID + def disconnectSignalByFunction(self, function): - # need get mapped function + """Disconnect a signal by function. + + Args: + function: The function to disconnect. + """ + # Need to get mapped function mappedFunction = function - self.signalManager.disconnectSignalByFunction(mappedFunction, contextName = self.getModuleName()) - - def registerAPI(self, key, value, application = ''): - ok = self.dynamicApiManager.registerAPI(key, value, application = application, contextName = self.getModuleName()) + self.signalManager.disconnectSignalByFunction( + mappedFunction, + contextName=self.getModuleName() + ) + + def registerAPI(self, key, value, application=''): + """Register an API. + + Args: + key: The API key. + value: The API value. + application: The application. + + Returns: + True if the API was registered, False otherwise. + """ + ok = self.dynamicApiManager.registerAPI( + key, + value, + application=application, + contextName=self.getModuleName() + ) return ok - def unregisterAPI(self, key, application = ''): - self.dynamicApiManager.unregisterAPI(key, application = application, contextName = self.getModuleName()) + + def unregisterAPI(self, key, application=''): + """Unregister an API. + + Args: + key: The API key. + application: The application. + """ + self.dynamicApiManager.unregisterAPI( + key, + application=application, + contextName=self.getModuleName() + ) + + # Pluggy-specific hook implementations + + @cthulhu_hookimpl + def activate(self): + """Activate the plugin. This method should be overridden by subclasses.""" + logger.info(f"Activating plugin: {self.getName()}") + + @cthulhu_hookimpl + def deactivate(self): + """Deactivate the plugin. This method should be overridden by subclasses.""" + logger.info(f"Deactivating plugin: {self.getName()}") diff --git a/src/cthulhu/plugin_system_manager.py b/src/cthulhu/plugin_system_manager.py index 48af6da..0516e3b 100644 --- a/src/cthulhu/plugin_system_manager.py +++ b/src/cthulhu/plugin_system_manager.py @@ -23,36 +23,40 @@ # Fork of Orca Screen Reader (GNOME) # Original source: https://gitlab.gnome.org/GNOME/orca -"""PluginManager for loading cthulhu plugins.""" -import os, inspect, sys, tarfile, shutil +"""Plugin System Manager for Cthulhu using pluggy.""" +import os +import inspect +import sys +import tarfile +import shutil +import importlib.util +import importlib.machinery +import logging from enum import IntEnum -version = sys.version[:3] # we only need major.minor version. -if version in ["3.3","3.4"]: - from importlib.machinery import SourceFileLoader -else: # Python 3.5+, no support for python < 3.3. - import importlib.util +# Import pluggy if available +try: + import pluggy + PLUGGY_AVAILABLE = True +except ImportError: + PLUGGY_AVAILABLE = False + print("Warning: pluggy not available. Using fallback plugin system.") -import gi -gi.require_version('Peas', '1.0') -from gi.repository import GObject -from gi.repository import Peas - -gi.require_version('Atspi', '2.0') -from gi.repository import Atspi +try: + from gi.repository import GObject + from gi.repository import Atspi + GOBJECT_AVAILABLE = True +except ImportError: + GOBJECT_AVAILABLE = False + print("Warning: GObject not available. Some functionality may be limited.") from cthulhu import resource_manager -class API(GObject.GObject): - """Interface that gives access to all the objects of Cthulhu.""" - def __init__(self, app): - GObject.GObject.__init__(self) - self.app = app +logger = logging.getLogger(__name__) class PluginType(IntEnum): """Types of plugins we support, depending on their directory location.""" - # pylint: disable=comparison-with-callable,inconsistent-return-statements,no-else-return # SYSTEM: provides system wide plugins SYSTEM = 1 # USER: provides per user plugin @@ -60,237 +64,884 @@ class PluginType(IntEnum): def __str__(self): if self.value == PluginType.SYSTEM: - return _("System plugin") + return "System plugin" elif self.value == PluginType.USER: - return _("User plugin") + return "User plugin" def get_root_dir(self): """Returns the directory where this type of plugins can be found.""" if self.value == PluginType.SYSTEM: - return os.path.dirname(os.path.realpath(os.path.abspath(inspect.getfile(inspect.currentframe())))) + '/plugins' + current_file = inspect.getfile(inspect.currentframe()) + current_dir = os.path.dirname(os.path.realpath(os.path.abspath(current_file))) + return os.path.join(current_dir, 'plugins') elif self.value == PluginType.USER: - return os.path.expanduser('~') + '/.local/share/cthulhu/plugins' + return os.path.expanduser('~/.local/share/cthulhu/plugins') -class PluginSystemManager(): - """Cthulhu Plugin Manager to handle a set of plugins. - Attributes: - DEFAULT_LOADERS (tuple): Default loaders used by the plugin manager. For - possible values see - https://developer.gnome.org/libpeas/stable/PeasEngine.html#peas-engine-enable-loader - """ - DEFAULT_LOADERS = ("python3", ) +class PluginInfo: + """Information about a plugin.""" + + def __init__(self, name, module_name, module_dir, metadata=None): + """Initialize a PluginInfo object. + + Args: + name: The plugin name. + module_name: The module name. + module_dir: The module directory. + metadata: The plugin metadata. + """ + self.name = name + self.module_name = module_name + self.module_dir = module_dir + self.metadata = metadata or {} + self.builtin = False + self.hidden = False + self.module = None + self.instance = None + self.loaded = False + + def get_module_name(self): + """Get the module name. + + Returns: + The module name. + """ + return self.module_name + + def get_name(self): + """Get the plugin name. + + Returns: + The plugin name. + """ + return self.metadata.get('name', self.name) + + def get_version(self): + """Get the plugin version. + + Returns: + The plugin version. + """ + return self.metadata.get('version', '0.0.0') + + def get_description(self): + """Get the plugin description. + + Returns: + The plugin description. + """ + return self.metadata.get('description', '') + + def get_authors(self): + """Get the plugin authors. + + Returns: + A list of plugin authors. + """ + authors = self.metadata.get('authors', []) + if isinstance(authors, str): + authors = [authors] + return authors + + def get_website(self): + """Get the plugin website. + + Returns: + The plugin website. + """ + return self.metadata.get('website', '') + + def get_copyright(self): + """Get the plugin copyright. + + Returns: + The plugin copyright. + """ + return self.metadata.get('copyright', '') + + def get_module_dir(self): + """Get the plugin module directory. + + Returns: + The plugin module directory. + """ + return self.module_dir + + def get_data_dir(self): + """Get the plugin data directory. + + Returns: + The plugin data directory. + """ + return self.module_dir + + def get_dependencies(self): + """Get the plugin dependencies. + + Returns: + A list of plugin dependencies. + """ + return self.metadata.get('dependencies', []) + + def get_help_uri(self): + """Get the plugin help URI. + + Returns: + The plugin help URI. + """ + return self.metadata.get('help_uri', '') + + def get_icon_name(self): + """Get the plugin icon name. + + Returns: + The plugin icon name. + """ + return self.metadata.get('icon_name', '') + + def get_settings(self): + """Get the plugin settings. + + Returns: + The plugin settings. + """ + return None # To be implemented + + def is_builtin(self): + """Check if the plugin is built-in. + + Returns: + True if the plugin is built-in, False otherwise. + """ + return self.builtin + + def is_hidden(self): + """Check if the plugin is hidden. + + Returns: + True if the plugin is hidden, False otherwise. + """ + return self.hidden + + def is_available(self): + """Check if the plugin is available. + + Returns: + True if the plugin is available, False otherwise. + """ + return True + + def is_loaded(self): + """Check if the plugin is loaded. + + Returns: + True if the plugin is loaded, False otherwise. + """ + return self.loaded + +class CthulhuHookSpecs: + """Hook specifications for Cthulhu plugins.""" + + def activate(self): + """Called when the plugin is activated.""" + pass + + def deactivate(self): + """Called when the plugin is deactivated.""" + pass + + +if GOBJECT_AVAILABLE: + class API(GObject.GObject): + """Interface that gives access to all the objects of Cthulhu.""" + def __init__(self, app): + GObject.GObject.__init__(self) + self.app = app +else: + class API: + """Interface that gives access to all the objects of Cthulhu.""" + def __init__(self, app): + self.app = app + + +class PluginSystemManager: + """Cthulhu Plugin Manager using pluggy.""" + def __init__(self, app): + """Initialize the plugin system manager. + + Args: + app: The Cthulhu application instance. + """ self.app = app - self.engine = Peas.Engine.get_default() - - for loader in self.DEFAULT_LOADERS: - self.engine.enable_loader(loader) - - self._setupPluginsDir() - self._setupExtensionSet() + + # Initialize plugin manager + if PLUGGY_AVAILABLE: + self.plugin_manager = pluggy.PluginManager("cthulhu") + hook_spec = pluggy.HookspecMarker("cthulhu") + for name, method in inspect.getmembers(CthulhuHookSpecs, inspect.isfunction): + setattr(CthulhuHookSpecs, name, hook_spec(method)) + self.plugin_manager.add_hookspecs(CthulhuHookSpecs) + else: + self.plugin_manager = None + + # List of plugin infos + self._plugins = {} # module_name -> PluginInfo + self._active_plugins = [] + self._ignore_plugin_module_path = [] + + self._setup_plugins_dir() if self.app: self.gsettingsManager = self.app.getSettingsManager() - # settings else: - # settings self.gsettingsManager = gsettings_manager.getSettingsManager(self.app) - self._activePlugins = [] - self._ignorePluginModulePath = [] + @property def plugins(self): - """Gets the engine's plugin list.""" - return self.engine.get_plugin_list() + """Get the list of all plugins. + + Returns: + A list of PluginInfo objects. + """ + return list(self._plugins.values()) + + def _setup_plugins_dir(self): + """Set up plugin directories.""" + # Ensure plugin directories exist + system_plugins_dir = PluginType.SYSTEM.get_root_dir() + user_plugins_dir = PluginType.USER.get_root_dir() + + os.makedirs(system_plugins_dir, exist_ok=True) + os.makedirs(user_plugins_dir, exist_ok=True) + + def getApp(self): + """Get the application instance. + + Returns: + The Cthulhu application instance. + """ + return self.app + @classmethod def getPluginType(cls, pluginInfo): - """Gets the PluginType for the specified Peas.PluginInfo.""" - paths = [pluginInfo.get_data_dir(), PluginType.SYSTEM.get_root_dir()] - if os.path.commonprefix(paths) == PluginType.SYSTEM.get_root_dir(): + """Get the plugin type for a plugin. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin type (PluginType.SYSTEM or PluginType.USER). + """ + plugin_dir = pluginInfo.get_module_dir() + if plugin_dir.startswith(PluginType.SYSTEM.get_root_dir()): return PluginType.SYSTEM return PluginType.USER - + def getExtension(self, pluginInfo): + """Get the plugin extension. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin instance or None if not found. + """ if not pluginInfo: return None - - return self.extension_set.get_extension(pluginInfo) + + return pluginInfo.instance + def rescanPlugins(self): - self.engine.garbage_collect() - self.engine.rescan_plugins() - def getApp(self): - return self.app - def getPluginInfoByName(self, pluginName, pluginType=PluginType.USER): - """Gets the plugin info for the specified plugin name. + """Rescan for plugins in the plugin directories.""" + old_plugins = self._plugins.copy() + self._plugins = {} + + # Scan system plugins + self._scan_plugins_in_directory(PluginType.SYSTEM.get_root_dir(), PluginType.SYSTEM) + + # Scan user plugins + self._scan_plugins_in_directory(PluginType.USER.get_root_dir(), PluginType.USER) + + # Preserve loaded state from old plugins + for name, old_info in old_plugins.items(): + if name in self._plugins and old_info.loaded: + self._plugins[name].loaded = True + self._plugins[name].instance = old_info.instance + self._plugins[name].module = old_info.module + + # Garbage collect + if PLUGGY_AVAILABLE: + plugins_to_remove = set(old_plugins.keys()) - set(self._plugins.keys()) + for name in plugins_to_remove: + if old_plugins[name].instance: + try: + self.plugin_manager.unregister(old_plugins[name].instance) + except Exception as e: + logger.error(f"Error unregistering plugin {name}: {e}") + + def _scan_plugins_in_directory(self, directory, plugin_type): + """Scan for plugins in a directory. + Args: - pluginName (str): The name from the .plugin file of the module. - Returns: - Peas.PluginInfo: The plugin info if it exists. Otherwise, `None`. + directory: The directory to scan. + plugin_type: The plugin type. """ - for pluginInfo in self.plugins: - if pluginInfo.get_module_name() == pluginName and PluginSystemManager.getPluginType(pluginInfo) == pluginType: - return pluginInfo + if not os.path.exists(directory) or not os.path.isdir(directory): + return + + for item in os.listdir(directory): + plugin_dir = os.path.join(directory, item) + if not os.path.isdir(plugin_dir): + continue + + if plugin_dir in self._ignore_plugin_module_path: + continue + + # Look for plugin files + plugin_file = os.path.join(plugin_dir, "plugin.py") + metadata_file = os.path.join(plugin_dir, "plugin.info") + + if os.path.isfile(plugin_file): + # Extract plugin info + module_name = os.path.basename(plugin_dir) + metadata = self._load_plugin_metadata(metadata_file) + + plugin_info = PluginInfo( + metadata.get('name', module_name), + module_name, + plugin_dir, + metadata + ) + + plugin_info.builtin = (plugin_type == PluginType.SYSTEM and + metadata.get('builtin', 'false') == 'true') + plugin_info.hidden = metadata.get('hidden', 'false') == 'true' + + self._plugins[module_name] = plugin_info + + def _load_plugin_metadata(self, metadata_file): + """Load plugin metadata from a file. + + Args: + metadata_file: The metadata file path. + + Returns: + A dictionary of metadata. + """ + metadata = {} + + if os.path.isfile(metadata_file): + try: + with open(metadata_file, 'r') as f: + for line in f: + line = line.strip() + if not line or line.startswith('#'): + continue + + if '=' in line: + key, value = line.split('=', 1) + metadata[key.strip()] = value.strip() + except Exception as e: + logger.error(f"Error loading plugin metadata: {e}") + + return metadata + + def getPluginInfoByName(self, pluginName, pluginType=PluginType.USER): + """Get plugin information by name. + + Args: + pluginName: The plugin name. + pluginType: The plugin type. + + Returns: + The plugin information or None if not found. + """ + plugin_info = self._plugins.get(pluginName) + if plugin_info and self.getPluginType(plugin_info) == pluginType: + return plugin_info return None + def getActivePlugins(self): - return self._activePlugins + """Get the list of active plugin names. + + Returns: + A list of active plugin names. + """ + return self._active_plugins + def setActivePlugins(self, activePlugins): - self._activePlugins = activePlugins + """Set the list of active plugins and sync their state. + + Args: + activePlugins: A list of plugin names to activate. + """ + self._active_plugins = activePlugins self.syncAllPluginsActive() + def isPluginBuildIn(self, pluginInfo): + """Check if a plugin is built-in. + + Args: + pluginInfo: The plugin information. + + Returns: + True if the plugin is built-in, False otherwise. + """ return pluginInfo.is_builtin() - def isPluginHidden(self, pluginInfo): + +def isPluginHidden(self, pluginInfo): + """Check if a plugin is hidden. + + Args: + pluginInfo: The plugin information. + + Returns: + True if the plugin is hidden, False otherwise. + """ return pluginInfo.is_hidden() + def getPluginAuthors(self, pluginInfo): + """Get the plugin authors. + + Args: + pluginInfo: The plugin information. + + Returns: + A list of plugin authors. + """ return pluginInfo.get_authors() + def getPluginCopyright(self, pluginInfo): + """Get the plugin copyright. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin copyright. + """ return pluginInfo.get_copyright() + def getPluginDataDir(self, pluginInfo): + """Get the plugin data directory. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin data directory. + """ return pluginInfo.get_data_dir() + def getPluginDependencies(self, pluginInfo): + """Get the plugin dependencies. + + Args: + pluginInfo: The plugin information. + + Returns: + A list of plugin dependencies. + """ return pluginInfo.get_dependencies() + def getPluginDescription(self, pluginInfo): + """Get the plugin description. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin description. + """ return pluginInfo.get_description() + def getPlugingetHelpUri(self, pluginInfo): + """Get the plugin help URI. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin help URI. + """ return pluginInfo.get_help_uri() + def getPluginIconName(self, pluginInfo): + """Get the plugin icon name. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin icon name. + """ return pluginInfo.get_icon_name() + def getPluginModuleDir(self, pluginInfo): + """Get the plugin module directory. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin module directory. + """ return pluginInfo.get_module_dir() + def getPluginModuleName(self, pluginInfo): + """Get the plugin module name. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin module name. + """ return pluginInfo.get_module_name() + def getPluginName(self, pluginInfo): + """Get the plugin name. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin name. + """ return pluginInfo.get_name() + def getPluginSettings(self, pluginInfo): + """Get the plugin settings. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin settings. + """ return pluginInfo.get_settings() + def getPluginVersion(self, pluginInfo): + """Get the plugin version. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin version. + """ return pluginInfo.get_version() + def getPluginWebsite(self, pluginInfo): + """Get the plugin website. + + Args: + pluginInfo: The plugin information. + + Returns: + The plugin website. + """ return pluginInfo.get_website() - # has_dependency and get_external_data seems broken-> takes exactly 2 arguments (1 given) but documentation doesnt say any parameter - #def hasPluginDependency(self, pluginInfo): - # return pluginInfo.has_dependency() - #def getPluginExternalData(self, pluginInfo): - # return pluginInfo.get_external_data() + def isPluginAvailable(self, pluginInfo): + """Check if a plugin is available. + + Args: + pluginInfo: The plugin information. + + Returns: + True if the plugin is available, False otherwise. + """ try: return pluginInfo.is_available() except: return False + def isPluginLoaded(self, pluginInfo): + """Check if a plugin is loaded. + + Args: + pluginInfo: The plugin information. + + Returns: + True if the plugin is loaded, False otherwise. + """ try: return pluginInfo.is_loaded() except: return False def getIgnoredPlugins(self): - return self._ignorePluginModulePath + """Get the list of ignored plugin paths. + + Returns: + A list of ignored plugin paths. + """ + return self._ignore_plugin_module_path + def setIgnoredPlugins(self, pluginModulePath, ignored): + """Set a plugin path as ignored or not. + + Args: + pluginModulePath: The plugin module path. + ignored: Whether to ignore the plugin. + """ if pluginModulePath.endswith('/'): pluginModulePath = pluginModulePath[:-1] + if ignored: - if not pluginModulePath in self.getIgnoredPlugins(): - self._ignorePluginModulePath.append(pluginModulePath) + if pluginModulePath not in self.getIgnoredPlugins(): + self._ignore_plugin_module_path.append(pluginModulePath) else: - if pluginModulePath in self.getIgnoredPlugins(): - self._ignorePluginModulePath.remove(pluginModulePath) - + if pluginModulePath in self.getIgnoredPlugins(): + self._ignore_plugin_module_path.remove(pluginModulePath) + def setPluginActive(self, pluginInfo, active): + """Set the active state of a plugin. + + Args: + pluginInfo: The plugin information. + active: Whether to activate the plugin. + """ if self.isPluginBuildIn(pluginInfo): active = True + pluginName = self.getPluginModuleName(pluginInfo) + if active: - if not pluginName in self.getActivePlugins(): + if pluginName not in self.getActivePlugins(): if self.loadPlugin(pluginInfo): - self._activePlugins.append(pluginName ) + self._active_plugins.append(pluginName) else: - if pluginName in self.getActivePlugins(): + if pluginName in self.getActivePlugins(): if self.unloadPlugin(pluginInfo): - self._activePlugins.remove(pluginName ) + self._active_plugins.remove(pluginName) + def isPluginActive(self, pluginInfo): + """Check if a plugin is active. + + Args: + pluginInfo: The plugin information. + + Returns: + True if the plugin is active, False otherwise. + """ if self.isPluginBuildIn(pluginInfo): return True + if self.isPluginLoaded(pluginInfo): return True + active_plugin_names = self.getActivePlugins() return self.getPluginModuleName(pluginInfo) in active_plugin_names + def syncAllPluginsActive(self, ForceAllPlugins=False): + """Sync the active state of all plugins. + + Args: + ForceAllPlugins: Whether to force all plugins to sync. + """ self.unloadAllPlugins(ForceAllPlugins) self.loadAllPlugins(ForceAllPlugins) - + def loadAllPlugins(self, ForceAllPlugins=False): - """Loads plugins from settings.""" + """Load all active plugins. + + Args: + ForceAllPlugins: Whether to force all plugins to load. + """ for pluginInfo in self.plugins: if self.isPluginActive(pluginInfo) or ForceAllPlugins: self.loadPlugin(pluginInfo) - + def loadPlugin(self, pluginInfo): + """Load a plugin. + + Args: + pluginInfo: The plugin information. + + Returns: + True if the plugin was loaded successfully, False otherwise. + """ resourceManager = self.getApp().getResourceManager() moduleName = pluginInfo.get_module_name() + try: if pluginInfo not in self.plugins: - print("Plugin missing: {}".format(moduleName)) + print(f"Plugin missing: {moduleName}") return False + + # Add resource context resourceManager.addResourceContext(moduleName) - self.engine.load_plugin(pluginInfo) + + # If already loaded, nothing to do + if pluginInfo.loaded: + return True + + # Import plugin module + plugin_file = os.path.join(pluginInfo.get_module_dir(), "plugin.py") + spec = importlib.util.spec_from_file_location(moduleName, plugin_file) + module = importlib.util.module_from_spec(spec) + spec.loader.exec_module(module) + pluginInfo.module = module + + # Find Plugin class + plugin_class = None + for attr_name in dir(module): + attr = getattr(module, attr_name) + if (inspect.isclass(attr) and + attr.__module__ == module.__name__ and + hasattr(attr, 'activate')): + plugin_class = attr + break + + if not plugin_class: + print(f"No plugin class found in {moduleName}") + return False + + # Create plugin instance + plugin_instance = plugin_class() + pluginInfo.instance = plugin_instance + + # Initialize plugin + if hasattr(plugin_instance, 'setApp'): + plugin_instance.setApp(self.getApp()) + + if hasattr(plugin_instance, 'setPluginInfo'): + plugin_instance.setPluginInfo(pluginInfo) + + # Register with pluggy + if PLUGGY_AVAILABLE and self.plugin_manager: + self.plugin_manager.register(plugin_instance) + + # Activate plugin + if hasattr(plugin_instance, 'activate'): + plugin_instance.activate() + + pluginInfo.loaded = True + return True + except Exception as e: - print('loadPlugin:',e) + print(f'loadPlugin: {e}') + import traceback + traceback.print_exc() return False - return True - + def unloadAllPlugins(self, ForceAllPlugins=False): - """Loads plugins from settings.""" + """Unload all inactive plugins. + + Args: + ForceAllPlugins: Whether to force all plugins to unload. + """ for pluginInfo in self.plugins: if not self.isPluginActive(pluginInfo) or ForceAllPlugins: self.unloadPlugin(pluginInfo) - + def unloadPlugin(self, pluginInfo): + """Unload a plugin. + + Args: + pluginInfo: The plugin information. + + Returns: + True if the plugin was unloaded successfully, False otherwise. + """ resourceManager = self.getApp().getResourceManager() moduleName = pluginInfo.get_module_name() + try: if pluginInfo not in self.plugins: - print("Plugin missing: {}".format(moduleName)) + print(f"Plugin missing: {moduleName}") return False + if self.isPluginBuildIn(pluginInfo): return False - self.engine.unload_plugin(pluginInfo) - self.getApp().getResourceManager().removeResourceContext(moduleName) - self.engine.garbage_collect() + + # If not loaded, nothing to do + if not pluginInfo.loaded: + return True + + # Deactivate plugin + plugin_instance = pluginInfo.instance + if plugin_instance: + if hasattr(plugin_instance, 'deactivate'): + plugin_instance.deactivate() + + # Unregister from pluggy + if PLUGGY_AVAILABLE and self.plugin_manager: + self.plugin_manager.unregister(plugin_instance) + + # Clean up + pluginInfo.instance = None + pluginInfo.loaded = False + + # Remove resource context + resourceManager.removeResourceContext(moduleName) + + return True + except Exception as e: - print('unloadPlugin:',e) + print(f'unloadPlugin: {e}') return False - return True + def installPlugin(self, pluginFilePath, pluginType=PluginType.USER): + """Install a plugin from a tarball. + + Args: + pluginFilePath: The plugin file path. + pluginType: The plugin type. + + Returns: + True if the plugin was installed successfully, False otherwise. + """ if not self.isValidPluginFile(pluginFilePath): return False + pluginFolder = pluginType.get_root_dir() if not pluginFolder.endswith('/'): pluginFolder += '/' + if not os.path.exists(pluginFolder): - os.mkdir(pluginFolder) + os.makedirs(pluginFolder) else: if not os.path.isdir(pluginFolder): return False + try: with tarfile.open(pluginFilePath) as tar: tar.extractall(path=pluginFolder) except Exception as e: print(e) + return False pluginModulePath = self.getModuleDirByPluginFile(pluginFilePath) - if pluginModulePath != '': + if pluginModulePath: pluginModulePath = pluginFolder + pluginModulePath - self.setIgnoredPlugins(pluginModulePath[:-1], False) # without ending / - print('install', pluginFilePath) + self.setIgnoredPlugins(pluginModulePath[:-1], False) # without ending / + print(f'install: {pluginFilePath}') self.callPackageTriggers(pluginModulePath, 'onPostInstall') + self.rescanPlugins() - return True + def getModuleDirByPluginFile(self, pluginFilePath): + """Get the module directory from a plugin file. + + Args: + pluginFilePath: The plugin file path. + + Returns: + The module directory, or an empty string if not found. + """ if not isinstance(pluginFilePath, str): return '' - if pluginFilePath == '': + + if not pluginFilePath: return '' + if not os.path.exists(pluginFilePath): return '' + try: with tarfile.open(pluginFilePath) as tar: tarMembers = tar.getmembers() @@ -299,50 +950,80 @@ class PluginSystemManager(): return tarMember.name except Exception as e: print(e) + return '' + def isValidPluginFile(self, pluginFilePath): + """Check if a file is a valid plugin package. + + Args: + pluginFilePath: The plugin file path. + + Returns: + True if the file is a valid plugin package, False otherwise. + """ if not isinstance(pluginFilePath, str): return False - if pluginFilePath == '': + + if not pluginFilePath: return False + if not os.path.exists(pluginFilePath): return False + pluginFolder = '' pluginFileExists = False packageFileExists = False + try: with tarfile.open(pluginFilePath) as tar: tarMembers = tar.getmembers() for tarMember in tarMembers: if tarMember.isdir(): - if pluginFolder == '': + if not pluginFolder: pluginFolder = tarMember.name + if tarMember.isfile(): if tarMember.name.endswith('.plugin'): pluginFileExists = True if tarMember.name.endswith('package.py'): - pluginFileExists = True + packageFileExists = True + if not tarMember.name.startswith(pluginFolder): return False except Exception as e: print(e) return False + return pluginFileExists + def uninstallPlugin(self, pluginInfo): + """Uninstall a plugin. + + Args: + pluginInfo: The plugin information. + + Returns: + True if the plugin was uninstalled successfully, False otherwise. + """ if self.isPluginBuildIn(pluginInfo): return False - # do we want to allow removing system plugins? - if PluginSystemManager.getPluginType(pluginInfo) == PluginType.SYSTEM: + + # Do we want to allow removing system plugins? + if self.getPluginType(pluginInfo) == PluginType.SYSTEM: return False + pluginFolder = pluginInfo.get_data_dir() if not pluginFolder.endswith('/'): pluginFolder += '/' + if not os.path.isdir(pluginFolder): return False + if self.isPluginActive(pluginInfo): self.setPluginActive(pluginInfo, False) - SettingsManager = self.app.getSettingsManager() - # TODO SettingsManager.set_settings_value_list('active-plugins', self.getActivePlugins()) + # TODO: Update active plugins in settings + self.callPackageTriggers(pluginFolder, 'onPreUninstall') try: @@ -350,22 +1031,34 @@ class PluginSystemManager(): except Exception as e: print(e) return False + self.setIgnoredPlugins(pluginFolder, True) self.rescanPlugins() - + return True + def callPackageTriggers(self, pluginPath, trigger): + """Call package trigger functions. + + Args: + pluginPath: The plugin path. + trigger: The trigger name. + """ if not os.path.exists(pluginPath): return + if not pluginPath.endswith('/'): pluginPath += '/' + packageModulePath = pluginPath + 'package.py' if not os.path.isfile(packageModulePath): return + if not os.access(packageModulePath, os.R_OK): return + package = self.getApp().getAPIHelper().importModule('package', packageModulePath) - + if trigger == 'onPostInstall': try: package.onPostInstall(pluginPath, self.getApp()) @@ -377,96 +1070,111 @@ class PluginSystemManager(): except Exception as e: print(e) - def _setupExtensionSet(self): - plugin_iface = API(self.getApp()) - self.extension_set = Peas.ExtensionSet.new(self.engine, - Peas.Activatable, - ["object"], - [plugin_iface]) - self.extension_set.connect("extension-removed", - self.__extensionRemoved) - self.extension_set.connect("extension-added", - self.__extensionAdded) - def _setupPluginsDir(self): - system_plugins_dir = PluginType.SYSTEM.get_root_dir() - user_plugins_dir = PluginType.USER.get_root_dir() - if os.path.exists(user_plugins_dir): - self.engine.add_search_path(user_plugins_dir) - if os.path.exists(system_plugins_dir): - self.engine.add_search_path(system_plugins_dir) - - def __extensionRemoved(self, unusedSet, pluginInfo, extension): - extension.deactivate() - - def __extensionAdded(self, unusedSet, pluginInfo, extension): - extension.setApp(self.getApp()) - extension.setPluginInfo(pluginInfo) - extension.activate() - def __loadedPlugins(self, engine, unusedSet): - """Handles the changing of the loaded plugin list.""" - self.getApp().settings.ActivePlugins = engine.get_property("loaded-plugins") - -class APIHelper(): +class APIHelper: + """API Helper for plugin system.""" + def __init__(self, app): + """Initialize the API helper. + + Args: + app: The Cthulhu application instance. + """ self.app = app self.cthulhuKeyBindings = None - - ''' - _pluginAPIManager.seCthulhuAPI('Logger', _logger) - _pluginAPIManager.setCthulhuAPI('SettingsManager', _settingsManager) - _pluginAPIManager.setCthulhuAPI('ScriptManager', _scriptManager) - _pluginAPIManager.setCthulhuAPI('EventManager', _eventManager) - _pluginAPIManager.setCthulhuAPI('Speech', speech) - _pluginAPIManager.setCthulhuAPI('Sound', sound) - _pluginAPIManager.setCthulhuAPI('Braille', braille) - _pluginAPIManager.setCthulhuAPI('Debug', debug) - _pluginAPIManager.setCthulhuAPI('Messages', messages) - _pluginAPIManager.setCthulhuAPI('MouseReview', mouse_review) - _pluginAPIManager.setCthulhuAPI('NotificationMessages', notification_messages) - _pluginAPIManager.setCthulhuAPI('CthulhuState', cthulhu_state) - _pluginAPIManager.setCthulhuAPI('CthulhuPlatform', cthulhu_platform) - _pluginAPIManager.setCthulhuAPI('Settings', settings) - _pluginAPIManager.setCthulhuAPI('Keybindings', keybindings) - ''' + def outputMessage(self, Message, interrupt=False): + """Output a message. + + Args: + Message: The message to output. + interrupt: Whether to interrupt current speech. + """ settings = self.app.getDynamicApiManager().getAPI('Settings') braille = self.app.getDynamicApiManager().getAPI('Braille') speech = self.app.getDynamicApiManager().getAPI('Speech') - if speech != None: - if (settings.enableSpeech): + + if speech is not None: + if settings.enableSpeech: if interrupt: speech.cancel() - if Message != '': + if Message: speech.speak(Message) - if braille != None: - if (settings.enableBraille): + + if braille is not None: + if settings.enableBraille: braille.displayMessage(Message) + def createInputEventHandler(self, function, name, learnModeEnabled=True): + """Create an input event handler. + + Args: + function: The function to call. + name: The handler name. + learnModeEnabled: Whether learn mode is enabled. + + Returns: + The input event handler. + """ EventManager = self.app.getDynamicApiManager().getAPI('EventManager') newInputEventHandler = EventManager.input_event.InputEventHandler(function, name, learnModeEnabled) return newInputEventHandler - def registerGestureByString(self, function, name, gestureString, profile, application, learnModeEnabled = True, contextName = None): + + def registerGestureByString(self, function, name, gestureString, profile, application, learnModeEnabled=True, contextName=None): + """Register a gesture by string. + + Args: + function: The function to call. + name: The gesture name. + gestureString: The gesture string. + profile: The profile. + application: The application. + learnModeEnabled: Whether learn mode is enabled. + contextName: The context name. + + Returns: + The registered gestures. + """ gestureList = gestureString.split(',') registeredGestures = [] + for gesture in gestureList: if gesture.startswith('kb:'): shortcutString = gesture[3:] - registuredGesture = self.registerShortcutByString(function, name, shortcutString, profile, application, learnModeEnabled, contextName=contextName) + registuredGesture = self.registerShortcutByString( + function, name, shortcutString, profile, application, learnModeEnabled, contextName=contextName + ) if registuredGesture: registeredGestures.append(registuredGesture) + return registeredGestures - def registerShortcutByString(self, function, name, shortcutString, profile, application, learnModeEnabled = True, contextName = None): + + def registerShortcutByString(self, function, name, shortcutString, profile, application, learnModeEnabled=True, contextName=None): + """Register a shortcut by string. + + Args: + function: The function to call. + name: The shortcut name. + shortcutString: The shortcut string. + profile: The profile. + application: The application. + learnModeEnabled: Whether learn mode is enabled. + contextName: The context name. + + Returns: + The registered shortcut. + """ keybindings = self.app.getDynamicApiManager().getAPI('Keybindings') settings = self.app.getDynamicApiManager().getAPI('Settings') resourceManager = self.app.getResourceManager() - + clickCount = 0 cthulhuKey = False shiftKey = False ctrlKey = False altKey = False key = '' + shortcutList = shortcutString.split('+') for shortcutElement in shortcutList: shortcutElementLower = shortcutElement.lower() @@ -482,15 +1190,16 @@ class APIHelper(): altKey = True else: key = shortcutElementLower + if clickCount == 0: clickCount = 1 - if self.cthulhuKeyBindings == None: + if self.cthulhuKeyBindings is None: self.cthulhuKeyBindings = keybindings.KeyBindings() - + tryFunction = resource_manager.TryFunction(function) newInputEventHandler = self.createInputEventHandler(tryFunction.runInputEvent, name, learnModeEnabled) - + currModifierMask = keybindings.NO_MODIFIER_MASK if cthulhuKey: currModifierMask = currModifierMask | 1 << keybindings.MODIFIER_CTHULHU @@ -500,45 +1209,67 @@ class APIHelper(): currModifierMask = currModifierMask | 1 << Atspi.ModifierType.ALT if ctrlKey: currModifierMask = currModifierMask | 1 << Atspi.ModifierType.CONTROL - - newKeyBinding = keybindings.KeyBinding(key, keybindings.defaultModifierMask, currModifierMask, newInputEventHandler, clickCount) + + newKeyBinding = keybindings.KeyBinding( + key, keybindings.defaultModifierMask, currModifierMask, newInputEventHandler, clickCount + ) self.cthulhuKeyBindings.add(newKeyBinding) - + settings.keyBindingsMap["default"] = self.cthulhuKeyBindings if contextName: resourceContext = resourceManager.getResourceContext(contextName) if resourceContext: - resourceEntry = resource_manager.ResourceEntry('keyboard', newKeyBinding, function, tryFunction, shortcutString) + resourceEntry = resource_manager.ResourceEntry( + 'keyboard', newKeyBinding, function, tryFunction, shortcutString + ) resourceContext.addGesture(profile, application, newKeyBinding, resourceEntry) - + return newKeyBinding - - def unregisterShortcut(self, KeyBindingToRemove, contextName = None): + + def unregisterShortcut(self, KeyBindingToRemove, contextName=None): + """Unregister a shortcut. + + Args: + KeyBindingToRemove: The key binding to remove. + contextName: The context name. + + Returns: + True if the shortcut was unregistered, False otherwise. + """ ok = False keybindings = self.app.getDynamicApiManager().getAPI('Keybindings') settings = self.app.getDynamicApiManager().getAPI('Settings') resourceManager = self.app.getResourceManager() - - if self.cthulhuKeyBindings == None: + + if self.cthulhuKeyBindings is None: self.cthulhuKeyBindings = keybindings.KeyBindings() + try: self.cthulhuKeyBindings.remove(KeyBindingToRemove) settings.keyBindingsMap["default"] = self.cthulhuKeyBindings ok = True except KeyError: pass + if contextName: resourceContext = resourceManager.getResourceContext(contextName) if resourceContext: resourceContext.removeGesture(KeyBindingToRemove) - + return ok + def importModule(self, moduleName, moduleLocation): - if version in ["3.3","3.4"]: - return SourceFileLoader(moduleName, moduleLocation).load_module() - else: - spec = importlib.util.spec_from_file_location(moduleName, moduleLocation) - driver_mod = importlib.util.module_from_spec(spec) - spec.loader.exec_module(driver_mod) - return driver_mod + """Import a module from a file location. + + Args: + moduleName: The module name. + moduleLocation: The module location. + + Returns: + The imported module. + """ + spec = importlib.util.spec_from_file_location(moduleName, moduleLocation) + module = importlib.util.module_from_spec(spec) + spec.loader.exec_module(module) + return module