2013-05-06 18:50:31 +02:00
|
|
|
// -*- 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 Shell = imports.gi.Shell;
|
2013-07-17 15:26:06 -04:00
|
|
|
const Signals = imports.signals;
|
2013-05-06 18:50:31 +02:00
|
|
|
|
|
|
|
const Main = imports.ui.main;
|
|
|
|
|
2013-10-25 08:51:58 +11:00
|
|
|
const ScreencastIface = '<node> \
|
|
|
|
<interface name="org.gnome.Shell.Screencast"> \
|
|
|
|
<method name="Screencast"> \
|
|
|
|
<arg type="s" direction="in" name="file_template"/> \
|
|
|
|
<arg type="a{sv}" direction="in" name="options"/> \
|
|
|
|
<arg type="b" direction="out" name="success"/> \
|
|
|
|
<arg type="s" direction="out" name="filename_used"/> \
|
|
|
|
</method> \
|
|
|
|
<method name="ScreencastArea"> \
|
|
|
|
<arg type="i" direction="in" name="x"/> \
|
|
|
|
<arg type="i" direction="in" name="y"/> \
|
|
|
|
<arg type="i" direction="in" name="width"/> \
|
|
|
|
<arg type="i" direction="in" name="height"/> \
|
|
|
|
<arg type="s" direction="in" name="file_template"/> \
|
|
|
|
<arg type="a{sv}" direction="in" name="options"/> \
|
|
|
|
<arg type="b" direction="out" name="success"/> \
|
|
|
|
<arg type="s" direction="out" name="filename_used"/> \
|
|
|
|
</method> \
|
|
|
|
<method name="StopScreencast"> \
|
|
|
|
<arg type="b" direction="out" name="success"/> \
|
|
|
|
</method> \
|
|
|
|
</interface> \
|
|
|
|
</node>';
|
2013-05-06 18:50:31 +02:00
|
|
|
|
2017-07-18 19:41:25 +02:00
|
|
|
var ScreencastService = new Lang.Class({
|
2013-05-06 18:50:31 +02:00
|
|
|
Name: 'ScreencastService',
|
|
|
|
|
|
|
|
_init: function() {
|
|
|
|
this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(ScreencastIface, this);
|
|
|
|
this._dbusImpl.export(Gio.DBus.session, '/org/gnome/Shell/Screencast');
|
|
|
|
|
|
|
|
Gio.DBus.session.own_name('org.gnome.Shell.Screencast', Gio.BusNameOwnerFlags.REPLACE, null, null);
|
|
|
|
|
2014-01-14 23:49:47 +01:00
|
|
|
this._recorders = new Map();
|
2013-05-06 18:50:31 +02:00
|
|
|
|
2014-10-03 16:40:49 +02:00
|
|
|
this._lockdownSettings = new Gio.Settings({ schema_id: 'org.gnome.desktop.lockdown' });
|
|
|
|
|
2013-07-17 15:24:20 -04:00
|
|
|
Main.sessionMode.connect('updated', Lang.bind(this, this._sessionUpdated));
|
2013-05-06 18:50:31 +02:00
|
|
|
},
|
|
|
|
|
2013-07-17 15:26:06 -04:00
|
|
|
get isRecording() {
|
2014-01-14 23:49:47 +01:00
|
|
|
return this._recorders.size > 0;
|
2013-07-17 15:26:06 -04:00
|
|
|
},
|
|
|
|
|
2013-05-06 18:50:31 +02:00
|
|
|
_ensureRecorderForSender: function(sender) {
|
|
|
|
let recorder = this._recorders.get(sender);
|
|
|
|
if (!recorder) {
|
2013-08-14 10:34:58 +02:00
|
|
|
recorder = new Shell.Recorder({ stage: global.stage,
|
|
|
|
screen: global.screen });
|
2013-05-06 18:50:31 +02:00
|
|
|
recorder._watchNameId =
|
|
|
|
Gio.bus_watch_name(Gio.BusType.SESSION, sender, 0, null,
|
|
|
|
Lang.bind(this, this._onNameVanished));
|
|
|
|
this._recorders.set(sender, recorder);
|
2013-07-17 15:26:06 -04:00
|
|
|
this.emit('updated');
|
2013-05-06 18:50:31 +02:00
|
|
|
}
|
|
|
|
return recorder;
|
|
|
|
},
|
|
|
|
|
2013-07-17 15:24:20 -04:00
|
|
|
_sessionUpdated: function() {
|
2013-05-06 18:50:31 +02:00
|
|
|
if (Main.sessionMode.allowScreencast)
|
|
|
|
return;
|
|
|
|
|
2014-10-03 14:46:36 +02:00
|
|
|
for (let sender of this._recorders.keys())
|
|
|
|
this._stopRecordingForSender(sender);
|
2013-05-06 18:50:31 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
_onNameVanished: function(connection, name) {
|
|
|
|
this._stopRecordingForSender(name);
|
|
|
|
},
|
|
|
|
|
|
|
|
_stopRecordingForSender: function(sender) {
|
|
|
|
let recorder = this._recorders.get(sender);
|
|
|
|
if (!recorder)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Gio.bus_unwatch_name(recorder._watchNameId);
|
|
|
|
recorder.close();
|
|
|
|
this._recorders.delete(sender);
|
2013-07-17 15:26:06 -04:00
|
|
|
this.emit('updated');
|
2013-05-06 18:50:31 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
|
|
|
|
_applyOptionalParameters: function(recorder, options) {
|
|
|
|
for (let option in options)
|
|
|
|
options[option] = options[option].deep_unpack();
|
|
|
|
|
|
|
|
if (options['pipeline'])
|
|
|
|
recorder.set_pipeline(options['pipeline']);
|
|
|
|
if (options['framerate'])
|
|
|
|
recorder.set_framerate(options['framerate']);
|
2015-02-16 16:56:05 +01:00
|
|
|
if ('draw-cursor' in options)
|
2013-05-06 18:50:31 +02:00
|
|
|
recorder.set_draw_cursor(options['draw-cursor']);
|
|
|
|
},
|
|
|
|
|
|
|
|
ScreencastAsync: function(params, invocation) {
|
|
|
|
let returnValue = [false, ''];
|
2014-10-03 16:40:49 +02:00
|
|
|
if (!Main.sessionMode.allowScreencast ||
|
|
|
|
this._lockdownSettings.get_boolean('disable-save-to-disk')) {
|
2013-05-06 18:50:31 +02:00
|
|
|
invocation.return_value(GLib.Variant.new('(bs)', returnValue));
|
2013-11-04 14:06:33 +01:00
|
|
|
return;
|
|
|
|
}
|
2013-05-06 18:50:31 +02:00
|
|
|
|
|
|
|
let sender = invocation.get_sender();
|
|
|
|
let recorder = this._ensureRecorderForSender(sender);
|
|
|
|
if (!recorder.is_recording()) {
|
|
|
|
let [fileTemplate, options] = params;
|
|
|
|
|
|
|
|
recorder.set_file_template(fileTemplate);
|
|
|
|
this._applyOptionalParameters(recorder, options);
|
2013-05-22 16:42:00 +02:00
|
|
|
let [success, fileName] = recorder.record();
|
|
|
|
returnValue = [success, fileName ? fileName : ''];
|
2014-10-03 16:49:21 +02:00
|
|
|
if (!success)
|
|
|
|
this._stopRecordingForSender(sender);
|
2013-05-06 18:50:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
invocation.return_value(GLib.Variant.new('(bs)', returnValue));
|
|
|
|
},
|
|
|
|
|
|
|
|
ScreencastAreaAsync: function(params, invocation) {
|
|
|
|
let returnValue = [false, ''];
|
2014-10-03 16:40:49 +02:00
|
|
|
if (!Main.sessionMode.allowScreencast ||
|
|
|
|
this._lockdownSettings.get_boolean('disable-save-to-disk')) {
|
2013-05-06 18:50:31 +02:00
|
|
|
invocation.return_value(GLib.Variant.new('(bs)', returnValue));
|
2013-11-04 14:06:33 +01:00
|
|
|
return;
|
|
|
|
}
|
2013-05-06 18:50:31 +02:00
|
|
|
|
|
|
|
let sender = invocation.get_sender();
|
|
|
|
let recorder = this._ensureRecorderForSender(sender);
|
|
|
|
|
|
|
|
if (!recorder.is_recording()) {
|
|
|
|
let [x, y, width, height, fileTemplate, options] = params;
|
|
|
|
|
2013-11-04 14:09:19 +01:00
|
|
|
if (x < 0 || y < 0 ||
|
|
|
|
width <= 0 || height <= 0 ||
|
|
|
|
x + width > global.screen_width ||
|
|
|
|
y + height > global.screen_height) {
|
|
|
|
invocation.return_error_literal(Gio.IOErrorEnum,
|
|
|
|
Gio.IOErrorEnum.CANCELLED,
|
|
|
|
"Invalid params");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-05-06 18:50:31 +02:00
|
|
|
recorder.set_file_template(fileTemplate);
|
|
|
|
recorder.set_area(x, y, width, height);
|
|
|
|
this._applyOptionalParameters(recorder, options);
|
2013-05-22 16:42:00 +02:00
|
|
|
let [success, fileName] = recorder.record();
|
|
|
|
returnValue = [success, fileName ? fileName : ''];
|
2014-10-03 16:49:21 +02:00
|
|
|
if (!success)
|
|
|
|
this._stopRecordingForSender(sender);
|
2013-05-06 18:50:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
invocation.return_value(GLib.Variant.new('(bs)', returnValue));
|
|
|
|
},
|
|
|
|
|
|
|
|
StopScreencastAsync: function(params, invocation) {
|
|
|
|
let success = this._stopRecordingForSender(invocation.get_sender());
|
|
|
|
invocation.return_value(GLib.Variant.new('(b)', [success]));
|
|
|
|
}
|
|
|
|
});
|
2013-07-17 15:26:06 -04:00
|
|
|
Signals.addSignalMethods(ScreencastService.prototype);
|