Blame SOURCES/0001-heads-up-display-Add-extension-for-showing-persisten.patch

001c36
From 8da1760af68496c6073be4d6b3c8266b64347925 Mon Sep 17 00:00:00 2001
20dab4
From: Ray Strode <rstrode@redhat.com>
20dab4
Date: Tue, 24 Aug 2021 15:03:57 -0400
20dab4
Subject: [PATCH] heads-up-display: Add extension for showing persistent heads
20dab4
 up display message
20dab4
20dab4
---
20dab4
 extensions/heads-up-display/extension.js      | 320 ++++++++++++++++++
20dab4
 extensions/heads-up-display/headsUpMessage.js | 150 ++++++++
20dab4
 extensions/heads-up-display/meson.build       |   8 +
20dab4
 extensions/heads-up-display/metadata.json.in  |  11 +
20dab4
 ...ll.extensions.heads-up-display.gschema.xml |  54 +++
20dab4
 extensions/heads-up-display/prefs.js          | 175 ++++++++++
20dab4
 extensions/heads-up-display/stylesheet.css    |  32 ++
20dab4
 meson.build                                   |   1 +
20dab4
 8 files changed, 751 insertions(+)
20dab4
 create mode 100644 extensions/heads-up-display/extension.js
20dab4
 create mode 100644 extensions/heads-up-display/headsUpMessage.js
20dab4
 create mode 100644 extensions/heads-up-display/meson.build
20dab4
 create mode 100644 extensions/heads-up-display/metadata.json.in
20dab4
 create mode 100644 extensions/heads-up-display/org.gnome.shell.extensions.heads-up-display.gschema.xml
20dab4
 create mode 100644 extensions/heads-up-display/prefs.js
20dab4
 create mode 100644 extensions/heads-up-display/stylesheet.css
