/* -*- mode: js2; js2-basic-offset: 4; indent-tabs-mode: nil -*- */ const Caribou = imports.gi.Caribou; const Clutter = imports.gi.Clutter; const DBus = imports.dbus; const Gdk = imports.gi.Gdk; const Gio = imports.gi.Gio; const GLib = imports.gi.GLib; const Lang = imports.lang; const Meta = imports.gi.Meta; const Shell = imports.gi.Shell; const St = imports.gi.St; const BoxPointer = imports.ui.boxpointer; const Main = imports.ui.main; const MessageTray = imports.ui.messageTray; const PopupMenu = imports.ui.popupMenu; const Tweener = imports.ui.tweener; const KEYBOARD_SCHEMA = 'org.gnome.shell.keyboard'; const SHOW_KEYBOARD = 'show-keyboard'; const KEYBOARD_TYPE = 'keyboard-type'; const ENABLE_DRAGGABLE = 'enable-drag'; const ENABLE_FLOAT = 'enable-float'; // Key constants taken from Antler const PRETTY_KEYS = { 'BackSpace': '\u232b', 'space': ' ', 'Return': '\u23ce', 'Caribou_Prefs': '\u2328', 'Caribou_ShiftUp': '\u2b06', 'Caribou_ShiftDown': '\u2b07', 'Caribou_Emoticons': '\u263a', 'Caribou_Symbols': '123', 'Caribou_Symbols_More': '{#*', 'Caribou_Alpha': 'Abc', 'Tab': 'Tab', 'Escape': 'Esc', 'Control_L': 'Ctrl', 'Alt_L': 'Alt' }; const CaribouKeyboardIface = { name: 'org.gnome.Caribou.Keyboard', methods: [ { name: 'Show', inSignature: '', outSignature: '' }, { name: 'Hide', inSignature: '', outSignature: '' }, { name: 'SetCursorLocation', inSignature: 'iiii', outSignature: '' }, { name: 'SetEntryLocation', inSignature: 'iiii', outSignature: '' } ], properties: [ { name: 'Name', signature: 's', access: 'read' } ] }; function Key() { this._init.apply(this, arguments); } Key.prototype = { _init : function(key, key_width, key_height) { this._key = key; this._width = key_width; this._height = key_height; this.actor = this._getKey(); this._extended_keys = this._key.get_extended_keys(); this._extended_keyboard = {}; if (this._key.name == "Control_L" || this._key.name == "Alt_L") this._key.latch = true; this._key.connect('key-pressed', Lang.bind(this, function () { this.actor.checked = true })); this._key.connect('key-released', Lang.bind(this, function () { this.actor.checked = false; })); if (this._extended_keys.length > 0) { this._grabbed = false; this._eventCaptureId = 0; this._key.connect('notify::show-subkeys', Lang.bind(this, this._onShowSubkeysChanged)); this._boxPointer = new BoxPointer.BoxPointer(St.Side.BOTTOM, { x_fill: true, y_fill: true, x_align: St.Align.START }); // Adds style to existing keyboard style to avoid repetition this._boxPointer.actor.add_style_class_name('keyboard-subkeys'); this._getExtendedKeys(); this.actor._extended_keys = this._extended_keyboard; this._boxPointer.actor.hide(); Main.chrome.addActor(this._boxPointer.actor, { visibleInFullscreen: true, affectsStruts: false }); } }, _getKey: function () { let label = this._key.name; if (label.length > 1) { let pretty = PRETTY_KEYS[label]; if (pretty) label = pretty; else label = this._getUnichar(this._key); } label = GLib.markup_escape_text(label, -1); let button = new St.Button ({ label: label, style_class: 'keyboard-key' }); button.width = this._width; button.key_width = this._key.width; button.height = this._height; button.draggable = false; button.connect('button-press-event', Lang.bind(this, function () { this._key.press(); })); button.connect('button-release-event', Lang.bind(this, function () { this._key.release(); })); return button; }, _getUnichar: function(key) { let keyval = key.keyval; let unichar = Gdk.keyval_to_unicode(keyval); if (unichar) { return String.fromCharCode(unichar); } else { return key.name; } }, _getExtendedKeys: function () { this._extended_keyboard = new St.BoxLayout({ style_class: 'keyboard-layout', vertical: false }); for (let i = 0; i < this._extended_keys.length; ++i) { let extended_key = this._extended_keys[i]; let label = this._getUnichar(extended_key); let key = new St.Button({ label: label, style_class: 'keyboard-key' }); key.extended_key = extended_key; key.width = this._width; key.height = this._height; key.draggable = false; key.connect('button-press-event', Lang.bind(this, function () { extended_key.press(); })); key.connect('button-release-event', Lang.bind(this, function () { extended_key.release(); })); this._extended_keyboard.add(key); } this._boxPointer.bin.add_actor(this._extended_keyboard); }, _onEventCapture: function (actor, event) { let source = event.get_source(); if (event.type() == Clutter.EventType.BUTTON_PRESS || (event.type() == Clutter.EventType.BUTTON_RELEASE && source.draggable)) { if (this._extended_keyboard.contains(source)) { if (source.draggable) { source.extended_key.press(); source.extended_key.release(); } this._ungrab(); return false; } this._boxPointer.actor.hide(); this._ungrab(); return true; } return false; }, _ungrab: function () { global.stage.disconnect(this._eventCaptureId); this._eventCaptureId = 0; this._grabbed = false; Main.popModal(this.actor); }, _onShowSubkeysChanged: function () { if (this._key.show_subkeys) { this.actor.fake_release(); this._boxPointer.actor.raise_top(); this._boxPointer.setPosition(this.actor, 5, 0.5); this._boxPointer.show(true); this.actor.set_hover(false); if (!this._grabbed) { Main.pushModal(this.actor); this._eventCaptureId = global.stage.connect('captured-event', Lang.bind(this, this._onEventCapture)); this._grabbed = true; } this._key.release(); } else { if (this._grabbed) this._ungrab(); this._boxPointer.hide(true); } } }; function Keyboard() { this._init.apply(this, arguments); } Keyboard.prototype = { _init: function () { DBus.session.exportObject('/org/gnome/Caribou/Keyboard', this); DBus.session.acquire_name('org.gnome.Caribou.Keyboard', 0, null, null); this.actor = new St.BoxLayout({ name: 'keyboard', vertical: true, reactive: true }); this._keyboardSettings = new Gio.Settings({ schema: KEYBOARD_SCHEMA }); this._keyboardSettings.connect('changed', Lang.bind(this, this._display)); this._setupKeyboard(); Main.layoutManager.connect('monitors-changed', Lang.bind(this, this._redraw)); Main.layoutManager.bottomBox.add_actor(this.actor); }, init: function () { this._display(); }, _setupKeyboard: function() { if (this._keyboardNotifyId) this._keyboard.disconnect(this._keyboardNotifyId); let children = this.actor.get_children(); for (let i = 0; i < children.length; i++) children[i].destroy(); this._keyboard = new Caribou.KeyboardModel({ keyboard_type: this._keyboardSettings.get_string(KEYBOARD_TYPE) }); this._groups = {}; this._current_page = null; // Initialize keyboard key measurements this._numOfHorizKeys = 0; this._numOfVertKeys = 0; this._floatId = 0; this._addKeys(); this._keyboardNotifyId = this._keyboard.connect('notify::active-group', Lang.bind(this, this._onGroupChanged)); }, _display: function () { if (this._keyboard.keyboard_type != this._keyboardSettings.get_string(KEYBOARD_TYPE)) this._setupKeyboard(); this._showKeyboard = this._keyboardSettings.get_boolean(SHOW_KEYBOARD); this._draggable = this._keyboardSettings.get_boolean(ENABLE_DRAGGABLE); this._floating = this._keyboardSettings.get_boolean(ENABLE_FLOAT); if (this._floating) { this._floatId = this.actor.connect('button-press-event', Lang.bind(this, this._startDragging)); this._dragging = false; } else this.actor.disconnect(this._floatId); if (this._showKeyboard) this.show(); else { this.hide(); this.destroySource(); } }, _startDragging: function (actor, event) { if (this._dragging) // don't allow two drags at the same time return; this._dragging = true; this._preDragStageMode = global.stage_input_mode; Clutter.grab_pointer(this.actor); global.set_stage_input_mode(Shell.StageInputMode.FULLSCREEN); this._releaseId = this.actor.connect('button-release-event', Lang.bind(this, this._endDragging)); this._motionId = this.actor.connect('motion-event', Lang.bind(this, this._motionEvent)); [this._dragStartX, this._dragStartY] = event.get_coords(); [this._currentX, this._currentY] = this.actor.get_position(); }, _endDragging: function () { if (this._dragging) { this.actor.disconnect(this._releaseId); this.actor.disconnect(this._motionId); Clutter.ungrab_pointer(); global.set_stage_input_mode(this._preDragStageMode); global.unset_cursor(); this._dragging = false; } return true; }, _motionEvent: function(actor, event) { let absX, absY; [absX, absY] = event.get_coords(); global.set_cursor(Shell.Cursor.DND_IN_DRAG); this._moveHandle(absX, absY); return true; }, _moveHandle: function (stageX, stageY) { let x, y; x = stageX - this._dragStartX + this._currentX; y = stageY - this._dragStartY + this._currentY; this.actor.set_position(x,y); }, _addKeys: function () { let groups = this._keyboard.get_groups(); for (let i = 0; i < groups.length; ++i) { let gname = groups[i]; let group = this._keyboard.get_group(gname); group.connect('notify::active-level', Lang.bind(this, this._onLevelChanged)); let layers = {}; let levels = group.get_levels(); for (let j = 0; j < levels.length; ++j) { let lname = levels[j]; let level = group.get_level(lname); let layout = new St.BoxLayout({ style_class: 'keyboard-layout', vertical: true }); this._loadRows(level, layout); layers[lname] = layout; this.actor.add(layout, { x_fill: false }); layout.hide(); } this._groups[gname] = layers; } this._setActiveLayer(); }, _getTrayIcon: function () { let trayButton = new St.Button ({ label: "tray", style_class: 'keyboard-key' }); trayButton.key_width = 1; trayButton.connect('button-press-event', Lang.bind(this, function () { Main.layoutManager.updateForTray(); })); Main.overview.connect('showing', Lang.bind(this, function () { trayButton.reactive = false; trayButton.add_style_pseudo_class('grayed'); })); Main.overview.connect('hiding', Lang.bind(this, function () { trayButton.reactive = true; trayButton.remove_style_pseudo_class('grayed'); })); return trayButton; }, _addRows : function (keys, layout) { let keyboard_row = new St.BoxLayout(); for (let i = 0; i < keys.length; ++i) { let children = keys[i].get_children(); let right_box = new St.BoxLayout({ style_class: 'keyboard-row' }); let left_box = new St.BoxLayout({ style_class: 'keyboard-row' }); for (let j = 0; j < children.length; ++j) { if (this._numOfHorizKeys == 0) this._numOfHorizKeys = children.length; let key = children[j]; let button = new Key(key, 0, 0); if (key.align == 'right') right_box.add(button.actor); else left_box.add(button.actor); if (key.name == "Caribou_Prefs") { key.connect('key-released', Lang.bind(this, this._onPrefsClick)); // Add new key for hiding message tray right_box.add(this._getTrayIcon()); } } keyboard_row.add(left_box, { expand: true, x_fill: false, x_align: St.Align.START }); keyboard_row.add(right_box, { expand: true, x_fill: false, x_align: St.Align.END }); } layout.add(keyboard_row); }, _manageTray: function () { this.createSource(); }, _onPrefsClick: function () { this.hide(); this._manageTray(); }, _loadRows : function (level, layout) { let rows = level.get_rows(); for (let i = 0; i < rows.length; ++i) { let row = rows[i]; if (this._numOfVertKeys == 0) this._numOfVertKeys = rows.length; this._addRows(row.get_columns(), layout); } }, _redraw: function () { let monitor = Main.layoutManager.bottomMonitor; let maxHeight = monitor.height / 3; this.actor.width = monitor.width; let layout = this._current_page; let verticalSpacing = layout.get_theme_node().get_length('spacing'); let padding = layout.get_theme_node().get_length('padding'); let box = layout.get_children()[0].get_children()[0]; let horizontalSpacing = box.get_theme_node().get_length('spacing'); let allHorizontalSpacing = (this._numOfHorizKeys - 1) * horizontalSpacing; let keyWidth = Math.floor((this.actor.width - allHorizontalSpacing - 2 * padding) / this._numOfHorizKeys); let allVerticalSpacing = (this._numOfVertKeys - 1) * verticalSpacing; let keyHeight = Math.floor((maxHeight - allVerticalSpacing - 2 * padding) / this._numOfVertKeys); let keySize = Math.min(keyWidth, keyHeight); this.actor.height = keySize * this._numOfVertKeys + allVerticalSpacing + 2 * padding; let rows = this._current_page.get_children(); for (let i = 0; i < rows.length; ++i) { let keyboard_row = rows[i]; let boxes = keyboard_row.get_children(); for (let j = 0; j < boxes.length; ++j) { let keys = boxes[j].get_children(); for (let k = 0; k < keys.length; ++k) { let child = keys[k]; child.width = keySize * child.key_width; child.height = keySize; child.draggable = this._draggable; if (child._extended_keys) { let extended_keys = child._extended_keys.get_children(); for (let n = 0; n < extended_keys.length; ++n) { let extended_key = extended_keys[n]; extended_key.width = keySize; extended_key.height = keySize; extended_key.draggable = this._draggable; } } } } } }, _onLevelChanged: function () { this._setActiveLayer(); this._redraw(); }, _onGroupChanged: function () { this._setActiveLayer(); this._redraw(); }, _setActiveLayer: function () { let active_group_name = this._keyboard.active_group; let active_group = this._keyboard.get_group(active_group_name); let active_level = active_group.active_level; let layers = this._groups[active_group_name]; if (this._current_page != null) { this._current_page.hide(); } this._current_page = layers[active_level]; this._current_page.show(); }, createSource: function () { if (this._source == null) { this._source = new KeyboardSource(this); Main.messageTray.add(this._source); } }, destroySource: function () { if (this._source) { this._source.destroy(); this._source = null; } }, show: function () { this._redraw(); Main.layoutManager.showKeyboard(); }, hide: function () { Main.layoutManager.hideKeyboard(); }, // Window placement method _updatePosition: function (x, y) { let primary = Main.layoutManager.primaryMonitor; x -= this.actor.width / 2; // Determines bottom/top centered if (y <= primary.height / 2) y += this.actor.height / 2; else y -= 3 * this.actor.height / 2; // Accounting for monitor boundaries if (x < primary.x) x = primary.x; if (x + this.actor.width > primary.width) x = primary.width - this.actor.width; this.actor.set_position(x, y); }, _moveTemporarily: function () { this._currentWindow = global.screen.get_display().focus_window; let rect = this._currentWindow.get_outer_rect(); this._currentWindow.x = rect.x; this._currentWindow.y = rect.y; let newX = this._currentWindow.x; let newY = 3 * this.actor.height / 2; this._currentWindow.move_frame(true, newX, newY); }, _setLocation: function (x, y) { if (this._floating) this._updatePosition(x, y); else { if (y >= 2 * this.actor.height) this._moveTemporarily(); } }, // D-Bus methods Show: function() { this.destroySource(); this.show(); }, Hide: function() { if (this._currentWindow) { this._currentWindow.move_frame(true, this._currentWindow.x, this._currentWindow.y); this._currentWindow = null; } this.hide(); this._manageTray(); }, SetCursorLocation: function(x, y, w, h) { this._setLocation(x, y); }, SetEntryLocation: function(x, y, w, h) { this._setLocation(x, y); }, get Name() { return 'gnome-shell'; } }; DBus.conformExport(Keyboard.prototype, CaribouKeyboardIface); function KeyboardSource() { this._init.apply(this, arguments); } KeyboardSource.prototype = { __proto__: MessageTray.Source.prototype, _init: function(keyboard) { this._keyboard = keyboard; MessageTray.Source.prototype._init.call(this, _("Keyboard")); this._setSummaryIcon(this.createNotificationIcon()); }, createNotificationIcon: function() { return new St.Icon({ icon_name: 'input-keyboard', icon_type: St.IconType.SYMBOLIC, icon_size: this.ICON_SIZE }); }, handleSummaryClick: function() { let event = Clutter.get_current_event(); if (event.type() != Clutter.EventType.BUTTON_RELEASE) return false; if (event.get_button() != 1) return false; this.open(); return true; }, open: function() { this._keyboard.show(); this._keyboard.destroySource(); } };