2018-07-20 14:50:50 +00:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2022-02-08 16:18:01 +00:00
|
|
|
/* exported RemoteAccessApplet, ScreenRecordingIndicator */
|
2018-07-20 14:50:50 +00:00
|
|
|
|
2022-02-08 16:18:01 +00:00
|
|
|
const { Atk, Clutter, GLib, GObject, Meta, St } = imports.gi;
|
2018-07-20 14:50:50 +00:00
|
|
|
|
2021-08-17 07:58:23 +00:00
|
|
|
const Main = imports.ui.main;
|
2018-07-20 14:50:50 +00:00
|
|
|
const PanelMenu = imports.ui.panelMenu;
|
|
|
|
const PopupMenu = imports.ui.popupMenu;
|
|
|
|
|
2022-01-27 09:48:59 +00:00
|
|
|
// Minimum amount of time the shared indicator is visible (in micro seconds)
|
|
|
|
const MIN_SHARED_INDICATOR_VISIBLE_TIME_US = 5 * GLib.TIME_SPAN_SECOND;
|
|
|
|
|
2019-10-28 18:35:33 +00:00
|
|
|
var RemoteAccessApplet = GObject.registerClass(
|
|
|
|
class RemoteAccessApplet extends PanelMenu.SystemIndicator {
|
2019-07-16 09:24:13 +00:00
|
|
|
_init() {
|
|
|
|
super._init();
|
2018-07-20 14:50:50 +00:00
|
|
|
|
2020-04-21 16:04:56 +00:00
|
|
|
let controller = global.backend.get_remote_access_controller();
|
2018-07-20 14:50:50 +00:00
|
|
|
|
|
|
|
if (!controller)
|
|
|
|
return;
|
|
|
|
|
|
|
|
this._handles = new Set();
|
2020-04-21 16:06:06 +00:00
|
|
|
this._sharedIndicator = null;
|
|
|
|
this._recordingIndicator = null;
|
2018-07-20 14:50:50 +00:00
|
|
|
this._menuSection = null;
|
|
|
|
|
2019-08-20 00:20:08 +00:00
|
|
|
controller.connect('new-handle', (o, handle) => {
|
2018-07-20 14:50:50 +00:00
|
|
|
this._onNewHandle(handle);
|
|
|
|
});
|
2017-10-31 01:19:44 +00:00
|
|
|
}
|
2018-07-20 14:50:50 +00:00
|
|
|
|
|
|
|
_ensureControls() {
|
2020-04-21 16:06:06 +00:00
|
|
|
if (this._sharedIndicator && this._recordingIndicator)
|
2018-07-20 14:50:50 +00:00
|
|
|
return;
|
|
|
|
|
2020-04-21 16:06:06 +00:00
|
|
|
this._sharedIndicator = this._addIndicator();
|
2022-01-27 09:48:59 +00:00
|
|
|
this._sharedIndicator.visible = false;
|
2020-04-21 16:06:06 +00:00
|
|
|
this._sharedIndicator.icon_name = 'screen-shared-symbolic';
|
|
|
|
this._sharedIndicator.add_style_class_name('remote-access-indicator');
|
|
|
|
|
|
|
|
this._sharedItem =
|
2018-07-20 14:50:50 +00:00
|
|
|
new PopupMenu.PopupSubMenuMenuItem(_("Screen is Being Shared"),
|
|
|
|
true);
|
2020-04-21 16:06:06 +00:00
|
|
|
this._sharedItem.menu.addAction(_("Turn off"),
|
|
|
|
() => {
|
|
|
|
for (let handle of this._handles) {
|
|
|
|
if (!handle.is_recording)
|
|
|
|
handle.stop();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
this._sharedItem.icon.icon_name = 'screen-shared-symbolic';
|
|
|
|
this.menu.addMenuItem(this._sharedItem);
|
|
|
|
|
|
|
|
this._recordingIndicator = this._addIndicator();
|
|
|
|
this._recordingIndicator.icon_name = 'media-record-symbolic';
|
|
|
|
this._recordingIndicator.add_style_class_name('screencast-indicator');
|
|
|
|
}
|
|
|
|
|
|
|
|
_isScreenShared() {
|
|
|
|
return [...this._handles].some(handle => !handle.is_recording);
|
|
|
|
}
|
|
|
|
|
|
|
|
_isRecording() {
|
2022-02-09 07:00:07 +00:00
|
|
|
const recordingHandles =
|
|
|
|
[...this._handles].filter(handle => handle.is_recording);
|
|
|
|
|
|
|
|
// Screenshot UI screencasts have their own panel, so don't show this
|
|
|
|
// indicator if there's only a screenshot UI screencast.
|
|
|
|
if (Main.screenshotUI.screencast_in_progress)
|
|
|
|
return recordingHandles.length > 1;
|
|
|
|
|
|
|
|
return recordingHandles.length > 0;
|
2017-10-31 01:19:44 +00:00
|
|
|
}
|
2018-07-20 14:50:50 +00:00
|
|
|
|
2022-01-27 09:48:59 +00:00
|
|
|
_hideSharedIndicator() {
|
|
|
|
this._sharedIndicator.visible = false;
|
|
|
|
delete this._hideSharedIndicatorId;
|
|
|
|
return GLib.SOURCE_REMOVE;
|
|
|
|
}
|
|
|
|
|
2018-07-20 14:50:50 +00:00
|
|
|
_sync() {
|
2022-01-27 09:48:59 +00:00
|
|
|
if (this._hideSharedIndicatorId) {
|
|
|
|
GLib.source_remove(this._hideSharedIndicatorId);
|
|
|
|
delete this._hideSharedIndicatorId;
|
|
|
|
}
|
|
|
|
|
2020-04-21 16:06:06 +00:00
|
|
|
if (this._isScreenShared()) {
|
2022-01-27 09:48:59 +00:00
|
|
|
if (!this._sharedIndicator.visible)
|
|
|
|
this._visibleTimeUs = GLib.get_monotonic_time();
|
2020-04-21 16:06:06 +00:00
|
|
|
this._sharedIndicator.visible = true;
|
|
|
|
this._sharedItem.visible = true;
|
2018-07-20 14:50:50 +00:00
|
|
|
} else {
|
2022-01-27 09:48:59 +00:00
|
|
|
if (this._sharedIndicator.visible) {
|
|
|
|
const currentTimeUs = GLib.get_monotonic_time();
|
|
|
|
const timeSinceVisibleUs = currentTimeUs - this._visibleTimeUs;
|
|
|
|
|
|
|
|
if (timeSinceVisibleUs >= MIN_SHARED_INDICATOR_VISIBLE_TIME_US) {
|
|
|
|
this._hideSharedIndicator();
|
|
|
|
} else {
|
|
|
|
const timeUntilHideUs =
|
|
|
|
MIN_SHARED_INDICATOR_VISIBLE_TIME_US - timeSinceVisibleUs;
|
|
|
|
this._hideSharedIndicatorId =
|
|
|
|
GLib.timeout_add(
|
|
|
|
GLib.PRIORITY_DEFAULT,
|
|
|
|
timeUntilHideUs / GLib.TIME_SPAN_MILLISECOND,
|
|
|
|
this._hideSharedIndicator.bind(this));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-21 16:06:06 +00:00
|
|
|
this._sharedItem.visible = false;
|
2018-07-20 14:50:50 +00:00
|
|
|
}
|
2020-04-21 16:06:06 +00:00
|
|
|
|
|
|
|
this._recordingIndicator.visible = this._isRecording();
|
2017-10-31 01:19:44 +00:00
|
|
|
}
|
2018-07-20 14:50:50 +00:00
|
|
|
|
|
|
|
_onStopped(handle) {
|
|
|
|
this._handles.delete(handle);
|
|
|
|
this._sync();
|
2017-10-31 01:19:44 +00:00
|
|
|
}
|
2018-07-20 14:50:50 +00:00
|
|
|
|
|
|
|
_onNewHandle(handle) {
|
2020-10-13 14:28:51 +00:00
|
|
|
// We can't possibly know about all types of screen sharing on X11, so
|
|
|
|
// showing these controls on X11 might give a false sense of security.
|
|
|
|
// Thus, only enable these controls when using Wayland, where we are
|
|
|
|
// in control of sharing.
|
|
|
|
//
|
|
|
|
// We still want to show screen recordings though, to indicate when
|
|
|
|
// the built in screen recorder is active, no matter the session type.
|
|
|
|
if (!Meta.is_wayland_compositor() && !handle.is_recording)
|
|
|
|
return;
|
|
|
|
|
2018-07-20 14:50:50 +00:00
|
|
|
this._handles.add(handle);
|
|
|
|
handle.connect('stopped', this._onStopped.bind(this));
|
|
|
|
|
2020-04-21 16:06:06 +00:00
|
|
|
this._ensureControls();
|
|
|
|
this._sync();
|
2017-10-31 01:19:44 +00:00
|
|
|
}
|
2019-07-16 09:24:13 +00:00
|
|
|
});
|
2022-02-08 16:18:01 +00:00
|
|
|
|
|
|
|
var ScreenRecordingIndicator = GObject.registerClass({
|
|
|
|
Signals: { 'menu-set': {} },
|
|
|
|
}, class ScreenRecordingIndicator extends PanelMenu.ButtonBox {
|
|
|
|
_init() {
|
|
|
|
super._init({
|
|
|
|
reactive: true,
|
|
|
|
can_focus: true,
|
|
|
|
track_hover: true,
|
|
|
|
accessible_name: _('Stop Screencast'),
|
|
|
|
accessible_role: Atk.Role.PUSH_BUTTON,
|
|
|
|
});
|
|
|
|
this.add_style_class_name('screen-recording-indicator');
|
|
|
|
|
|
|
|
this._box = new St.BoxLayout();
|
|
|
|
this.add_child(this._box);
|
|
|
|
|
|
|
|
this._label = new St.Label({
|
|
|
|
text: '0:00',
|
|
|
|
y_align: Clutter.ActorAlign.CENTER,
|
|
|
|
});
|
|
|
|
this._box.add_child(this._label);
|
|
|
|
|
|
|
|
this._icon = new St.Icon({ icon_name: 'stop-symbolic' });
|
|
|
|
this._box.add_child(this._icon);
|
|
|
|
|
|
|
|
this.hide();
|
|
|
|
Main.screenshotUI.connect(
|
|
|
|
'notify::screencast-in-progress',
|
|
|
|
this._onScreencastInProgressChanged.bind(this));
|
|
|
|
}
|
|
|
|
|
|
|
|
vfunc_event(event) {
|
|
|
|
if (event.type() === Clutter.EventType.TOUCH_BEGIN ||
|
|
|
|
event.type() === Clutter.EventType.BUTTON_PRESS)
|
|
|
|
Main.screenshotUI.stopScreencast();
|
|
|
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
_updateLabel() {
|
|
|
|
const minutes = this._secondsPassed / 60;
|
|
|
|
const seconds = this._secondsPassed % 60;
|
|
|
|
this._label.text = '%d:%02d'.format(minutes, seconds);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onScreencastInProgressChanged() {
|
|
|
|
if (Main.screenshotUI.screencast_in_progress) {
|
|
|
|
this.show();
|
|
|
|
|
|
|
|
this._secondsPassed = 0;
|
|
|
|
this._updateLabel();
|
|
|
|
|
|
|
|
this._timeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT, 1000, () => {
|
|
|
|
this._secondsPassed += 1;
|
|
|
|
this._updateLabel();
|
|
|
|
return GLib.SOURCE_CONTINUE;
|
|
|
|
});
|
|
|
|
GLib.Source.set_name_by_id(
|
|
|
|
this._timeoutId, '[gnome-shell] screen recording indicator tick');
|
|
|
|
} else {
|
|
|
|
this.hide();
|
|
|
|
|
|
|
|
GLib.source_remove(this._timeoutId);
|
|
|
|
delete this._timeoutId;
|
|
|
|
|
|
|
|
delete this._secondsPassed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|