20dab4
20dab4
diff --git a/extensions/heads-up-display/extension.js b/extensions/heads-up-display/extension.js
20dab4
new file mode 100644
001c36
index 00000000..e4ef9e85
20dab4
--- /dev/null
20dab4
+++ b/extensions/heads-up-display/extension.js
20dab4
@@ -0,0 +1,320 @@
20dab4
+/* exported init enable disable */
20dab4
+
20dab4
+
20dab4
+const Signals = imports.signals;
20dab4
+
20dab4
+const {
20dab4
+    Atk, Clutter, Gio, GLib, GObject, Gtk, Meta, Shell, St,
20dab4
+} = imports.gi;
20dab4
+
20dab4
+const ExtensionUtils = imports.misc.extensionUtils;
20dab4
+const Me = ExtensionUtils.getCurrentExtension();
20dab4
+
20dab4
+const Main = imports.ui.main;
20dab4
+const HeadsUpMessage = Me.imports.headsUpMessage;
20dab4
+
20dab4
+const Gettext = imports.gettext.domain('gnome-shell-extensions');
20dab4
+const _ = Gettext.gettext;
20dab4
+
20dab4
+class Extension {
20dab4
+    constructor() {
20dab4
+        ExtensionUtils.initTranslations();
20dab4
+    }
20dab4
+
20dab4
+    enable() {
20dab4
+        this._settings = ExtensionUtils.getSettings('org.gnome.shell.extensions.heads-up-display');
20dab4
+        this._idleTimeoutChangedId = this._settings.connect('changed::idle-timeout', this._onIdleTimeoutChanged.bind(this));
20dab4
+        this._settingsChangedId = this._settings.connect('changed', this._updateMessage.bind(this));
20dab4
+
20dab4
+        this._idleMonitor = Meta.IdleMonitor.get_core();
20dab4
+        this._messageInhibitedUntilIdle = false;
20dab4
+        this._oldMapWindow = Main.wm._mapWindow;
20dab4
+        Main.wm._mapWindow = this._mapWindow;
20dab4
+        this._windowManagerMapId = global.window_manager.connect('map', this._onWindowMap.bind(this));
20dab4
+
20dab4
+        if (Main.layoutManager._startingUp)
20dab4
+            this._startupCompleteId = Main.layoutManager.connect('startup-complete', this._onStartupComplete.bind(this));
20dab4
+        else
20dab4
+            this._onStartupComplete(this);
20dab4
+    }
20dab4
+
20dab4
+    disable() {
20dab4
+        this._dismissMessage();
20dab4
+
20dab4
+        if (this._idleWatchId) {
20dab4
+            this._idleMonitor.remove_watch(this._idleWatchId);
20dab4
+            this._idleWatchId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (this._sessionModeUpdatedId) {
20dab4
+            Main.sessionMode.disconnect(this._sessionModeUpdatedId);
20dab4
+            this._sessionModeUpdatedId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (this._overviewShowingId) {
20dab4
+            Main.overview.disconnect(this._overviewShowingId);
20dab4
+            this._overviewShowingId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (this._overviewHiddenId) {
20dab4
+            Main.overview.disconnect(this._overviewHiddenId);
20dab4
+            this._overviewHiddenId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (this._panelConnectionId) {
20dab4
+            Main.layoutManager.panelBox.disconnect(this._panelConnectionId);
20dab4
+            this._panelConnectionId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (this._oldMapWindow) {
20dab4
+            Main.wm._mapWindow = this._oldMapWindow;
20dab4
+            this._oldMapWindow = null;
20dab4
+        }
20dab4
+
20dab4
+        if (this._windowManagerMapId) {
20dab4
+            global.window_manager.disconnect(this._windowManagerMapId);
20dab4
+            this._windowManagerMapId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (this._startupCompleteId) {
20dab4
+            Main.layoutManager.disconnect(this._startupCompleteId);
20dab4
+            this._startupCompleteId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (this._settingsChangedId) {
20dab4
+            this._settings.disconnect(this._settingsChangedId);
20dab4
+            this._settingsChangedId = 0;
20dab4
+        }
20dab4
+    }
20dab4
+
20dab4
+    _onWindowMap(shellwm, actor) {
20dab4
+        let windowObject = actor.meta_window;
20dab4
+        let windowType = windowObject.get_window_type();
20dab4
+
20dab4
+        if (windowType != Meta.WindowType.NORMAL)
20dab4
+            return;
20dab4
+
20dab4
+        if (!this._message || !this._message.visible)
20dab4
+            return;
20dab4
+
20dab4
+        let messageRect = new Meta.Rectangle({ x: this._message.x, y: this._message.y, width: this._message.width, height: this._message.height });
20dab4
+        let windowRect = windowObject.get_frame_rect();
20dab4
+
20dab4
+        if (windowRect.intersect(messageRect)) {
20dab4
+            windowObject.move_frame(false, windowRect.x, this._message.y + this._message.height);
20dab4
+        }
20dab4
+    }
20dab4
+
20dab4
+    _onStartupComplete() {
20dab4
+        this._overviewShowingId = Main.overview.connect('showing', this._updateMessage.bind(this));
20dab4
+        this._overviewHiddenId = Main.overview.connect('hidden', this._updateMessage.bind(this));
20dab4
+        this._panelConnectionId = Main.layoutManager.panelBox.connect('notify::visible', this._updateMessage.bind(this));
20dab4
+        this._sessionModeUpdatedId = Main.sessionMode.connect('updated', this._onSessionModeUpdated.bind(this));
20dab4
+
20dab4
+        this._updateMessage();
20dab4
+    }
20dab4
+
20dab4
+    _onSessionModeUpdated() {
20dab4
+         if (!Main.sessionMode.hasWindows)
20dab4
+             this._messageInhibitedUntilIdle = false;
20dab4
+         this._updateMessage();
20dab4
+    }
20dab4
+
20dab4
+    _onIdleTimeoutChanged() {
20dab4
+        if (this._idleWatchId) {
20dab4
+            this._idleMonitor.remove_watch(this._idleWatchId);
20dab4
+            this._idleWatchId = 0;
20dab4
+        }
20dab4
+        this._messageInhibitedUntilIdle = false;
20dab4
+    }
20dab4
+
20dab4
+    _updateMessage() {
20dab4
+        if (this._messageInhibitedUntilIdle) {
20dab4
+            if (this._message)
20dab4
+                this._dismissMessage();
20dab4
+            return;
20dab4
+        }
20dab4
+
20dab4
+        if (this._idleWatchId) {
20dab4
+            this._idleMonitor.remove_watch(this._idleWatchId);
20dab4
+            this._idleWatchId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (Main.sessionMode.hasOverview && Main.overview.visible) {
20dab4
+            this._dismissMessage();
20dab4
+            return;
20dab4
+        }
20dab4
+
20dab4
+        if (!Main.layoutManager.panelBox.visible) {
20dab4
+            this._dismissMessage();
20dab4
+            return;
20dab4
+        }
20dab4
+
20dab4
+        let supportedModes = [];
20dab4
+
20dab4
+        if (this._settings.get_boolean('show-when-unlocked'))
20dab4
+            supportedModes.push('user');
20dab4
+
20dab4
+        if (this._settings.get_boolean('show-when-unlocking'))
20dab4
+            supportedModes.push('unlock-dialog');
20dab4
+
20dab4
+        if (this._settings.get_boolean('show-when-locked'))
20dab4
+            supportedModes.push('lock-screen');
20dab4
+
20dab4
+        if (this._settings.get_boolean('show-on-login-screen'))
20dab4
+            supportedModes.push('gdm');
20dab4
+
20dab4
+        if (!supportedModes.includes(Main.sessionMode.currentMode) &&
20dab4
+            !supportedModes.includes(Main.sessionMode.parentMode)) {
20dab4
+            this._dismissMessage();
20dab4
+            return;
20dab4
+        }
20dab4
+
20dab4
+        let heading = this._settings.get_string('message-heading');
20dab4
+        let body = this._settings.get_string('message-body');
20dab4
+
20dab4
+        if (!heading && !body) {
20dab4
+            this._dismissMessage();
20dab4
+            return;
20dab4
+        }
20dab4
+
20dab4
+        if (!this._message) {
20dab4
+            this._message = new HeadsUpMessage.HeadsUpMessage(heading, body);
20dab4
+
20dab4
+            this._message.connect('notify::allocation', this._adaptSessionForMessage.bind(this));
20dab4
+            this._message.connect('clicked', this._onMessageClicked.bind(this));
20dab4
+        }
20dab4
+
20dab4
+        this._message.reactive = true;
20dab4
+        this._message.track_hover = true;
20dab4
+
20dab4
+        this._message.setHeading(heading);
20dab4
+        this._message.setBody(body);
20dab4
+
20dab4
+        if (!Main.sessionMode.hasWindows) {
20dab4
+            this._message.track_hover = false;
20dab4
+            this._message.reactive = false;
20dab4
+        }
20dab4
+    }
20dab4
+
20dab4
+    _onMessageClicked() {
20dab4
+        if (!Main.sessionMode.hasWindows)
20dab4
+          return;
20dab4
+
20dab4
+        if (this._idleWatchId) {
20dab4
+            this._idleMonitor.remove_watch(this._idleWatchId);
20dab4
+            this._idleWatchId = 0;
20dab4
+        }
20dab4
+
20dab4
+        let idleTimeout = this._settings.get_uint('idle-timeout');
20dab4
+        this._idleWatchId = this._idleMonitor.add_idle_watch(idleTimeout * 1000, this._onUserIdle.bind(this));
20dab4
+        this._messageInhibitedUntilIdle = true;
20dab4
+        this._updateMessage();
20dab4
+    }
20dab4
+
20dab4
+    _onUserIdle() {
20dab4
+        this._messageInhibitedUntilIdle = false;
20dab4
+        this._updateMessage();
20dab4
+    }
20dab4
+
20dab4
+    _dismissMessage() {
20dab4
+        if (!this._message) {
20dab4
+            return;
20dab4
+        }
20dab4
+
20dab4
+        this._message.visible = false;
20dab4
+        this._message.destroy();
20dab4
+        this._message = null;
20dab4
+        this._resetMessageTray();
20dab4
+        this._resetLoginDialog();
20dab4
+    }
20dab4
+
20dab4
+    _resetMessageTray() {
20dab4
+        if (!Main.messageTray)
20dab4
+            return;
20dab4
+
20dab4
+        Main.messageTray.actor.set_translation(0, 0, 0);
20dab4
+    }
20dab4
+
20dab4
+    _alignMessageTray() {
20dab4
+        if (!Main.messageTray)
20dab4
+            return;
20dab4
+
20dab4
+        if (!this._message || !this._message.visible) {
20dab4
+            this._resetMessageTray()
20dab4
+            return;
20dab4
+        }
20dab4
+
20dab4
+        let panelBottom = Main.layoutManager.panelBox.y + Main.layoutManager.panelBox.height;
20dab4
+        let messageBottom = this._message.y + this._message.height;
20dab4
+
20dab4
+        Main.messageTray.actor.set_translation(0, messageBottom - panelBottom, 0);
20dab4
+    }
20dab4
+
20dab4
+    _resetLoginDialog() {
20dab4
+        if (!Main.sessionMode.isGreeter)
20dab4
+            return;
20dab4
+
20dab4
+        if (!Main.screenShield || !Main.screenShield._dialog)
20dab4
+            return;
20dab4
+
20dab4
+        let dialog = Main.screenShield._dialog;
20dab4
+
20dab4
+        if (this._authPromptAllocatedId) {
20dab4
+            dialog.disconnect(this._authPromptAllocatedId);
20dab4
+            this._authPromptAllocatedId = 0;
20dab4
+        }
20dab4
+
20dab4
+        dialog.style = null;
20dab4
+        dialog._bannerView.style = null;
20dab4
+    }
20dab4
+
20dab4
+    _adaptLoginDialogForMessage() {
20dab4
+        if (!Main.sessionMode.isGreeter)
20dab4
+            return;
20dab4
+
20dab4
+        if (!Main.screenShield || !Main.screenShield._dialog)
20dab4
+            return;
20dab4
+
20dab4
+        if (!this._message || !this._message.visible) {
20dab4
+            this._resetLoginDialog()
20dab4
+            return;
20dab4
+        }
20dab4
+
20dab4
+        let dialog = Main.screenShield._dialog;
20dab4
+
20dab4
+        let messageHeight = this._message.y + this._message.height;
20dab4
+        if (dialog._logoBin.visible)
20dab4
+            messageHeight -= dialog._logoBin.height;
20dab4
+
20dab4
+        if (messageHeight <= 0) {
20dab4
+            dialog.style = null;
20dab4
+            dialog._bannerView.style = null;
20dab4
+        } else {
20dab4
+            dialog.style = `margin-top: ${messageHeight}px;`;
20dab4
+
20dab4
+            let bannerOnSide = dialog._bannerView.x + dialog._bannerView.width < dialog._authPrompt.actor.x;
20dab4
+
20dab4
+            if (bannerOnSide)
20dab4
+               dialog._bannerView.style = `margin-bottom: ${messageHeight}px;`;
20dab4
+            else
20dab4
+               dialog._bannerView.style = `margin-top: ${messageHeight}px`;
20dab4
+        }
20dab4
+    }
20dab4
+
20dab4
+    _adaptSessionForMessage() {
20dab4
+        this._alignMessageTray();
20dab4
+
20dab4
+        if (Main.sessionMode.isGreeter) {
20dab4
+            this._adaptLoginDialogForMessage();
20dab4
+            if (!this._authPromptAllocatedId) {
20dab4
+                let dialog = Main.screenShield._dialog;
20dab4
+                this._authPromptAllocatedId = dialog._authPrompt.actor.connect("notify::allocation", this._adaptLoginDialogForMessage.bind(this));
20dab4
+            }
20dab4
+        }
20dab4
+    }
20dab4
+}
20dab4
+
20dab4
+function init() {
20dab4
+    return new Extension();
20dab4
+}
20dab4
diff --git a/extensions/heads-up-display/headsUpMessage.js b/extensions/heads-up-display/headsUpMessage.js
20dab4
new file mode 100644
001c36
index 00000000..d828d8c9
20dab4
--- /dev/null
20dab4
+++ b/extensions/heads-up-display/headsUpMessage.js
20dab4
@@ -0,0 +1,150 @@
20dab4
+const { Atk, Clutter, GObject, Pango, St } = imports.gi;
20dab4
+const Layout = imports.ui.layout;
20dab4
+const Main = imports.ui.main;
20dab4
+const Signals = imports.signals;
20dab4
+
20dab4
+var HeadsUpMessageBodyLabel = GObject.registerClass({
20dab4
+}, class HeadsUpMessageBodyLabel extends St.Label {
20dab4
+    _init(params) {
20dab4
+        super._init(params);
20dab4
+
20dab4
+        this.clutter_text.single_line_mode = false;
20dab4
+        this.clutter_text.line_wrap = true;
20dab4
+    }
20dab4
+
20dab4
+    vfunc_get_preferred_width(forHeight) {
20dab4
+        let workArea = Main.layoutManager.getWorkAreaForMonitor(Main.layoutManager.primaryIndex);
20dab4
+
20dab4
+        let [labelMinimumWidth, labelNaturalWidth] = super.vfunc_get_preferred_width(forHeight);
20dab4
+
20dab4
+        labelMinimumWidth = Math.min(labelMinimumWidth, .75 * workArea.width);
20dab4
+        labelNaturalWidth = Math.min(labelNaturalWidth, .75 * workArea.width);
20dab4
+
20dab4
+        return [labelMinimumWidth, labelNaturalWidth];
20dab4
+    }
20dab4
+
20dab4
+    vfunc_get_preferred_height(forWidth) {
20dab4
+        let workArea = Main.layoutManager.getWorkAreaForMonitor(Main.layoutManager.primaryIndex);
20dab4
+        let labelHeightUpperBound = .25 * workArea.height;
20dab4
+
20dab4
+        this.clutter_text.single_line_mode = true;
20dab4
+        this.clutter_text.line_wrap = false;
20dab4
+        let [lineHeight] = super.vfunc_get_preferred_height(-1);
20dab4
+        let numberOfLines = Math.floor(labelHeightUpperBound / lineHeight);
20dab4
+        numberOfLines = Math.max(numberOfLines, 1);
20dab4
+
20dab4
+        let labelHeight = lineHeight * numberOfLines;
20dab4
+
20dab4
+        this.clutter_text.single_line_mode = false;
20dab4
+        this.clutter_text.line_wrap = true;
20dab4
+        let [labelMinimumHeight, labelNaturalHeight] = super.vfunc_get_preferred_height(forWidth);
20dab4
+
20dab4
+        labelMinimumHeight = Math.min(labelMinimumHeight, labelHeight);
20dab4
+        labelNaturalHeight = Math.min(labelNaturalHeight, labelHeight);
20dab4
+
20dab4
+        return [labelMinimumHeight, labelNaturalHeight];
20dab4
+    }
20dab4
+
20dab4
+    vfunc_allocate(box, flags) {
20dab4
+        if (!this.visible)
20dab4
+            return;
20dab4
+
20dab4
+        super.vfunc_allocate(box, flags);
20dab4
+    }
20dab4
+});
20dab4
+
20dab4
+var HeadsUpMessage = GObject.registerClass({
20dab4
+}, class HeadsUpMessage extends St.Button {
20dab4
+    _init(heading, body) {
20dab4
+        super._init({
20dab4
+            style_class: 'message',
20dab4
+            accessible_role: Atk.Role.NOTIFICATION,
20dab4
+            can_focus: false,
20dab4
+        });
20dab4
+
20dab4
+        Main.layoutManager.addChrome(this, { affectsInputRegion: true });
20dab4
+
20dab4
+        this.add_style_class_name('heads-up-display-message');
20dab4
+
20dab4
+        this._panelAllocationId = Main.layoutManager.panelBox.connect ("notify::allocation", this._alignWithPanel.bind(this));
20dab4
+        this.connect("notify::allocation", this._alignWithPanel.bind(this));
20dab4
+
20dab4
+        this._messageTraySnappingId = Main.messageTray.connect ("notify::y", () => {
20dab4
+            if (!this.visible)
20dab4
+                return;
20dab4
+
20dab4
+            if (!Main.messageTray.visible)
20dab4
+                return;
20dab4
+
20dab4
+            if (Main.messageTray.y >= this.y && Main.messageTray.y < this.y + this.height)
20dab4
+                Main.messageTray.y = this.y + this.height;
20dab4
+        });
20dab4
+
20dab4
+
20dab4
+        let contentsBox = new St.BoxLayout({ style_class: 'heads-up-message-content',
20dab4
+                                             vertical: true,
20dab4
+                                             x_align: Clutter.ActorAlign.CENTER });
20dab4
+        this.add_actor(contentsBox);
20dab4
+
20dab4
+        this.headingLabel = new St.Label({ style_class: 'heads-up-message-heading',
20dab4
+                                           x_expand: true,
20dab4
+                                           x_align: Clutter.ActorAlign.CENTER });
20dab4
+        this.setHeading(heading);
20dab4
+        contentsBox.add_actor(this.headingLabel);
20dab4
+        this.contentsBox = contentsBox;
20dab4
+
20dab4
+        this.bodyLabel = new HeadsUpMessageBodyLabel({ style_class: 'heads-up-message-body',
20dab4
+                                                       x_expand: true,
20dab4
+                                                       y_expand: true });
20dab4
+        contentsBox.add_actor(this.bodyLabel);
20dab4
+
20dab4
+        this.setBody(body);
20dab4
+        this.bodyLabel.clutter_text.label = this.bodyLabel;
20dab4
+    }
20dab4
+
20dab4
+    _alignWithPanel() {
20dab4
+        if (!this.visible)
20dab4
+            return;
20dab4
+
20dab4
+        this.x = Main.panel.actor.x;
20dab4
+        this.x += Main.panel.actor.width / 2;
20dab4
+        this.x -= this.width / 2;
20dab4
+        this.x = Math.floor(this.x);
20dab4
+        this.y = Main.panel.actor.y + Main.panel.actor.height;
20dab4
+        this.queue_relayout();
20dab4
+    }
20dab4
+
20dab4
+    setHeading(text) {
20dab4
+        if (text) {
20dab4
+            let heading = text ? text.replace(/\n/g, ' ') : '';
20dab4
+            this.headingLabel.text = heading;
20dab4
+            this.headingLabel.visible = true;
20dab4
+        } else {
20dab4
+            this.headingLabel.text = text;
20dab4
+            this.headingLabel.visible = false;
20dab4
+        }
20dab4
+    }
20dab4
+
20dab4
+    setBody(text) {
20dab4
+        this.bodyLabel.text = text;
20dab4
+        if (text) {
20dab4
+            this.bodyLabel.visible = true;
20dab4
+        } else {
20dab4
+            this.bodyLabel.visible = false;
20dab4
+        }
20dab4
+    }
20dab4
+
20dab4
+    destroy() {
20dab4
+        if (this._panelAllocationId) {
20dab4
+            Main.layoutManager.panelBox.disconnect(this._panelAllocationId);
20dab4
+            this._panelAllocationId = 0;
20dab4
+        }
20dab4
+
20dab4
+        if (this._messageTraySnappingId) {
20dab4
+            Main.messageTray.disconnect(this._messageTraySnappingId);
20dab4
+            this._messageTraySnappingId = 0;
20dab4
+        }
20dab4
+
20dab4
+        super.destroy();
20dab4
+    }
20dab4
+});
20dab4
diff --git a/extensions/heads-up-display/meson.build b/extensions/heads-up-display/meson.build
20dab4
new file mode 100644
001c36
index 00000000..40c3de0a
20dab4
--- /dev/null
20dab4
+++ b/extensions/heads-up-display/meson.build
20dab4
@@ -0,0 +1,8 @@
20dab4
+extension_data += configure_file(
20dab4
+  input: metadata_name + '.in',
20dab4
+  output: metadata_name,
20dab4
+  configuration: metadata_conf
20dab4
+)
20dab4
+
20dab4
+extension_sources += files('headsUpMessage.js', 'prefs.js')
20dab4
+extension_schemas += files(metadata_conf.get('gschemaname') + '.gschema.xml')
20dab4
diff --git a/extensions/heads-up-display/metadata.json.in b/extensions/heads-up-display/metadata.json.in
20dab4
new file mode 100644
001c36
index 00000000..e7ab71aa
20dab4
--- /dev/null
20dab4
+++ b/extensions/heads-up-display/metadata.json.in
20dab4
@@ -0,0 +1,11 @@
20dab4
+{
20dab4
+"extension-id": "@extension_id@",
20dab4
+"uuid": "@uuid@",
20dab4
+"gettext-domain": "@gettext_domain@",
20dab4
+"name": "Heads-up Display Message",
20dab4
+"description": "Add a message to be displayed on screen always above all windows and chrome.",
20dab4
+"original-authors": [ "rstrode@redhat.com" ],
20dab4
+"shell-version": [ "@shell_current@" ],
20dab4
+"url": "@url@",
20dab4
+"session-modes":  [ "gdm", "lock-screen", "unlock-dialog", "user" ]
20dab4
+}
20dab4
diff --git a/extensions/heads-up-display/org.gnome.shell.extensions.heads-up-display.gschema.xml b/extensions/heads-up-display/org.gnome.shell.extensions.heads-up-display.gschema.xml
20dab4
new file mode 100644
001c36
index 00000000..ea1f3774
20dab4
--- /dev/null
20dab4
+++ b/extensions/heads-up-display/org.gnome.shell.extensions.heads-up-display.gschema.xml
20dab4
@@ -0,0 +1,54 @@
20dab4
+<schemalist gettext-domain="gnome-shell-extensions">
20dab4
+  
20dab4
+          path="/org/gnome/shell/extensions/heads-up-display/">
20dab4
+    <key name="idle-timeout" type="u">
20dab4
+      <default>30</default>
20dab4
+      <summary>Idle Timeout</summary>
20dab4
+      <description>
20dab4
+        Number of seconds until message is reshown after user goes idle.
20dab4
+      </description>
20dab4
+    </key>
20dab4
+    <key name="message-heading" type="s">
20dab4
+      <default>""</default>
20dab4
+      <summary>Message to show at top of display</summary>
20dab4
+      <description>
20dab4
+        The top line of the heads up display message.
20dab4
+      </description>
20dab4
+    </key>
20dab4
+    <key name="message-body" type="s">
20dab4
+      <default>""</default>
20dab4
+      <summary>Banner message</summary>
20dab4
+      <description>
20dab4
+        A message to always show at the top of the screen.
20dab4
+      </description>
20dab4
+    </key>
20dab4
+    <key name="show-on-login-screen" type="b">
20dab4
+      <default>true</default>
20dab4
+      <summary>Show on login screen</summary>
20dab4
+      <description>
20dab4
+        Whether or not the message should display on the login screen
20dab4
+      </description>
20dab4
+    </key>
20dab4
+    <key name="show-when-locked" type="b">
20dab4
+      <default>false</default>
20dab4
+      <summary>Show on screen shield</summary>
20dab4
+      <description>
20dab4
+        Whether or not the message should display when the screen is locked
20dab4
+      </description>
20dab4
+    </key>
20dab4
+    <key name="show-when-unlocking" type="b">
20dab4
+      <default>false</default>
20dab4
+      <summary>Show on unlock screen</summary>
20dab4
+      <description>
20dab4
+        Whether or not the message should display on the unlock screen.
20dab4
+      </description>
20dab4
+    </key>
20dab4
+    <key name="show-when-unlocked" type="b">
20dab4
+      <default>false</default>
20dab4
+      <summary>Show in user session</summary>
20dab4
+      <description>
20dab4
+        Whether or not the message should display when the screen is unlocked.
20dab4
+      </description>
20dab4
+    </key>
20dab4
+  </schema>
20dab4
+</schemalist>
20dab4
diff --git a/extensions/heads-up-display/prefs.js b/extensions/heads-up-display/prefs.js
20dab4
new file mode 100644
001c36
index 00000000..b4b6f94c
20dab4
--- /dev/null
20dab4
+++ b/extensions/heads-up-display/prefs.js
20dab4
@@ -0,0 +1,175 @@
20dab4
+
20dab4
+/* Desktop Icons GNOME Shell extension
20dab4
+ *
20dab4
+ * Copyright (C) 2017 Carlos Soriano <csoriano@redhat.com>
20dab4
+ *
20dab4
+ * This program is free software: you can redistribute it and/or modify
20dab4
+ * it under the terms of the GNU General Public License as published by
20dab4
+ * the Free Software Foundation, either version 3 of the License, or
20dab4
+ * (at your option) any later version.
20dab4
+ *
20dab4
+ * This program is distributed in the hope that it will be useful,
20dab4
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
20dab4
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20dab4
+ * GNU General Public License for more details.
20dab4
+ *
20dab4
+ * You should have received a copy of the GNU General Public License
20dab4
+ * along with this program.  If not, see <http://www.gnu.org/licenses/>.
20dab4
+ */
20dab4
+
20dab4
+const { Gio, GObject, Gdk, Gtk } = imports.gi;
20dab4
+const ExtensionUtils = imports.misc.extensionUtils;
20dab4
+const Gettext = imports.gettext.domain('gnome-shell-extensions');
20dab4
+const _ = Gettext.gettext;
20dab4
+const N_ = e => e;
20dab4
+const cssData = `
20dab4
+   .no-border {
20dab4
+       border: none;
20dab4
+   }
20dab4
+
20dab4
+   .border {
20dab4
+       border: 1px solid;
20dab4
+       border-radius: 3px;
20dab4
+       border-color: #b6b6b3;
20dab4
+       box-shadow: inset 0 0 0 1px rgba(74, 144, 217, 0);
20dab4
+       background-color: white;
20dab4
+   }
20dab4
+
20dab4
+   .margins {
20dab4
+       padding-left: 8px;
20dab4
+       padding-right: 8px;
20dab4
+       padding-bottom: 8px;
20dab4
+   }
20dab4
+
20dab4
+   .message-label {
20dab4
+       font-weight: bold;
20dab4
+   }
20dab4
+`;
20dab4
+
20dab4
+var settings;
20dab4
+
20dab4
+function init() {
20dab4
+    settings = ExtensionUtils.getSettings("org.gnome.shell.extensions.heads-up-display");
20dab4
+    let cssProvider = new Gtk.CssProvider();
20dab4
+    cssProvider.load_from_data(cssData);
20dab4
+
20dab4
+    let screen = Gdk.Screen.get_default();
20dab4
+    Gtk.StyleContext.add_provider_for_screen(screen, cssProvider, Gtk.STYLE_PROVIDER_PRIORITY_APPLICATION);
20dab4
+}
20dab4
+
20dab4
+function buildPrefsWidget() {
20dab4
+    ExtensionUtils.initTranslations();
20dab4
+
20dab4
+    let contents = new Gtk.Box({
20dab4
+        orientation: Gtk.Orientation.VERTICAL,
20dab4
+        border_width: 20,
20dab4
+        spacing: 10,
20dab4
+    });
20dab4
+
20dab4
+    contents.add(buildSwitch('show-when-locked', _("Show message when screen is locked")));
20dab4
+    contents.add(buildSwitch('show-when-unlocking', _("Show message on unlock screen")));
20dab4
+    contents.add(buildSwitch('show-when-unlocked', _("Show message when screen is unlocked")));
20dab4
+    contents.add(buildSpinButton('idle-timeout', _("Seconds after user goes idle before reshowing message")));
20dab4
+
20dab4
+    let outerMessageBox = new Gtk.Box({
20dab4
+        orientation: Gtk.Orientation.VERTICAL,
20dab4
+        border_width: 0,
20dab4
+        spacing: 5,
20dab4
+    });
20dab4
+    contents.add(outerMessageBox);
20dab4
+
20dab4
+    let messageLabel = new Gtk.Label({
20dab4
+        label: 'Message',
20dab4
+        halign: Gtk.Align.START,
20dab4
+    });
20dab4
+    messageLabel.get_style_context().add_class("message-label");
20dab4
+    outerMessageBox.add(messageLabel);
20dab4
+
20dab4
+    let innerMessageBox = new Gtk.Box({
20dab4
+        orientation: Gtk.Orientation.VERTICAL,
20dab4
+        border_width: 0,
20dab4
+        spacing: 0,
20dab4
+    });
20dab4
+    innerMessageBox.get_style_context().add_class("border");
20dab4
+    outerMessageBox.add(innerMessageBox);
20dab4
+
20dab4
+    innerMessageBox.add(buildEntry('message-heading', _("Message Heading")));
20dab4
+    innerMessageBox.add(buildTextView('message-body', _("Message Body")));
20dab4
+    contents.show_all();
20dab4
+    return contents;
20dab4
+}
20dab4
+
20dab4
+function buildTextView(key, labelText) {
20dab4
+    let textView = new Gtk.TextView({
20dab4
+        accepts_tab: false,
20dab4
+        wrap_mode: Gtk.WrapMode.WORD,
20dab4
+    });
20dab4
+    settings.bind(key, textView.get_buffer(), 'text', Gio.SettingsBindFlags.DEFAULT);
20dab4
+
20dab4
+    let scrolledWindow = new Gtk.ScrolledWindow({
20dab4
+        expand: true,
20dab4
+    });
20dab4
+    let styleContext = scrolledWindow.get_style_context();
20dab4
+    styleContext.add_class("margins");
20dab4
+
20dab4
+    scrolledWindow.add(textView);
20dab4
+    return scrolledWindow;
20dab4
+}
20dab4
+function buildEntry(key, labelText) {
20dab4
+    let entry = new Gtk.Entry({ placeholder_text: labelText });
20dab4
+    let styleContext = entry.get_style_context();
20dab4
+    styleContext.add_class("no-border");
20dab4
+    settings.bind(key, entry, 'text', Gio.SettingsBindFlags.DEFAULT);
20dab4
+
20dab4
+    entry.get_settings()['gtk-entry-select-on-focus'] = false;
20dab4
+
20dab4
+    return entry;
20dab4
+}
20dab4
+
20dab4
+function buildSpinButton(key, labelText) {
20dab4
+    let hbox = new Gtk.Box({
20dab4
+        orientation: Gtk.Orientation.HORIZONTAL,
20dab4
+        spacing: 10,
20dab4
+    });
20dab4
+    let label = new Gtk.Label({
20dab4
+        label: labelText,
20dab4
+        xalign: 0,
20dab4
+    });
20dab4
+    let adjustment = new Gtk.Adjustment({
20dab4
+        value: 0,
20dab4
+        lower: 0,
20dab4
+        upper: 2147483647,
20dab4
+        step_increment: 1,
20dab4
+        page_increment: 60,
20dab4
+        page_size: 60,
20dab4
+    });
20dab4
+    let spinButton = new Gtk.SpinButton({
20dab4
+        adjustment: adjustment,
20dab4
+        climb_rate: 1.0,
20dab4
+        digits: 0,
20dab4
+        max_width_chars: 3,
20dab4
+        width_chars: 3,
20dab4
+    });
20dab4
+    settings.bind(key, spinButton, 'value', Gio.SettingsBindFlags.DEFAULT);
20dab4
+    hbox.pack_start(label, true, true, 0);
20dab4
+    hbox.add(spinButton);
20dab4
+    return hbox;
20dab4
+}
20dab4
+
20dab4
+function buildSwitch(key, labelText) {
20dab4
+    let hbox = new Gtk.Box({
20dab4
+        orientation: Gtk.Orientation.HORIZONTAL,
20dab4
+        spacing: 10,
20dab4
+    });
20dab4
+    let label = new Gtk.Label({
20dab4
+        label: labelText,
20dab4
+        xalign: 0,
20dab4
+    });
20dab4
+    let switcher = new Gtk.Switch({
20dab4
+        active: settings.get_boolean(key),
20dab4
+    });
20dab4
+    settings.bind(key, switcher, 'active', Gio.SettingsBindFlags.DEFAULT);
20dab4
+    hbox.pack_start(label, true, true, 0);
20dab4
+    hbox.add(switcher);
20dab4
+    return hbox;
20dab4
+}
20dab4
diff --git a/extensions/heads-up-display/stylesheet.css b/extensions/heads-up-display/stylesheet.css
20dab4
new file mode 100644
001c36
index 00000000..93034469
20dab4
--- /dev/null
20dab4
+++ b/extensions/heads-up-display/stylesheet.css
20dab4
@@ -0,0 +1,32 @@
20dab4
+.heads-up-display-message {
20dab4
+    background-color: rgba(0.24, 0.24, 0.24, 0.80);
20dab4
+    border: 1px solid black;
20dab4
+    border-radius: 6px;
20dab4
+    color: #eeeeec;
20dab4
+    font-size: 11pt;
20dab4
+    margin-top: 0.5em;
20dab4
+    margin-bottom: 0.5em;
20dab4
+    padding: 0.9em;
20dab4
+}
20dab4
+
20dab4
+.heads-up-display-message:insensitive {
20dab4
+    background-color: rgba(0.24, 0.24, 0.24, 0.33);
20dab4
+}
20dab4
+
20dab4
+.heads-up-display-message:hover {
20dab4
+    background-color: rgba(0.24, 0.24, 0.24, 0.2);
20dab4
+    border: 1px solid rgba(0.0, 0.0, 0.0, 0.5);
20dab4
+    color: #4d4d4d;
20dab4
+    transition-duration: 250ms;
20dab4
+}
20dab4
+
20dab4
+.heads-up-message-heading {
20dab4
+    height: 1.75em;
20dab4
+    font-size: 1.25em;
20dab4
+    font-weight: bold;
20dab4
+    text-align: center;
20dab4
+}
20dab4
+
20dab4
+.heads-up-message-body {
20dab4
+    text-align: center;
20dab4
+}
20dab4
diff --git a/meson.build b/meson.build
001c36
index ba84f8f3..c5fc86ef 100644
20dab4
--- a/meson.build
20dab4
+++ b/meson.build
001c36
@@ -44,6 +44,7 @@ classic_extensions = [
20dab4
 default_extensions = classic_extensions
20dab4
 default_extensions += [
20dab4
   'drive-menu',
20dab4
+  'heads-up-display',
20dab4
   'screenshot-window-sizer',
20dab4
   'windowsNavigator',
20dab4
   'workspace-indicator'
20dab4
-- 
001c36
2.32.0
20dab4