gnome:extensions:vpn_status
Differences
This shows you the differences between two versions of the page.
Next revision | Previous revision | ||
gnome:extensions:vpn_status [2022/09/23 16:14] – created peter | gnome:extensions:vpn_status [2022/09/23 16:23] (current) – peter | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== Gnome - Extensions - VPN Status ====== | ====== Gnome - Extensions - VPN Status ====== | ||
+ | |||
+ | Displays if the system is connected to a VPN or not. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Create the directory for the extension ===== | ||
+ | |||
+ | <code bash> | ||
+ | mkdir ~/ | ||
+ | </ | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Create a metadata.json file ===== | ||
+ | |||
+ | **metadata.json** is a mandatory file of the extension, containing information about the extension such as its UUID, name and description. | ||
+ | |||
+ | <file json metadata.json> | ||
+ | { | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | ], | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Create a extensions.js file ===== | ||
+ | |||
+ | **extensions.js** is the core file of the extension and contains the function hooks **init()**, **enable()** and **disable()** used by GNOME Shell to load, enable and disable the extension. | ||
+ | |||
+ | <file javascript extensions.js> | ||
+ | /* | ||
+ | VPN Status. | ||
+ | |||
+ | Displays whether a VPN is running or not in the top bar. | ||
+ | |||
+ | Author: Peter Roux | ||
+ | */ | ||
+ | |||
+ | |||
+ | /* The Main instance our UI elements. */ | ||
+ | const Main = imports.ui.main; | ||
+ | |||
+ | /* The library that allows you to create UI elements. */ | ||
+ | const St = imports.gi.St; | ||
+ | |||
+ | /* This supports Canvas and UI elements. */ | ||
+ | const Clutter = imports.gi.Clutter; | ||
+ | |||
+ | /* For the animations of the UI elements. | ||
+ | const Tweener = imports.ui.tweener; | ||
+ | const Tweener = imports.tweener.tweener; | ||
+ | */ | ||
+ | |||
+ | |||
+ | const Util = imports.misc.util; | ||
+ | const PanelMenu = imports.ui.panelMenu; | ||
+ | const Mainloop = imports.mainloop; | ||
+ | const GLib = imports.gi.GLib; | ||
+ | |||
+ | |||
+ | /* Initialise global variables to use as button to click and text labels. */ | ||
+ | let text, button, vpn_label; | ||
+ | |||
+ | |||
+ | /* | ||
+ | Hide the info. | ||
+ | |||
+ | This function is called when the label is opacity 0%. | ||
+ | |||
+ | As the label remains an UI element, but not visible, it has to be deleted explicitily. | ||
+ | When the label reaches 0% opacity, it is removed from the Main instance. | ||
+ | */ | ||
+ | function _hideInfo() { | ||
+ | Main.uiGroup.remove_actor(text); | ||
+ | text.destroy(); | ||
+ | text = null; | ||
+ | } | ||
+ | |||
+ | |||
+ | /* | ||
+ | Show some info. | ||
+ | |||
+ | This function is called when the main button is clicked. | ||
+ | */ | ||
+ | function _showInfo() { | ||
+ | /* If the text is not already present, create a new UI element using the St library. | ||
+ | | ||
+ | if (!text) { | ||
+ | text = new St.Label({ style: ' | ||
+ | Main.uiGroup.add_actor(text); | ||
+ | } | ||
+ | |||
+ | /* Set text to be fully visible at first. */ | ||
+ | text.opacity = 255; | ||
+ | |||
+ | /* Select the monitor to display the info label on. | ||
+ | Here, the primary monitor is used. */ | ||
+ | let monitor = Main.layoutManager.primaryMonitor; | ||
+ | |||
+ | /* Change the position of the text to the center of the monitor. */ | ||
+ | text.set_position(Math.floor(monitor.width / 2 - text.width / 2), | ||
+ | Math.floor(monitor.height / 2 - text.height / 2)); | ||
+ | |||
+ | /* Using tweener for the animations. | ||
+ | Set the opacity at 0% after 5 seconds, with the type of transition easeOutQuad. | ||
+ | When this animation is completed, execute our function _hideInfo. | ||
+ | | ||
+ | /* | ||
+ | Tweener.addTween(text, | ||
+ | time: 5, | ||
+ | | ||
+ | | ||
+ | */ | ||
+ | |||
+ | // Tweener` is a deprecated module....so using this... | ||
+ | text.ease({ | ||
+ | //x: newX, | ||
+ | //y: 10, | ||
+ | opacity: 0, | ||
+ | duration: 2000, | ||
+ | mode: Clutter.AnimationMode.EASE_OUT_BOUNCE, | ||
+ | onComplete: _hideInfo | ||
+ | }); | ||
+ | } | ||
+ | |||
+ | |||
+ | /* | ||
+ | Initialize the extension. | ||
+ | */ | ||
+ | function init() { | ||
+ | } | ||
+ | |||
+ | |||
+ | |||
+ | /* | ||
+ | This does the actual work of determining and displaying the VPN status. | ||
+ | */ | ||
+ | function _refresh() { | ||
+ | let [res, out, err, exit] = GLib.spawn_sync(null, | ||
+ | |||
+ | try { | ||
+ | if (vpn_label) { | ||
+ | if (exit == 256) | ||
+ | vpn_label.set_text(" | ||
+ | else if (exit == 0) | ||
+ | { | ||
+ | //let [res2, out2, err2, exit2] = GLib.spawn_sync(null, | ||
+ | let [res2, out2, err2, exit2] = GLib.spawn_sync(null, | ||
+ | if (exit2 == 0) | ||
+ | vpn_label.set_text(" | ||
+ | else | ||
+ | vpn_label.set_text(" | ||
+ | } | ||
+ | else | ||
+ | vpn_label.set_text(" | ||
+ | } | ||
+ | } | ||
+ | catch (e) { | ||
+ | // logError(e, ' | ||
+ | } | ||
+ | |||
+ | if (this._timeout) { | ||
+ | Mainloop.source_remove(this._timeout); | ||
+ | this._timeout = null; | ||
+ | } | ||
+ | |||
+ | // Calls the refresh function every 2 seconds. | ||
+ | this._timeout = Mainloop.timeout_add(2, | ||
+ | } | ||
+ | |||
+ | |||
+ | /* Create any objects, connecting signals etc. */ | ||
+ | function enable() { | ||
+ | /* | ||
+ | Create a button for the top panel. | ||
+ | We pass to the constructor a map of properties, properties from St.bin and its parent classes, stWidget. | ||
+ | A style class (from the css theming of gnome shell). | ||
+ | We make it reactive to mouse clicks. | ||
+ | We mark the button as also being able to receive keyboard focus via keyboard navigation. | ||
+ | The button will expand the x space, but we do not want to expand the y space. | ||
+ | We want the button to be reactive on the hover of a mouse, so we set the value of the track_hover property to true. | ||
+ | */ | ||
+ | button = new St.Bin({ style_class: | ||
+ | reactive: true, | ||
+ | can_focus: true, | ||
+ | x_expand: true, | ||
+ | y_expand: false, | ||
+ | track_hover: | ||
+ | |||
+ | /* We create an icon with the " | ||
+ | let vpn_label = new St.Icon({ icon_name: ' | ||
+ | style_class: | ||
+ | */ | ||
+ | |||
+ | /* Create a vpn label to display the VPN status. */ | ||
+ | vpn_label = new St.Label({ | ||
+ | text: _(" | ||
+ | y_align: Clutter.ActorAlign.CENTER | ||
+ | }); | ||
+ | |||
+ | /* | ||
+ | Set the vpn label as a child of the button. | ||
+ | In the structure of actors we have the vpn label inside the button that is a container. | ||
+ | */ | ||
+ | button.set_child(vpn_label); | ||
+ | |||
+ | /* | ||
+ | Connect the actor signal " | ||
+ | When the button is pressed, this signal is emitted, we capture it and execute the _showInfo function. | ||
+ | */ | ||
+ | button.connect(' | ||
+ | |||
+ | |||
+ | /* Add the button that we created to the right side panel of the top panel (where the sound and wifi settings are). */ | ||
+ | Main.panel._rightBox.insert_child_at_index(button, | ||
+ | |||
+ | /* Call the refresh function. */ | ||
+ | this._refresh(); | ||
+ | } | ||
+ | |||
+ | |||
+ | /* Revert any changes, disconnect signals and destroy objects. */ | ||
+ | function disable() { | ||
+ | /* Destroy the timeout object. */ | ||
+ | if (this._timeout) { | ||
+ | Mainloop.source_remove(this._timeout); | ||
+ | this._timeout = null; | ||
+ | } | ||
+ | |||
+ | |||
+ | /* Destroy the vpn object. */ | ||
+ | if (vpn_label !== null) { | ||
+ | vpn_label.destroy(); | ||
+ | vpn_label = null; | ||
+ | } | ||
+ | |||
+ | /* Destroy the text object. */ | ||
+ | if (text !== null) { | ||
+ | text.destroy(); | ||
+ | text = null; | ||
+ | } | ||
+ | |||
+ | /* Remove the button from the right panel. */ | ||
+ | Main.panel._rightBox.remove_child(button); | ||
+ | |||
+ | /* Destroy the button object. */ | ||
+ | if (button !== null) { | ||
+ | button.destroy(); | ||
+ | button = null; | ||
+ | } | ||
+ | |||
+ | } | ||
+ | </ | ||
+ | |||
+ | <WRAP info> | ||
+ | **NOTE: | ||
+ | </ | ||
+ | |||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Reload GNOME to pick up the extension ===== | ||
+ | |||
+ | Press **Alt+F2**, and enter **r** and press **ENTER** to restart GNOME Shell. | ||
+ | |||
+ | <WRAP info> | ||
+ | **NOTE: | ||
+ | |||
+ | <code bash> | ||
+ | sudo journalctl / | ||
+ | </ | ||
+ | |||
+ | </ | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== Activate the extension ===== | ||
+ | |||
+ | Open the **Gnome Tweak Tool** and activate the extension. | ||
+ | |||
+ | ---- | ||
+ | |||
+ | ===== References ===== | ||
+ | |||
+ | https:// | ||
+ | |||
+ | http:// | ||
+ | |||
+ | https:// | ||
gnome/extensions/vpn_status.1663949678.txt.gz · Last modified: 2022/09/23 16:14 by peter