/* Application functionality, like global new window actions etc. */ var Extension; if (imports.misc.extensionUtils.extensions) { Extension = imports.misc.extensionUtils.extensions["paperwm@hedning:matrix.org"]; } else { Extension = imports.ui.main.extensionManager.lookup("paperwm@hedning:matrix.org"); } var GLib = imports.gi.GLib var Gio = imports.gi.Gio; var Tiling = Extension.imports.tiling var Kludges = Extension.imports.kludges; var Shell = imports.gi.Shell; var Tracker = Shell.WindowTracker.get_default(); var CouldNotLaunch = Symbol(); // Lookup table for custom handlers, keys being the app id var customHandlers, customSpawnHandlers; function init() { customHandlers = { 'org.gnome.Terminal.desktop': newGnomeTerminal }; customSpawnHandlers = { 'com.gexperts.Tilix.desktop': mkCommandLineSpawner('tilix --working-directory %d') }; function spawnWithFallback(fallback, ...args) { try { return trySpawnWindow(...args); } catch(e) { return fallback(); } } let overrideWithFallback = Kludges.overrideWithFallback; overrideWithFallback( Shell.App, "open_new_window", (fallback, app, workspaceId) => { return spawnWithFallback(fallback, app, global.workspace_manager.get_workspace_by_index(workspaceId)); } ); overrideWithFallback( Shell.App, "launch_action", (fallback, app, name, ...args) => { log(`ShellApp.launch_action ${name}`); if (name === 'new-window') return spawnWithFallback(fallback, app); else { return fallback(); } } ); overrideWithFallback( Gio.DesktopAppInfo, "launch", (fallback, appInfo) => { log(`DesktopAppInfo.launch`); return spawnWithFallback(fallback, appInfo.get_id()); } ); overrideWithFallback( Gio.DesktopAppInfo, "launch_action", (fallback, appInfo, name, ...args) => { log(`DesktopAppInfo.launch_action ${name}`); if (name === 'new-window') return spawnWithFallback(fallback, appInfo.get_id()); else { return fallback(); } } ); } function launchFromWorkspaceDir(app, workspace=null) { if (typeof(app) === 'string') { app = new Shell.App({ app_info: Gio.DesktopAppInfo.new(app) }); } let dir = getWorkspaceDirectory(workspace); let cmd = app.app_info.get_commandline(); if (!cmd || dir == '') { throw CouldNotLaunch; } /* Note: One would think working directory could be specified in the AppLaunchContext The dbus spec https://specifications.freedesktop.org/desktop-entry-spec/1.1/ar01s07.html indicates otherwise (for dbus activated actions). Can affect arbitrary environment variables of exec activated actions, but no environment variable determine working directory of new processes. */ // TODO: substitute correct values according to https://specifications.freedesktop.org/desktop-entry-spec/desktop-entry-spec-latest.html#exec-variables cmd = cmd.replace(/%./g, ""); let [success, cmdArgs] = GLib.shell_parse_argv(cmd); if (!success) { print("launchFromWorkspaceDir:", "Could not parse command line", cmd); throw CouldNotLaunch; } GLib.spawn_async(dir, cmdArgs, GLib.get_environ(), GLib.SpawnFlags.SEARCH_PATH, null); } function newGnomeTerminal(metaWindow, app) { /* Note: this action activation is _not_ bound to the window - instead it relies on the window being active when called. If the new window doesn't start in the same directory it's probably because 'vte.sh' haven't been sourced by the shell in this terminal */ app.action_group.activate_action( "win.new-terminal", new imports.gi.GLib.Variant("(ss)", ["window", "current"])); } function duplicateWindow(metaWindow) { metaWindow = metaWindow || global.display.focus_window; let app = Tracker.get_window_app(metaWindow); let promisedWindow = new Promise( (resolve, reject) => { let id = global.display.connect('window-created', (d, mw) => { global.display.disconnect(id); resolve(mw); }); } ); let handler = customHandlers[app.id]; if (handler) { let space = Tiling.spaces.spaceOfWindow(metaWindow); handler(metaWindow, app, space); return promisedWindow; } let workspaceId = metaWindow.get_workspace().workspace_index; let original = Kludges.getSavedProp(Shell.App.prototype, "open_new_window"); original.call(app, workspaceId); return promisedWindow; } function trySpawnWindow(app, workspace) { if (typeof(app) === 'string') { app = new Shell.App({ app_info: Gio.DesktopAppInfo.new(app) }); } let promisedWindow = new Promise( (resolve, reject) => { let id = global.display.connect('window-created', (d, mw) => { global.display.disconnect(id); resolve(mw); }); } ); let handler = customSpawnHandlers[app.id]; if (handler) { let space = Tiling.spaces.selectedSpace; handler(app, space); } else { launchFromWorkspaceDir(app, workspace); } return promisedWindow; } function spawnWindow(app, workspace) { if (typeof(app) === 'string') { app = new Shell.App({ app_info: Gio.DesktopAppInfo.new(app) }); } log(`foo`); try { return trySpawnWindow(app, workspace); } catch(e) { let promisedWindow = new Promise( (resolve, reject) => { let id = global.display.connect('window-created', (d, mw) => { global.display.disconnect(id); resolve(mw); }); } ); // Let the overide take care any fallback app.open_new_window(-1); return promisedWindow; } } function getWorkspaceDirectory(workspace=null) { let space = workspace ? Tiling.spaces.get(workspace) : Tiling.spaces.selectedSpace; let dir = space.settings.get_string("directory"); if (dir[0] === "~") { dir = GLib.getenv("HOME") + dir.slice(1); } return dir; } function expandCommandline(commandline, workspace) { let dir = getWorkspaceDirectory(workspace) commandline = commandline.replace(/%d/g, () => GLib.shell_quote(dir)); return commandline } function mkCommandLineSpawner(commandlineTemplate, spawnInWorkspaceDir=false) { return (app, space) => { let workspace = space.workspace; let commandline = expandCommandline(commandlineTemplate, workspace); print("Launching", commandline); let workingDir = spawnInWorkspaceDir ? getWorkspaceDirectory(workspace) : null; let [success, cmdArgs] = GLib.shell_parse_argv(commandline); if (success) { success = GLib.spawn_async(workingDir, cmdArgs, GLib.get_environ(), GLib.SpawnFlags.SEARCH_PATH, null); } if (!success) { Extension.imports.extension.notify( `Failed to run custom spawn handler for ${app.id}`, `Attempted to run '${commandline}'`); } } }