7d9a39c72c
This allows e.g. gnome-tweak-tool to install an extension from a zip file and load it into the running shell.
504 lines
16 KiB
JavaScript
504 lines
16 KiB
JavaScript
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
|
|
|
const Gio = imports.gi.Gio;
|
|
const GLib = imports.gi.GLib;
|
|
const Lang = imports.lang;
|
|
const Meta = imports.gi.Meta;
|
|
const Shell = imports.gi.Shell;
|
|
|
|
const Config = imports.misc.config;
|
|
const ExtensionSystem = imports.ui.extensionSystem;
|
|
const ExtensionDownloader = imports.ui.extensionDownloader;
|
|
const ExtensionUtils = imports.misc.extensionUtils;
|
|
const Main = imports.ui.main;
|
|
const Screenshot = imports.ui.screenshot;
|
|
const ViewSelector = imports.ui.viewSelector;
|
|
|
|
const GnomeShellIface = '<node> \
|
|
<interface name="org.gnome.Shell"> \
|
|
<method name="Eval"> \
|
|
<arg type="s" direction="in" name="script" /> \
|
|
<arg type="b" direction="out" name="success" /> \
|
|
<arg type="s" direction="out" name="result" /> \
|
|
</method> \
|
|
<method name="FocusSearch"/> \
|
|
<method name="ShowOSD"> \
|
|
<arg type="a{sv}" direction="in" name="params"/> \
|
|
</method> \
|
|
<method name="ShowMonitorLabels"> \
|
|
<arg type="a{uv}" direction="in" name="params" /> \
|
|
</method> \
|
|
<method name="ShowMonitorLabels2"> \
|
|
<arg type="a{sv}" direction="in" name="params" /> \
|
|
</method> \
|
|
<method name="HideMonitorLabels" /> \
|
|
<method name="FocusApp"> \
|
|
<arg type="s" direction="in" name="id"/> \
|
|
</method> \
|
|
<method name="ShowApplications" /> \
|
|
<method name="GrabAccelerator"> \
|
|
<arg type="s" direction="in" name="accelerator"/> \
|
|
<arg type="u" direction="in" name="flags"/> \
|
|
<arg type="u" direction="out" name="action"/> \
|
|
</method> \
|
|
<method name="GrabAccelerators"> \
|
|
<arg type="a(su)" direction="in" name="accelerators"/> \
|
|
<arg type="au" direction="out" name="actions"/> \
|
|
</method> \
|
|
<method name="UngrabAccelerator"> \
|
|
<arg type="u" direction="in" name="action"/> \
|
|
<arg type="b" direction="out" name="success"/> \
|
|
</method> \
|
|
<signal name="AcceleratorActivated"> \
|
|
<arg name="action" type="u" /> \
|
|
<arg name="parameters" type="a{sv}" /> \
|
|
</signal> \
|
|
<property name="Mode" type="s" access="read" /> \
|
|
<property name="OverviewActive" type="b" access="readwrite" /> \
|
|
<property name="ShellVersion" type="s" access="read" /> \
|
|
</interface> \
|
|
</node>';
|
|
|
|
const ScreenSaverIface = '<node> \
|
|
<interface name="org.gnome.ScreenSaver"> \
|
|
<method name="Lock"> \
|
|
</method> \
|
|
<method name="GetActive"> \
|
|
<arg name="active" direction="out" type="b" /> \
|
|
</method> \
|
|
<method name="SetActive"> \
|
|
<arg name="value" direction="in" type="b" /> \
|
|
</method> \
|
|
<method name="GetActiveTime"> \
|
|
<arg name="value" direction="out" type="u" /> \
|
|
</method> \
|
|
<signal name="ActiveChanged"> \
|
|
<arg name="new_value" type="b" /> \
|
|
</signal> \
|
|
<signal name="WakeUpScreen" /> \
|
|
</interface> \
|
|
</node>';
|
|
|
|
var GnomeShell = new Lang.Class({
|
|
Name: 'GnomeShellDBus',
|
|
|
|
_init() {
|
|
this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(GnomeShellIface, this);
|
|
this._dbusImpl.export(Gio.DBus.session, '/org/gnome/Shell');
|
|
|
|
this._extensionsService = new GnomeShellExtensions();
|
|
this._screenshotService = new Screenshot.ScreenshotService();
|
|
|
|
this._grabbedAccelerators = new Map();
|
|
this._grabbers = new Map();
|
|
|
|
global.display.connect('accelerator-activated',
|
|
(display, action, deviceid, timestamp) => {
|
|
this._emitAcceleratorActivated(action, deviceid, timestamp);
|
|
});
|
|
|
|
this._cachedOverviewVisible = false;
|
|
Main.overview.connect('showing',
|
|
this._checkOverviewVisibleChanged.bind(this));
|
|
Main.overview.connect('hidden',
|
|
this._checkOverviewVisibleChanged.bind(this));
|
|
},
|
|
|
|
/**
|
|
* Eval:
|
|
* @code: A string containing JavaScript code
|
|
*
|
|
* This function executes arbitrary code in the main
|
|
* loop, and returns a boolean success and
|
|
* JSON representation of the object as a string.
|
|
*
|
|
* If evaluation completes without throwing an exception,
|
|
* then the return value will be [true, JSON.stringify(result)].
|
|
* If evaluation fails, then the return value will be
|
|
* [false, JSON.stringify(exception)];
|
|
*
|
|
*/
|
|
Eval(code) {
|
|
if (!global.settings.get_boolean('development-tools'))
|
|
return [false, ''];
|
|
|
|
let returnValue;
|
|
let success;
|
|
try {
|
|
returnValue = JSON.stringify(eval(code));
|
|
// A hack; DBus doesn't have null/undefined
|
|
if (returnValue == undefined)
|
|
returnValue = '';
|
|
success = true;
|
|
} catch (e) {
|
|
returnValue = '' + e;
|
|
success = false;
|
|
}
|
|
return [success, returnValue];
|
|
},
|
|
|
|
FocusSearch() {
|
|
Main.overview.focusSearch();
|
|
},
|
|
|
|
ShowOSD(params) {
|
|
for (let param in params)
|
|
params[param] = params[param].deep_unpack();
|
|
|
|
let monitorIndex = params['monitor'] || -1;
|
|
let label = params['label'] || undefined;
|
|
let level = params['level'] || undefined;
|
|
|
|
let icon = null;
|
|
if (params['icon'])
|
|
icon = Gio.Icon.new_for_string(params['icon']);
|
|
|
|
Main.osdWindowManager.show(monitorIndex, icon, label, level);
|
|
},
|
|
|
|
FocusApp(id) {
|
|
this.ShowApplications();
|
|
Main.overview.viewSelector.appDisplay.selectApp(id);
|
|
},
|
|
|
|
ShowApplications() {
|
|
Main.overview.viewSelector.showApps();
|
|
},
|
|
|
|
GrabAcceleratorAsync(params, invocation) {
|
|
let [accel, flags] = params;
|
|
let sender = invocation.get_sender();
|
|
let bindingAction = this._grabAcceleratorForSender(accel, flags, sender);
|
|
return invocation.return_value(GLib.Variant.new('(u)', [bindingAction]));
|
|
},
|
|
|
|
GrabAcceleratorsAsync(params, invocation) {
|
|
let [accels] = params;
|
|
let sender = invocation.get_sender();
|
|
let bindingActions = [];
|
|
for (let i = 0; i < accels.length; i++) {
|
|
let [accel, flags] = accels[i];
|
|
bindingActions.push(this._grabAcceleratorForSender(accel, flags, sender));
|
|
}
|
|
return invocation.return_value(GLib.Variant.new('(au)', [bindingActions]));
|
|
},
|
|
|
|
UngrabAcceleratorAsync(params, invocation) {
|
|
let [action] = params;
|
|
let grabbedBy = this._grabbedAccelerators.get(action);
|
|
if (invocation.get_sender() != grabbedBy)
|
|
return invocation.return_value(GLib.Variant.new('(b)', [false]));
|
|
|
|
let ungrabSucceeded = global.display.ungrab_accelerator(action);
|
|
if (ungrabSucceeded)
|
|
this._grabbedAccelerators.delete(action);
|
|
return invocation.return_value(GLib.Variant.new('(b)', [ungrabSucceeded]));
|
|
},
|
|
|
|
_emitAcceleratorActivated(action, deviceid, timestamp) {
|
|
let destination = this._grabbedAccelerators.get(action);
|
|
if (!destination)
|
|
return;
|
|
|
|
let connection = this._dbusImpl.get_connection();
|
|
let info = this._dbusImpl.get_info();
|
|
let params = { 'device-id': GLib.Variant.new('u', deviceid),
|
|
'timestamp': GLib.Variant.new('u', timestamp),
|
|
'action-mode': GLib.Variant.new('u', Main.actionMode) };
|
|
connection.emit_signal(destination,
|
|
this._dbusImpl.get_object_path(),
|
|
info ? info.name : null,
|
|
'AcceleratorActivated',
|
|
GLib.Variant.new('(ua{sv})', [action, params]));
|
|
},
|
|
|
|
_grabAcceleratorForSender(accelerator, flags, sender) {
|
|
let bindingAction = global.display.grab_accelerator(accelerator);
|
|
if (bindingAction == Meta.KeyBindingAction.NONE)
|
|
return Meta.KeyBindingAction.NONE;
|
|
|
|
let bindingName = Meta.external_binding_name_for_action(bindingAction);
|
|
Main.wm.allowKeybinding(bindingName, flags);
|
|
|
|
this._grabbedAccelerators.set(bindingAction, sender);
|
|
|
|
if (!this._grabbers.has(sender)) {
|
|
let id = Gio.bus_watch_name(Gio.BusType.SESSION, sender, 0, null,
|
|
this._onGrabberBusNameVanished.bind(this));
|
|
this._grabbers.set(sender, id);
|
|
}
|
|
|
|
return bindingAction;
|
|
},
|
|
|
|
_ungrabAccelerator(action) {
|
|
let ungrabSucceeded = global.display.ungrab_accelerator(action);
|
|
if (ungrabSucceeded)
|
|
this._grabbedAccelerators.delete(action);
|
|
},
|
|
|
|
_onGrabberBusNameVanished(connection, name) {
|
|
let grabs = this._grabbedAccelerators.entries();
|
|
for (let [action, sender] of grabs) {
|
|
if (sender == name)
|
|
this._ungrabAccelerator(action);
|
|
}
|
|
Gio.bus_unwatch_name(this._grabbers.get(name));
|
|
this._grabbers.delete(name);
|
|
},
|
|
|
|
ShowMonitorLabelsAsync(params, invocation) {
|
|
let sender = invocation.get_sender();
|
|
let [dict] = params;
|
|
Main.osdMonitorLabeler.show(sender, dict);
|
|
},
|
|
|
|
ShowMonitorLabels2Async(params, invocation) {
|
|
let sender = invocation.get_sender();
|
|
let [dict] = params;
|
|
Main.osdMonitorLabeler.show2(sender, dict);
|
|
},
|
|
|
|
HideMonitorLabelsAsync(params, invocation) {
|
|
let sender = invocation.get_sender();
|
|
Main.osdMonitorLabeler.hide(sender);
|
|
},
|
|
|
|
|
|
Mode: global.session_mode,
|
|
|
|
_checkOverviewVisibleChanged() {
|
|
if (Main.overview.visible !== this._cachedOverviewVisible) {
|
|
this._cachedOverviewVisible = Main.overview.visible;
|
|
this._dbusImpl.emit_property_changed('OverviewActive', new GLib.Variant('b', this._cachedOverviewVisible));
|
|
}
|
|
},
|
|
|
|
get OverviewActive() {
|
|
return this._cachedOverviewVisible;
|
|
},
|
|
|
|
set OverviewActive(visible) {
|
|
if (visible)
|
|
Main.overview.show();
|
|
else
|
|
Main.overview.hide();
|
|
},
|
|
|
|
ShellVersion: Config.PACKAGE_VERSION
|
|
});
|
|
|
|
const GnomeShellExtensionsIface = '<node> \
|
|
<interface name="org.gnome.Shell.Extensions"> \
|
|
<method name="ListExtensions"> \
|
|
<arg type="a{sa{sv}}" direction="out" name="extensions" /> \
|
|
</method> \
|
|
<method name="GetExtensionInfo"> \
|
|
<arg type="s" direction="in" name="extension" /> \
|
|
<arg type="a{sv}" direction="out" name="info" /> \
|
|
</method> \
|
|
<method name="GetExtensionErrors"> \
|
|
<arg type="s" direction="in" name="extension" /> \
|
|
<arg type="as" direction="out" name="errors" /> \
|
|
</method> \
|
|
<signal name="ExtensionStatusChanged"> \
|
|
<arg type="s" name="uuid"/> \
|
|
<arg type="i" name="state"/> \
|
|
<arg type="s" name="error"/> \
|
|
</signal> \
|
|
<method name="InstallRemoteExtension"> \
|
|
<arg type="s" direction="in" name="uuid"/> \
|
|
<arg type="s" direction="out" name="result"/> \
|
|
</method> \
|
|
<method name="UninstallExtension"> \
|
|
<arg type="s" direction="in" name="uuid"/> \
|
|
<arg type="b" direction="out" name="success"/> \
|
|
</method> \
|
|
<method name="LaunchExtensionPrefs"> \
|
|
<arg type="s" direction="in" name="uuid"/> \
|
|
</method> \
|
|
<method name="ReloadExtension"> \
|
|
<arg type="s" direction="in" name="uuid"/> \
|
|
</method> \
|
|
<method name="CheckForUpdates"> \
|
|
</method> \
|
|
<method name="LoadUserExtension"> \
|
|
<arg type="s" direction="in" name="uuid"/> \
|
|
<arg type="b" direction="out" name="success"/> \
|
|
</method> \
|
|
<property name="ShellVersion" type="s" access="read" /> \
|
|
</interface> \
|
|
</node>';
|
|
|
|
var GnomeShellExtensions = new Lang.Class({
|
|
Name: 'GnomeShellExtensionsDBus',
|
|
|
|
_init() {
|
|
this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(GnomeShellExtensionsIface, this);
|
|
this._dbusImpl.export(Gio.DBus.session, '/org/gnome/Shell');
|
|
ExtensionSystem.connect('extension-state-changed',
|
|
this._extensionStateChanged.bind(this));
|
|
},
|
|
|
|
|
|
ListExtensions() {
|
|
let out = {};
|
|
for (let uuid in ExtensionUtils.extensions) {
|
|
let dbusObj = this.GetExtensionInfo(uuid);
|
|
out[uuid] = dbusObj;
|
|
}
|
|
return out;
|
|
},
|
|
|
|
GetExtensionInfo(uuid) {
|
|
let extension = ExtensionUtils.extensions[uuid];
|
|
if (!extension)
|
|
return {};
|
|
|
|
let obj = {};
|
|
Lang.copyProperties(extension.metadata, obj);
|
|
|
|
// Only serialize the properties that we actually need.
|
|
const serializedProperties = ["type", "state", "path", "error", "hasPrefs"];
|
|
|
|
serializedProperties.forEach(prop => {
|
|
obj[prop] = extension[prop];
|
|
});
|
|
|
|
let out = {};
|
|
for (let key in obj) {
|
|
let val = obj[key];
|
|
let type;
|
|
switch (typeof val) {
|
|
case 'string':
|
|
type = 's';
|
|
break;
|
|
case 'number':
|
|
type = 'd';
|
|
break;
|
|
case 'boolean':
|
|
type = 'b';
|
|
break;
|
|
default:
|
|
continue;
|
|
}
|
|
out[key] = GLib.Variant.new(type, val);
|
|
}
|
|
|
|
return out;
|
|
},
|
|
|
|
GetExtensionErrors(uuid) {
|
|
let extension = ExtensionUtils.extensions[uuid];
|
|
if (!extension)
|
|
return [];
|
|
|
|
if (!extension.errors)
|
|
return [];
|
|
|
|
return extension.errors;
|
|
},
|
|
|
|
InstallRemoteExtensionAsync([uuid], invocation) {
|
|
return ExtensionDownloader.installExtension(uuid, invocation);
|
|
},
|
|
|
|
UninstallExtension(uuid) {
|
|
return ExtensionDownloader.uninstallExtension(uuid);
|
|
},
|
|
|
|
LaunchExtensionPrefs(uuid) {
|
|
let appSys = Shell.AppSystem.get_default();
|
|
let app = appSys.lookup_app('gnome-shell-extension-prefs.desktop');
|
|
let info = app.get_app_info();
|
|
let timestamp = global.display.get_current_time_roundtrip();
|
|
info.launch_uris(['extension:///' + uuid],
|
|
global.create_app_launch_context(timestamp, -1));
|
|
},
|
|
|
|
ReloadExtension(uuid) {
|
|
let extension = ExtensionUtils.extensions[uuid];
|
|
if (!extension)
|
|
return;
|
|
|
|
ExtensionSystem.reloadExtension(extension);
|
|
},
|
|
|
|
CheckForUpdates() {
|
|
ExtensionDownloader.checkForUpdates();
|
|
},
|
|
|
|
LoadUserExtension(uuid) {
|
|
let extension = ExtensionUtils.extensions[uuid];
|
|
if (extension)
|
|
return true;
|
|
|
|
let dir = Gio.File.new_for_path(GLib.build_filenamev([global.userdatadir, 'extensions', uuid]));
|
|
try {
|
|
extension = ExtensionUtils.createExtensionObject(uuid, dir, ExtensionUtils.ExtensionType.PER_USER);
|
|
ExtensionSystem.loadExtension(extension);
|
|
} catch (e) {
|
|
log('Could not load user extension from %s'.format(dir.get_path()));
|
|
return false;
|
|
}
|
|
return true;
|
|
},
|
|
|
|
ShellVersion: Config.PACKAGE_VERSION,
|
|
|
|
_extensionStateChanged(_, newState) {
|
|
this._dbusImpl.emit_signal('ExtensionStatusChanged',
|
|
GLib.Variant.new('(sis)', [newState.uuid, newState.state, newState.error]));
|
|
}
|
|
});
|
|
|
|
var ScreenSaverDBus = new Lang.Class({
|
|
Name: 'ScreenSaverDBus',
|
|
|
|
_init(screenShield) {
|
|
this.parent();
|
|
|
|
this._screenShield = screenShield;
|
|
screenShield.connect('active-changed', shield => {
|
|
this._dbusImpl.emit_signal('ActiveChanged', GLib.Variant.new('(b)', [shield.active]));
|
|
});
|
|
screenShield.connect('wake-up-screen', shield => {
|
|
this._dbusImpl.emit_signal('WakeUpScreen', null);
|
|
});
|
|
|
|
this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(ScreenSaverIface, this);
|
|
this._dbusImpl.export(Gio.DBus.session, '/org/gnome/ScreenSaver');
|
|
|
|
Gio.DBus.session.own_name('org.gnome.ScreenSaver', Gio.BusNameOwnerFlags.REPLACE, null, null);
|
|
},
|
|
|
|
LockAsync(parameters, invocation) {
|
|
let tmpId = this._screenShield.connect('lock-screen-shown', () => {
|
|
this._screenShield.disconnect(tmpId);
|
|
|
|
invocation.return_value(null);
|
|
});
|
|
|
|
this._screenShield.lock(true);
|
|
},
|
|
|
|
SetActive(active) {
|
|
if (active)
|
|
this._screenShield.activate(true);
|
|
else
|
|
this._screenShield.deactivate(false);
|
|
},
|
|
|
|
GetActive() {
|
|
return this._screenShield.active;
|
|
},
|
|
|
|
GetActiveTime() {
|
|
let started = this._screenShield.activationTime;
|
|
if (started > 0)
|
|
return Math.floor((GLib.get_monotonic_time() - started) / 1000000);
|
|
else
|
|
return 0;
|
|
},
|
|
});
|