diff --git a/README.md b/README.md index 18d69ae..984de12 100644 --- a/README.md +++ b/README.md @@ -1,16 +1,21 @@ # VirtualBox applet - + + +This applet is a fork from https://github.com/eugene-rom/vbox-applet repository. + Gnome Shell VirtualBox applet provide menu to run VirtualBox machines and switch between running VMs. VM list populated after few seconds after extension is activated, to not slow down screen unlock. (Gnome Shell deactivates all extensions on screen lock and activates on unlock). Applet checks if machines is running and mark items with dot shortly after popup becomes visible. +Applet is able to run a VM in headless mode by toggling the headless option. +Applet is able to run a VM in detachable mode by toggling the detachable option. ![screenshot](screenshot.png?raw=true) diff --git a/screenshot.png b/screenshot.png index 9398c9d..9c56336 100644 Binary files a/screenshot.png and b/screenshot.png differ diff --git a/src/extension.js b/src/extension.js index 23dc246..4a4b5be 100644 --- a/src/extension.js +++ b/src/extension.js @@ -10,11 +10,13 @@ const Me = ExtensionUtils.getCurrentExtension(); const ByteArray = imports.byteArray; const TEXT_VBOXAPP = 'VirtualBox applet'; -const TEXT_LOGID = 'vbox-applet'; -const ICON_SIZE = 22; -const DEBUG = false; +const TEXT_LOGID = 'vbox-applet'; +const ICON_SIZE = 22; +const DEBUG = false; const SETTING_SORT = 'sort'; +const SETTING_HEADLESS = 'headless'; +const SETTING_DETACHABLE = 'detachable'; let settings; let vboxapplet; @@ -24,215 +26,244 @@ let sourceId2 = null; let sourceId3 = null; let VBoxApplet = GObject.registerClass( -class VBoxApplet extends PanelMenu.Button -{ - _init() - { - super._init( 0.0, TEXT_VBOXAPP ); - - this._populated = false; - this._menuitems = []; - let gicon = Gio.icon_new_for_string( Me.path + '/icons/vbox-symbolic.svg' ); - let icon = new St.Icon( { gicon: gicon, icon_size: ICON_SIZE } ); - this.add_child( icon ); - - this._tmpItem = new PopupMenu.PopupMenuItem( '...' ); - this.menu.addMenuItem( this._tmpItem ); - - this._startVbox = () => { - GLib.spawn_command_line_async( 'virtualbox' ); - }; - - this._startVM = ( name, id ) => { - if ( this._isVMRunning( id ) ) { - this._activateWindow( name ); - } - else { - GLib.spawn_command_line_async( 'vboxmanage startvm ' + id ); - } - }; - - this._toggleSort = ( menuitemSort ) => { - let sort = settings.get_boolean( SETTING_SORT ); - menuitemSort.setToggleState( !sort ); - settings.set_boolean( SETTING_SORT, !sort ); - sourceId1 = GLib.timeout_add( GLib.PRIORITY_DEFAULT, 10, this._populateMenu.bind(this) ); - }; - - this._parseVMList = ( vms ) => { - let res = []; - if ( vms.length !== 0 ) - { - let machines = vms.toString().split('\n'); - for ( let i = 0; i < machines.length; i++ ) - { - let machine = machines[i]; - if ( machine === '' ) { - continue; + class VBoxApplet extends PanelMenu.Button { + _init() { + super._init(0.0, TEXT_VBOXAPP); + + this._populated = false; + this._menuitems = []; + let gicon = Gio.icon_new_for_string(Me.path + '/icons/vbox-symbolic.svg'); + let icon = new St.Icon({ gicon: gicon, icon_size: ICON_SIZE }); + this.add_child(icon); + + this._tmpItem = new PopupMenu.PopupMenuItem('...'); + this.menu.addMenuItem(this._tmpItem); + + this._startVbox = () => { + GLib.spawn_command_line_async('virtualbox'); + }; + + this._startVM = (name, id) => { + if (this._isVMRunning(id)) { + this._activateWindow(name); + } + else { + let headless = settings.get_boolean(SETTING_HEADLESS); + let detachable = settings.get_boolean(SETTING_DETACHABLE); + + GLib.spawn_command_line_async('vboxmanage startvm ' + id + (headless ? ' --type headless' : detachable ? ' --type separate' : '')); + } + }; + + this._toggleSort = (menuitemSort) => { + let sort = settings.get_boolean(SETTING_SORT); + menuitemSort.setToggleState(!sort); + settings.set_boolean(SETTING_SORT, !sort); + sourceId1 = GLib.timeout_add(GLib.PRIORITY_DEFAULT, 10, this._populateMenu.bind(this)); + }; + + this._toggleHeadless = (menuitemHeadless, menuitemDetachable) => { + let headless = settings.get_boolean(SETTING_HEADLESS); + menuitemHeadless.setToggleState(!headless); + settings.set_boolean(SETTING_HEADLESS, !headless); + if(!headless){ + menuitemDetachable.setToggleState(true); + settings.set_boolean(SETTING_DETACHABLE, true); + } + }; + + this._toggleDetachable = (menuitemDetachable, menuitemHeadless) => { + let detachable = settings.get_boolean(SETTING_DETACHABLE); + menuitemDetachable.setToggleState(!detachable); + settings.set_boolean(SETTING_DETACHABLE, !detachable); + if(detachable){ + menuitemHeadless.setToggleState(false); + settings.set_boolean(SETTING_HEADLESS, false); + } + }; + + this._parseVMList = (vms) => { + let res = []; + if (vms.length !== 0) { + let machines = vms.toString().split('\n'); + for (let i = 0; i < machines.length; i++) { + let machine = machines[i]; + if (machine === '') { + continue; + } + + let info = machine.split('" {'); + let name = info[0].replace('"', ''); + let id = info[1].replace('}', ''); + + this._log('Machine name: ' + name + ', ID: ' + id); + + res.push({ name: name, id: id }); } + } + return res; + }; - let info = machine.split('" {'); - let name = info[0].replace('"', ''); - let id = info[1].replace('}', ''); + this._populateMenu = () => { + this._menuitems = []; + this.menu.removeAll(); - this._log( 'Machine name: ' + name + ', ID: ' + id ); + let vms; + let sort = settings.get_boolean(SETTING_SORT); + let headless = settings.get_boolean(SETTING_HEADLESS); + let detachable = settings.get_boolean(SETTING_DETACHABLE); - res.push( { name:name, id:id } ); + if(headless != detachable){ + settings.set_boolean(SETTING_DETACHABLE, headless); } - } - return res; - }; - this._populateMenu = () => { - this._menuitems = []; - this.menu.removeAll(); - - let vms; - let sort = settings.get_boolean( SETTING_SORT ); - try - { - let cmd = 'vboxmanage list ' + ( sort ? '-s' : '' ) + ' vms'; - this._log( 'Run \'' + cmd + '\'' ); - vms = ByteArray.toString( GLib.spawn_command_line_sync( cmd )[1] ); - } - catch (err) { - this._log( err ); - Main.notifyError( TEXT_VBOXAPP + ': ' + err ); - return; - } - - let machines = this._parseVMList( vms ); - - if ( machines.length !== 0 ) - { - for ( let i = 0; i < machines.length; i++ ) - { - let name = machines[i].name; - let id = machines[i].id; - - let menuitem = new PopupMenu.PopupMenuItem( name ); - menuitem._vmid = id; - menuitem.connect( 'activate', this._startVM.bind(this, name, id) ); - this.menu.addMenuItem(menuitem); - this._menuitems.push(menuitem); - } - } - - this.menu.addMenuItem( new PopupMenu.PopupSeparatorMenuItem() ); - - let menuitemRefresh = new PopupMenu.PopupMenuItem( 'Refresh' ); - menuitemRefresh.connect( 'activate', this._populateMenu.bind(this) ); - this.menu.addMenuItem( menuitemRefresh ); - - let menuitemSort = new PopupMenu.PopupSwitchMenuItem( 'Sort', sort ); - menuitemSort.connect( 'toggled', this._toggleSort.bind(this, menuitemSort) ); - this.menu.addMenuItem( menuitemSort ); - - let menuitemStartVBox = new PopupMenu.PopupMenuItem( 'VirtualBox...' ); - menuitemStartVBox.connect( 'activate', this._startVbox.bind(this) ); - this.menu.addMenuItem( menuitemStartVBox ); - - this._populated = true; - - this._onVisibilityChanged(); - - return false; - }; - - this._log = ( text ) => { - if ( DEBUG ) { - global.log( TEXT_LOGID, text ); - } - }; - - this._isVMRunning = ( id ) => { - let machines = this._getRunningVMs(); - return this._searchInVMs( machines, id ); - }; - - this._getRunningVMs = () => { - let vms; - try { - this._log( 'Run \'vboxmanage list runningvms\'' ); - vms = ByteArray.toString( GLib.spawn_command_line_sync( 'vboxmanage list runningvms' )[1] ); - } - catch (err) { - this._log( err ); - return; - } - - return this._parseVMList( vms ); - }; - - this._onVisibilityChanged = () => { - if ( this.menu.actor.visible && this._populated ) { - sourceId2 = GLib.timeout_add( GLib.PRIORITY_DEFAULT, 10, this._markRunning.bind(this) ); - } - }; - - this._markRunning = () => { - let machines = this._getRunningVMs(); - - for (var i = 0; i < this._menuitems.length; i++) { - let running = this._searchInVMs( machines, this._menuitems[i]._vmid ); - this._menuitems[i].setOrnament( running ? PopupMenu.Ornament.DOT : PopupMenu.Ornament.NONE ); - } - return false; - }; - - this._searchInVMs = ( machines, id ) => { - for ( var i = 0; i < machines.length; i++ ) { - if ( machines[i].id === id ) { - return true; - } - } - return false; - }; - - this._activateWindow = ( name ) => { - let a = global.get_window_actors(); - for (var i = 0; i < a.length; i++) - { - let mw = a[i].metaWindow; - let title = mw.get_title(); - - if ( title.startsWith( name ) && title.toLowerCase().includes('virtualbox') ) { - this._log( 'activate window: ' + title ); - mw.activate( global.get_current_time() ); - } - } - }; - - this.menu.actor.connect( 'notify::visible', this._onVisibilityChanged.bind(this) ); - sourceId3 = GLib.timeout_add_seconds( GLib.PRIORITY_DEFAULT, 3, this._populateMenu.bind(this) ); - } -} ); + try { + let cmd = 'vboxmanage list ' + (sort ? '-s' : '') + ' vms'; + this._log('Run \'' + cmd + '\''); + vms = ByteArray.toString(GLib.spawn_command_line_sync(cmd)[1]); + } + catch (err) { + this._log(err); + Main.notifyError(TEXT_VBOXAPP + ': ' + err); + return; + } + + let machines = this._parseVMList(vms); + + if (machines.length !== 0) { + for (let i = 0; i < machines.length; i++) { + let name = machines[i].name; + let id = machines[i].id; + + let menuitem = new PopupMenu.PopupMenuItem(name); + menuitem._vmid = id; + menuitem.connect('activate', this._startVM.bind(this, name, id)); + this.menu.addMenuItem(menuitem); + this._menuitems.push(menuitem); + } + } + + this.menu.addMenuItem(new PopupMenu.PopupSeparatorMenuItem()); + + let menuitemRefresh = new PopupMenu.PopupMenuItem('Refresh'); + menuitemRefresh.connect('activate', this._populateMenu.bind(this)); + this.menu.addMenuItem(menuitemRefresh); + + let menuitemSort = new PopupMenu.PopupSwitchMenuItem('Sort', sort); + menuitemSort.connect('toggled', this._toggleSort.bind(this, menuitemSort)); + this.menu.addMenuItem(menuitemSort); + + let menuitemDetachable = new PopupMenu.PopupSwitchMenuItem('Detachable', detachable); + let menuitemHeadless = new PopupMenu.PopupSwitchMenuItem('Headless', headless); + menuitemDetachable.connect('toggled', this._toggleDetachable.bind(this, menuitemDetachable, menuitemHeadless)); + this.menu.addMenuItem(menuitemDetachable); + + menuitemHeadless.connect('toggled', this._toggleHeadless.bind(this, menuitemHeadless, menuitemDetachable)); + this.menu.addMenuItem(menuitemHeadless); + + let menuitemStartVBox = new PopupMenu.PopupMenuItem('VirtualBox...'); + menuitemStartVBox.connect('activate', this._startVbox.bind(this)); + this.menu.addMenuItem(menuitemStartVBox); + + this._populated = true; + + this._onVisibilityChanged(); + + return false; + }; + + this._log = (text) => { + if (DEBUG) { + global.log(TEXT_LOGID, text); + } + }; + + this._isVMRunning = (id) => { + let machines = this._getRunningVMs(); + return this._searchInVMs(machines, id); + }; + + this._getRunningVMs = () => { + let vms; + try { + this._log('Run \'vboxmanage list runningvms\''); + vms = ByteArray.toString(GLib.spawn_command_line_sync('vboxmanage list runningvms')[1]); + } + catch (err) { + this._log(err); + return; + } + + return this._parseVMList(vms); + }; + + this._onVisibilityChanged = () => { + if (this.menu.actor.visible && this._populated) { + sourceId2 = GLib.timeout_add(GLib.PRIORITY_DEFAULT, 10, this._markRunning.bind(this)); + } + }; + + this._markRunning = () => { + let machines = this._getRunningVMs(); + + for (var i = 0; i < this._menuitems.length; i++) { + let running = this._searchInVMs(machines, this._menuitems[i]._vmid); + this._menuitems[i].setOrnament(running ? PopupMenu.Ornament.DOT : PopupMenu.Ornament.NONE); + } + return false; + }; + + this._searchInVMs = (machines, id) => { + for (var i = 0; i < machines.length; i++) { + if (machines[i].id === id) { + return true; + } + } + return false; + }; + + this._activateWindow = (name) => { + let a = global.get_window_actors(); + for (var i = 0; i < a.length; i++) { + let mw = a[i].metaWindow; + let title = mw.get_title(); + + if (title.startsWith(name) && title.toLowerCase().includes('virtualbox')) { + this._log('activate window: ' + title); + mw.activate(global.get_current_time()); + } + } + }; + + this.menu.actor.connect('notify::visible', this._onVisibilityChanged.bind(this)); + sourceId3 = GLib.timeout_add_seconds(GLib.PRIORITY_DEFAULT, 3, this._populateMenu.bind(this)); + } + }); function enable() { enabled = true; settings = ExtensionUtils.getSettings(); vboxapplet = new VBoxApplet; - Main.panel.addToStatusArea( TEXT_VBOXAPP, vboxapplet ); + Main.panel.addToStatusArea(TEXT_VBOXAPP, vboxapplet); } -function disable() -{ +function disable() { enabled = false; - + // GS guidelines requirements - if ( sourceId1 ) { - GLib.Source.remove( sourceId1 ); + if (sourceId1) { + GLib.Source.remove(sourceId1); sourceId1 = null; } - if ( sourceId2 ) { - GLib.Source.remove( sourceId2 ); + if (sourceId2) { + GLib.Source.remove(sourceId2); sourceId2 = null; } - if ( sourceId3 ) { - GLib.Source.remove( sourceId3 ); + if (sourceId3) { + GLib.Source.remove(sourceId3); sourceId3 = null; } - + vboxapplet.destroy(); settings = null; vboxapplet = null; diff --git a/src/metadata.json b/src/metadata.json index b71cd0b..34b94f2 100644 --- a/src/metadata.json +++ b/src/metadata.json @@ -1,9 +1,9 @@ { "name": "VirtualBox applet", - "version": 12, - "description": "Provide menu to run VirtualBox machines and switch between running VMs", + "version": 15, + "description": "Provide menu to run VirtualBox machines and switch between running VMs.\nIt provides options for detachable head and headless mode.\nThis is a fork of https://extensions.gnome.org/extension/1415/virtualbox-applet/ extension.", "settings-schema": "org.gnome.shell.extensions.vbox-applet", - "url": "https://github.com/eugene-rom/vbox-applet", - "uuid": "vbox-applet@gs.eros2.info", - "shell-version": ["3.36","3.38","40","41","42","43"] + "url": "https://github.com/Buba98/vbox-applet", + "uuid": "vbox-applet@buba98", + "shell-version": ["3.36","3.38","40","41","42","43", "44"] } diff --git a/src/schemas/org.gnome.shell.extensions.vbox-applet.gschema.xml b/src/schemas/org.gnome.shell.extensions.vbox-applet.gschema.xml index 6955c28..a1fc0b8 100644 --- a/src/schemas/org.gnome.shell.extensions.vbox-applet.gschema.xml +++ b/src/schemas/org.gnome.shell.extensions.vbox-applet.gschema.xml @@ -1,10 +1,21 @@ - + false Sort VMs list by name Sort VMs list by name + + false + Start an detachable vm + Start an detachable vm + + + false + Start an headless vm + Start an headless vm + - + \ No newline at end of file