PaperWM/app.js
Ole Jørgen Brønner 426d44bf11 app: mkCommandLineSpawner util expanding %d to workspace directory
Use `tilix --working-directory %d` for tilix. Otherwise the daemon will have the
workspace dir as working directory. Subsequent new windows might inherit that
directory. (eg. in spaces that doesn't specify an explicit directory(?))
2019-10-18 19:21:45 +02:00

193 lines
6.3 KiB
JavaScript

/*
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 handler = customHandlers[app.id];
if (handler) {
let space = Tiling.spaces.spaceOfWindow(metaWindow);
return handler(metaWindow, app, space);
}
let workspaceId = metaWindow.get_workspace().workspace_index;
let original = Kludges.getSavedProp(Shell.App.prototype, "open_new_window");
original.call(app, workspaceId);
return true;
}
function trySpawnWindow(app, workspace) {
if (typeof(app) === 'string') {
app = new Shell.App({ app_info: Gio.DesktopAppInfo.new(app) });
}
let handler = customSpawnHandlers[app.id];
if (handler) {
let space = Tiling.spaces.selectedSpace;
return handler(app, space);
} else {
launchFromWorkspaceDir(app, workspace);
}
}
function spawnWindow(app, workspace) {
if (typeof(app) === 'string') {
app = new Shell.App({ app_info: Gio.DesktopAppInfo.new(app) });
}
try {
return trySpawnWindow(app, workspace);
} catch(e) {
// Let the overide take care any fallback
return app.open_new_window(-1);
}
}
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}'`);
}
}
}