2011-09-28 09:16:26 -04:00
|
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2019-08-28 23:52:52 +02:00
|
|
|
|
/* exported KeyboardManager */
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-02-09 04:21:36 +01:00
|
|
|
|
const { Clutter, Gio, GLib, GObject, Meta, St } = imports.gi;
|
2020-08-13 23:35:27 +02:00
|
|
|
|
const ByteArray = imports.byteArray;
|
2012-11-10 22:47:57 +01:00
|
|
|
|
const Signals = imports.signals;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-02-09 04:21:36 +01:00
|
|
|
|
const InputSourceManager = imports.ui.status.keyboard;
|
2018-05-25 11:35:49 +02:00
|
|
|
|
const IBusManager = imports.misc.ibusManager;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
const BoxPointer = imports.ui.boxpointer;
|
2012-11-18 16:17:21 +01:00
|
|
|
|
const Layout = imports.ui.layout;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
const Main = imports.ui.main;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
const PageIndicators = imports.ui.pageIndicators;
|
2018-01-23 16:45:46 +01:00
|
|
|
|
const PopupMenu = imports.ui.popupMenu;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-08-02 01:13:10 +02:00
|
|
|
|
var KEYBOARD_REST_TIME = Layout.KEYBOARD_ANIMATION_TIME * 2;
|
2017-08-05 12:21:30 +02:00
|
|
|
|
var KEY_LONG_PRESS_TIME = 250;
|
2019-08-02 01:13:10 +02:00
|
|
|
|
var PANEL_SWITCH_ANIMATION_TIME = 500;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
var PANEL_SWITCH_RELATIVE_DISTANCE = 1 / 3; /* A third of the actor width */
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2011-08-29 13:02:42 -04:00
|
|
|
|
const A11Y_APPLICATIONS_SCHEMA = 'org.gnome.desktop.a11y.applications';
|
|
|
|
|
const SHOW_KEYBOARD = 'screen-keyboard-enabled';
|
|
|
|
|
|
2017-12-20 13:08:50 +01:00
|
|
|
|
/* KeyContainer puts keys in a grid where a 1:1 key takes this size */
|
|
|
|
|
const KEY_SIZE = 2;
|
|
|
|
|
|
2017-08-05 12:21:30 +02:00
|
|
|
|
const defaultKeysPre = [
|
2020-02-14 18:11:43 +01:00
|
|
|
|
[[], [], [{ width: 1.5, level: 1, extraClassName: 'shift-key-lowercase', icon: 'keyboard-shift-filled-symbolic' }], [{ label: '?123', width: 1.5, level: 2 }]],
|
|
|
|
|
[[], [], [{ width: 1.5, level: 0, extraClassName: 'shift-key-uppercase', icon: 'keyboard-shift-filled-symbolic' }], [{ label: '?123', width: 1.5, level: 2 }]],
|
2019-01-29 02:27:05 +01:00
|
|
|
|
[[], [], [{ label: '=/<', width: 1.5, level: 3 }], [{ label: 'ABC', width: 1.5, level: 0 }]],
|
|
|
|
|
[[], [], [{ label: '?123', width: 1.5, level: 2 }], [{ label: 'ABC', width: 1.5, level: 0 }]],
|
2017-08-05 12:21:30 +02:00
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
const defaultKeysPost = [
|
2020-02-14 18:11:43 +01:00
|
|
|
|
[[{ width: 1.5, keyval: Clutter.KEY_BackSpace, icon: 'edit-clear-symbolic' }],
|
|
|
|
|
[{ width: 2, keyval: Clutter.KEY_Return, extraClassName: 'enter-key', icon: 'keyboard-enter-symbolic' }],
|
|
|
|
|
[{ width: 3, level: 1, right: true, extraClassName: 'shift-key-lowercase', icon: 'keyboard-shift-filled-symbolic' }],
|
|
|
|
|
[{ action: 'emoji', icon: 'face-smile-symbolic' }, { action: 'languageMenu', extraClassName: 'layout-key', icon: 'keyboard-layout-filled-symbolic' }, { action: 'hide', extraClassName: 'hide-key', icon: 'go-down-symbolic' }]],
|
|
|
|
|
[[{ width: 1.5, keyval: Clutter.KEY_BackSpace, icon: 'edit-clear-symbolic' }],
|
|
|
|
|
[{ width: 2, keyval: Clutter.KEY_Return, extraClassName: 'enter-key', icon: 'keyboard-enter-symbolic' }],
|
|
|
|
|
[{ width: 3, level: 0, right: true, extraClassName: 'shift-key-uppercase', icon: 'keyboard-shift-filled-symbolic' }],
|
|
|
|
|
[{ action: 'emoji', icon: 'face-smile-symbolic' }, { action: 'languageMenu', extraClassName: 'layout-key', icon: 'keyboard-layout-filled-symbolic' }, { action: 'hide', extraClassName: 'hide-key', icon: 'go-down-symbolic' }]],
|
|
|
|
|
[[{ width: 1.5, keyval: Clutter.KEY_BackSpace, icon: 'edit-clear-symbolic' }],
|
|
|
|
|
[{ width: 2, keyval: Clutter.KEY_Return, extraClassName: 'enter-key', icon: 'keyboard-enter-symbolic' }],
|
2019-01-29 20:36:54 +01:00
|
|
|
|
[{ label: '=/<', width: 3, level: 3, right: true }],
|
2020-07-17 11:56:14 +00:00
|
|
|
|
[{ action: 'emoji', icon: 'face-smile-symbolic' }, { action: 'languageMenu', extraClassName: 'layout-key', icon: 'keyboard-layout-filled-symbolic' }, { action: 'hide', extraClassName: 'hide-key', icon: 'go-down-symbolic' }]],
|
2020-02-14 18:11:43 +01:00
|
|
|
|
[[{ width: 1.5, keyval: Clutter.KEY_BackSpace, icon: 'edit-clear-symbolic' }],
|
|
|
|
|
[{ width: 2, keyval: Clutter.KEY_Return, extraClassName: 'enter-key', icon: 'keyboard-enter-symbolic' }],
|
2019-01-29 20:36:54 +01:00
|
|
|
|
[{ label: '?123', width: 3, level: 2, right: true }],
|
2020-02-14 18:11:43 +01:00
|
|
|
|
[{ action: 'emoji', icon: 'face-smile-symbolic' }, { action: 'languageMenu', extraClassName: 'layout-key', icon: 'keyboard-layout-filled-symbolic' }, { action: 'hide', extraClassName: 'hide-key', icon: 'go-down-symbolic' }]],
|
2017-08-05 12:21:30 +02:00
|
|
|
|
];
|
|
|
|
|
|
2019-01-21 21:39:27 +01:00
|
|
|
|
var AspectContainer = GObject.registerClass(
|
|
|
|
|
class AspectContainer extends St.Widget {
|
|
|
|
|
_init(params) {
|
|
|
|
|
super._init(params);
|
|
|
|
|
this._ratio = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setRatio(relWidth, relHeight) {
|
|
|
|
|
this._ratio = relWidth / relHeight;
|
|
|
|
|
this.queue_relayout();
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-04 02:36:10 +02:00
|
|
|
|
vfunc_get_preferred_width(forHeight) {
|
|
|
|
|
let [min, nat] = super.vfunc_get_preferred_width(forHeight);
|
|
|
|
|
|
|
|
|
|
if (forHeight > 0)
|
|
|
|
|
nat = forHeight * this._ratio;
|
|
|
|
|
|
|
|
|
|
return [min, nat];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vfunc_get_preferred_height(forWidth) {
|
|
|
|
|
let [min, nat] = super.vfunc_get_preferred_height(forWidth);
|
|
|
|
|
|
|
|
|
|
if (forWidth > 0)
|
|
|
|
|
nat = forWidth / this._ratio;
|
|
|
|
|
|
|
|
|
|
return [min, nat];
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-09 21:30:26 +02:00
|
|
|
|
vfunc_allocate(box) {
|
2019-01-21 21:39:27 +01:00
|
|
|
|
if (box.get_width() > 0 && box.get_height() > 0) {
|
|
|
|
|
let sizeRatio = box.get_width() / box.get_height();
|
|
|
|
|
|
|
|
|
|
if (sizeRatio >= this._ratio) {
|
|
|
|
|
/* Restrict horizontally */
|
|
|
|
|
let width = box.get_height() * this._ratio;
|
|
|
|
|
let diff = box.get_width() - width;
|
|
|
|
|
|
|
|
|
|
box.x1 += Math.floor(diff / 2);
|
|
|
|
|
box.x2 -= Math.ceil(diff / 2);
|
|
|
|
|
} else {
|
2019-03-05 19:33:49 +01:00
|
|
|
|
/* Restrict vertically, align to bottom */
|
2019-01-21 21:39:27 +01:00
|
|
|
|
let height = box.get_width() / this._ratio;
|
2019-03-05 19:33:49 +01:00
|
|
|
|
box.y1 = box.y2 - Math.floor(height);
|
2019-01-21 21:39:27 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-09 21:30:26 +02:00
|
|
|
|
super.vfunc_allocate(box);
|
2019-01-21 21:39:27 +01:00
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2019-01-31 20:38:24 +01:00
|
|
|
|
var KeyContainer = GObject.registerClass(
|
2017-10-31 02:23:39 +01:00
|
|
|
|
class KeyContainer extends St.Widget {
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_init() {
|
2017-12-20 13:08:50 +01:00
|
|
|
|
let gridLayout = new Clutter.GridLayout({ orientation: Clutter.Orientation.HORIZONTAL,
|
|
|
|
|
column_homogeneous: true,
|
|
|
|
|
row_homogeneous: true });
|
2019-07-16 11:24:13 +02:00
|
|
|
|
super._init({
|
|
|
|
|
layout_manager: gridLayout,
|
|
|
|
|
x_expand: true,
|
2019-08-20 23:43:54 +02:00
|
|
|
|
y_expand: true,
|
2019-07-16 11:24:13 +02:00
|
|
|
|
});
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._gridLayout = gridLayout;
|
|
|
|
|
this._currentRow = 0;
|
|
|
|
|
this._currentCol = 0;
|
|
|
|
|
this._maxCols = 0;
|
|
|
|
|
|
|
|
|
|
this._currentRow = null;
|
|
|
|
|
this._rows = [];
|
2017-10-31 02:23:39 +01:00
|
|
|
|
}
|
2017-12-20 13:08:50 +01:00
|
|
|
|
|
2019-02-04 12:30:53 +01:00
|
|
|
|
appendRow() {
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._currentRow++;
|
|
|
|
|
this._currentCol = 0;
|
|
|
|
|
|
2019-08-19 21:18:54 +02:00
|
|
|
|
let row = {
|
|
|
|
|
keys: [],
|
|
|
|
|
width: 0,
|
|
|
|
|
};
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._rows.push(row);
|
2017-10-31 02:23:39 +01:00
|
|
|
|
}
|
2017-12-20 13:08:50 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
appendKey(key, width = 1, height = 1) {
|
2017-12-20 13:08:50 +01:00
|
|
|
|
let keyInfo = {
|
|
|
|
|
key,
|
|
|
|
|
left: this._currentCol,
|
|
|
|
|
top: this._currentRow,
|
|
|
|
|
width,
|
2019-08-20 23:43:54 +02:00
|
|
|
|
height,
|
2017-12-20 13:08:50 +01:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let row = this._rows[this._rows.length - 1];
|
|
|
|
|
row.keys.push(keyInfo);
|
|
|
|
|
row.width += width;
|
|
|
|
|
|
|
|
|
|
this._currentCol += width;
|
|
|
|
|
this._maxCols = Math.max(this._currentCol, this._maxCols);
|
2017-10-31 02:23:39 +01:00
|
|
|
|
}
|
2017-12-20 13:08:50 +01:00
|
|
|
|
|
2019-01-21 21:39:27 +01:00
|
|
|
|
layoutButtons(container) {
|
2017-12-20 13:08:50 +01:00
|
|
|
|
let nCol = 0, nRow = 0;
|
|
|
|
|
|
|
|
|
|
for (let i = 0; i < this._rows.length; i++) {
|
|
|
|
|
let row = this._rows[i];
|
|
|
|
|
|
|
|
|
|
/* When starting a new row, see if we need some padding */
|
|
|
|
|
if (nCol == 0) {
|
|
|
|
|
let diff = this._maxCols - row.width;
|
|
|
|
|
if (diff >= 1)
|
|
|
|
|
nCol = diff * KEY_SIZE / 2;
|
|
|
|
|
else
|
|
|
|
|
nCol = diff * KEY_SIZE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (let j = 0; j < row.keys.length; j++) {
|
|
|
|
|
let keyInfo = row.keys[j];
|
|
|
|
|
let width = keyInfo.width * KEY_SIZE;
|
|
|
|
|
let height = keyInfo.height * KEY_SIZE;
|
|
|
|
|
|
|
|
|
|
this._gridLayout.attach(keyInfo.key, nCol, nRow, width, height);
|
|
|
|
|
nCol += width;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nRow += KEY_SIZE;
|
|
|
|
|
nCol = 0;
|
|
|
|
|
}
|
2019-01-21 21:39:27 +01:00
|
|
|
|
|
|
|
|
|
if (container)
|
|
|
|
|
container.setRatio(this._maxCols, this._rows.length);
|
2017-12-20 13:08:50 +01:00
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
var Suggestions = GObject.registerClass(
|
|
|
|
|
class Suggestions extends St.BoxLayout {
|
|
|
|
|
_init() {
|
2019-10-21 20:44:00 +02:00
|
|
|
|
super._init({
|
|
|
|
|
style_class: 'word-suggestions',
|
|
|
|
|
vertical: false,
|
|
|
|
|
x_align: Clutter.ActorAlign.CENTER,
|
|
|
|
|
});
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.show();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-06 13:46:02 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
add(word, callback) {
|
2017-12-06 13:46:02 +01:00
|
|
|
|
let button = new St.Button({ label: word });
|
|
|
|
|
button.connect('clicked', callback);
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.add(button);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-06 13:46:02 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
clear() {
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.remove_all_children();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2019-07-16 11:24:13 +02:00
|
|
|
|
});
|
2017-12-06 13:46:02 +01:00
|
|
|
|
|
2017-10-31 02:19:44 +01:00
|
|
|
|
var LanguageSelectionPopup = class extends PopupMenu.PopupMenu {
|
|
|
|
|
constructor(actor) {
|
|
|
|
|
super(actor, 0.5, St.Side.BOTTOM);
|
2018-01-23 16:45:46 +01:00
|
|
|
|
|
|
|
|
|
let inputSourceManager = InputSourceManager.getInputSourceManager();
|
|
|
|
|
let inputSources = inputSourceManager.inputSources;
|
|
|
|
|
|
2018-09-11 18:00:22 +02:00
|
|
|
|
let item;
|
2018-01-23 16:45:46 +01:00
|
|
|
|
for (let i in inputSources) {
|
|
|
|
|
let is = inputSources[i];
|
|
|
|
|
|
2018-09-11 18:00:22 +02:00
|
|
|
|
item = this.addAction(is.displayName, () => {
|
2018-01-23 16:45:46 +01:00
|
|
|
|
inputSourceManager.activateInputSource(is, true);
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2019-07-13 01:50:58 +02:00
|
|
|
|
item.can_focus = false;
|
2018-01-23 16:45:46 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.addMenuItem(new PopupMenu.PopupSeparatorMenuItem());
|
2018-09-11 18:00:22 +02:00
|
|
|
|
item = this.addSettingsAction(_("Region & Language Settings"), 'gnome-region-panel.desktop');
|
2019-07-13 01:50:58 +02:00
|
|
|
|
item.can_focus = false;
|
2018-09-11 18:00:22 +02:00
|
|
|
|
|
2018-01-23 16:45:46 +01:00
|
|
|
|
this._capturedEventId = 0;
|
|
|
|
|
|
2017-10-31 01:38:18 +01:00
|
|
|
|
this._unmapId = actor.connect('notify::mapped', () => {
|
2018-01-23 16:45:46 +01:00
|
|
|
|
if (!actor.is_mapped())
|
|
|
|
|
this.close(true);
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-01-23 16:45:46 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_onCapturedEvent(actor, event) {
|
2018-01-23 16:45:46 +01:00
|
|
|
|
if (event.get_source() == this.actor ||
|
|
|
|
|
this.actor.contains(event.get_source()))
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
|
|
|
|
|
|
if (event.type() == Clutter.EventType.BUTTON_RELEASE || event.type() == Clutter.EventType.TOUCH_END)
|
|
|
|
|
this.close(true);
|
|
|
|
|
|
|
|
|
|
return Clutter.EVENT_STOP;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-01-23 16:45:46 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
open(animate) {
|
2017-10-31 02:19:44 +01:00
|
|
|
|
super.open(animate);
|
2018-01-23 16:45:46 +01:00
|
|
|
|
this._capturedEventId = global.stage.connect('captured-event',
|
2017-12-02 01:27:35 +01:00
|
|
|
|
this._onCapturedEvent.bind(this));
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-01-23 16:45:46 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
close(animate) {
|
2017-10-31 02:19:44 +01:00
|
|
|
|
super.close(animate);
|
2018-01-23 16:45:46 +01:00
|
|
|
|
if (this._capturedEventId != 0) {
|
|
|
|
|
global.stage.disconnect(this._capturedEventId);
|
|
|
|
|
this._capturedEventId = 0;
|
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-01-23 16:45:46 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
destroy() {
|
2018-01-23 16:45:46 +01:00
|
|
|
|
if (this._capturedEventId != 0)
|
|
|
|
|
global.stage.disconnect(this._capturedEventId);
|
|
|
|
|
if (this._unmapId != 0)
|
|
|
|
|
this.sourceActor.disconnect(this._unmapId);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
super.destroy();
|
|
|
|
|
}
|
|
|
|
|
};
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
var Key = GObject.registerClass({
|
|
|
|
|
Signals: {
|
|
|
|
|
'activated': {},
|
|
|
|
|
'long-press': {},
|
|
|
|
|
'pressed': { param_types: [GObject.TYPE_UINT, GObject.TYPE_STRING] },
|
|
|
|
|
'released': { param_types: [GObject.TYPE_UINT, GObject.TYPE_STRING] },
|
2019-08-20 23:43:54 +02:00
|
|
|
|
},
|
2019-07-16 11:24:13 +02:00
|
|
|
|
}, class Key extends St.BoxLayout {
|
2020-02-14 18:09:32 +01:00
|
|
|
|
_init(key, extendedKeys, icon = null) {
|
2019-07-16 11:24:13 +02:00
|
|
|
|
super._init({ style_class: 'key-container' });
|
|
|
|
|
|
2018-02-15 13:46:19 +01:00
|
|
|
|
this.key = key || "";
|
2020-02-14 18:09:32 +01:00
|
|
|
|
this.keyButton = this._makeKey(this.key, icon);
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-08-05 12:21:30 +02:00
|
|
|
|
/* Add the key in a container, so keys can be padded without losing
|
|
|
|
|
* logical proportions between those.
|
|
|
|
|
*/
|
2019-10-21 20:44:00 +02:00
|
|
|
|
this.add_child(this.keyButton);
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.connect('destroy', this._onDestroy.bind(this));
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-12-04 18:51:08 +01:00
|
|
|
|
this._extendedKeys = extendedKeys;
|
|
|
|
|
this._extendedKeyboard = null;
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._pressTimeoutId = 0;
|
2017-12-22 15:59:19 +01:00
|
|
|
|
this._capturedPress = false;
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
2017-12-22 15:59:19 +01:00
|
|
|
|
this._capturedEventId = 0;
|
|
|
|
|
this._unmapId = 0;
|
2018-02-15 12:18:59 +01:00
|
|
|
|
this._longPress = false;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_onDestroy() {
|
2014-09-19 18:44:45 -06:00
|
|
|
|
if (this._boxPointer) {
|
2018-08-21 07:38:23 -03:00
|
|
|
|
this._boxPointer.destroy();
|
2014-09-19 18:44:45 -06:00
|
|
|
|
this._boxPointer = null;
|
|
|
|
|
}
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
this.cancel();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2014-09-19 18:44:45 -06:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_ensureExtendedKeysPopup() {
|
2019-12-04 18:51:08 +01:00
|
|
|
|
if (this._extendedKeys.length === 0)
|
2017-08-05 12:21:30 +02:00
|
|
|
|
return;
|
|
|
|
|
|
2019-12-04 19:02:33 +01:00
|
|
|
|
if (this._boxPointer)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-10-17 23:40:24 +02:00
|
|
|
|
this._boxPointer = new BoxPointer.BoxPointer(St.Side.BOTTOM);
|
2018-08-21 07:38:23 -03:00
|
|
|
|
this._boxPointer.hide();
|
2019-06-19 00:24:01 +02:00
|
|
|
|
Main.layoutManager.addTopChrome(this._boxPointer);
|
2018-02-11 16:19:10 +01:00
|
|
|
|
this._boxPointer.setPosition(this.keyButton, 0.5);
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
|
|
|
|
// Adds style to existing keyboard style to avoid repetition
|
2019-04-09 15:21:15 -05:00
|
|
|
|
this._boxPointer.add_style_class_name('keyboard-subkeys');
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._getExtendedKeys();
|
2019-12-04 18:51:08 +01:00
|
|
|
|
this.keyButton._extendedKeys = this._extendedKeyboard;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_getKeyval(key) {
|
2019-01-21 21:40:10 +01:00
|
|
|
|
let unicode = key.charCodeAt(0);
|
2018-11-26 13:39:57 +01:00
|
|
|
|
return Clutter.unicode_to_keysym(unicode);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_press(key) {
|
2019-01-29 02:18:52 +01:00
|
|
|
|
this.emit('activated');
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
2019-12-04 18:51:08 +01:00
|
|
|
|
if (key !== this.key || this._extendedKeys.length === 0)
|
2017-12-05 20:05:18 +01:00
|
|
|
|
this.emit('pressed', this._getKeyval(key), key);
|
2018-02-15 12:18:59 +01:00
|
|
|
|
|
|
|
|
|
if (key == this.key) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._pressTimeoutId = GLib.timeout_add(GLib.PRIORITY_DEFAULT,
|
2019-12-04 20:40:07 +01:00
|
|
|
|
KEY_LONG_PRESS_TIME,
|
|
|
|
|
() => {
|
|
|
|
|
this._longPress = true;
|
|
|
|
|
this._pressTimeoutId = 0;
|
|
|
|
|
|
|
|
|
|
this.emit('long-press');
|
|
|
|
|
|
2019-12-04 18:51:08 +01:00
|
|
|
|
if (this._extendedKeys.length > 0) {
|
2019-12-04 20:40:07 +01:00
|
|
|
|
this._touchPressed = false;
|
2019-12-04 20:01:41 +01:00
|
|
|
|
this._ensureExtendedKeysPopup();
|
2019-12-04 20:40:07 +01:00
|
|
|
|
this.keyButton.set_hover(false);
|
|
|
|
|
this.keyButton.fake_release();
|
|
|
|
|
this._showSubkeys();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return GLib.SOURCE_REMOVE;
|
|
|
|
|
});
|
2017-08-05 12:21:30 +02:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_release(key) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
if (this._pressTimeoutId != 0) {
|
|
|
|
|
GLib.source_remove(this._pressTimeoutId);
|
|
|
|
|
this._pressTimeoutId = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-04 18:51:08 +01:00
|
|
|
|
if (!this._longPress && key === this.key && this._extendedKeys.length > 0)
|
2018-02-15 12:18:59 +01:00
|
|
|
|
this.emit('pressed', this._getKeyval(key), key);
|
|
|
|
|
|
2017-12-05 20:05:18 +01:00
|
|
|
|
this.emit('released', this._getKeyval(key), key);
|
2017-12-22 15:59:19 +01:00
|
|
|
|
this._hideSubkeys();
|
2018-02-15 12:18:59 +01:00
|
|
|
|
this._longPress = false;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-22 15:59:19 +01:00
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
cancel() {
|
|
|
|
|
if (this._pressTimeoutId != 0) {
|
|
|
|
|
GLib.source_remove(this._pressTimeoutId);
|
|
|
|
|
this._pressTimeoutId = 0;
|
|
|
|
|
}
|
|
|
|
|
this._touchPressed = false;
|
|
|
|
|
this.keyButton.set_hover(false);
|
|
|
|
|
this.keyButton.fake_release();
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_onCapturedEvent(actor, event) {
|
2017-12-22 15:59:19 +01:00
|
|
|
|
let type = event.type();
|
2019-08-19 21:38:51 +02:00
|
|
|
|
let press = type == Clutter.EventType.BUTTON_PRESS || type == Clutter.EventType.TOUCH_BEGIN;
|
|
|
|
|
let release = type == Clutter.EventType.BUTTON_RELEASE || type == Clutter.EventType.TOUCH_END;
|
2017-12-22 15:59:19 +01:00
|
|
|
|
|
|
|
|
|
if (event.get_source() == this._boxPointer.bin ||
|
|
|
|
|
this._boxPointer.bin.contains(event.get_source()))
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
|
|
|
|
|
|
if (press)
|
|
|
|
|
this._capturedPress = true;
|
|
|
|
|
else if (release && this._capturedPress)
|
|
|
|
|
this._hideSubkeys();
|
|
|
|
|
|
|
|
|
|
return Clutter.EVENT_STOP;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-22 15:59:19 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_showSubkeys() {
|
2018-08-21 07:29:44 -03:00
|
|
|
|
this._boxPointer.open(BoxPointer.PopupAnimation.FULL);
|
2017-12-22 15:59:19 +01:00
|
|
|
|
this._capturedEventId = global.stage.connect('captured-event',
|
2017-12-02 01:27:35 +01:00
|
|
|
|
this._onCapturedEvent.bind(this));
|
2017-10-31 01:38:18 +01:00
|
|
|
|
this._unmapId = this.keyButton.connect('notify::mapped', () => {
|
2018-02-11 16:19:10 +01:00
|
|
|
|
if (!this.keyButton.is_mapped())
|
2017-12-22 15:59:19 +01:00
|
|
|
|
this._hideSubkeys();
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-22 15:59:19 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_hideSubkeys() {
|
2017-12-22 15:59:19 +01:00
|
|
|
|
if (this._boxPointer)
|
2018-08-21 07:29:44 -03:00
|
|
|
|
this._boxPointer.close(BoxPointer.PopupAnimation.FULL);
|
2017-12-22 15:59:19 +01:00
|
|
|
|
if (this._capturedEventId) {
|
|
|
|
|
global.stage.disconnect(this._capturedEventId);
|
|
|
|
|
this._capturedEventId = 0;
|
|
|
|
|
}
|
|
|
|
|
if (this._unmapId) {
|
2018-02-11 16:19:10 +01:00
|
|
|
|
this.keyButton.disconnect(this._unmapId);
|
2017-12-22 15:59:19 +01:00
|
|
|
|
this._unmapId = 0;
|
|
|
|
|
}
|
|
|
|
|
this._capturedPress = false;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2020-02-14 18:09:32 +01:00
|
|
|
|
_makeKey(key, icon) {
|
2019-08-19 19:55:49 +02:00
|
|
|
|
let button = new St.Button({
|
2019-10-21 20:44:00 +02:00
|
|
|
|
style_class: 'keyboard-key',
|
|
|
|
|
x_expand: true,
|
|
|
|
|
});
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2020-02-14 18:09:32 +01:00
|
|
|
|
if (icon) {
|
|
|
|
|
let child = new St.Icon({ icon_name: icon });
|
|
|
|
|
button.set_child(child);
|
|
|
|
|
this._icon = child;
|
|
|
|
|
} else {
|
|
|
|
|
let label = GLib.markup_escape_text(key, -1);
|
|
|
|
|
button.set_label(label);
|
|
|
|
|
}
|
|
|
|
|
|
2017-08-05 12:21:30 +02:00
|
|
|
|
button.keyWidth = 1;
|
2017-10-31 01:38:18 +01:00
|
|
|
|
button.connect('button-press-event', () => {
|
|
|
|
|
this._press(key);
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
|
});
|
|
|
|
|
button.connect('button-release-event', () => {
|
|
|
|
|
this._release(key);
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
|
});
|
|
|
|
|
button.connect('touch-event', (actor, event) => {
|
|
|
|
|
// We only handle touch events here on wayland. On X11
|
|
|
|
|
// we do get emulated pointer events, which already works
|
|
|
|
|
// for single-touch cases. Besides, the X11 passive touch grab
|
|
|
|
|
// set up by Mutter will make us see first the touch events
|
|
|
|
|
// and later the pointer events, so it will look like two
|
|
|
|
|
// unrelated series of events, we want to avoid double handling
|
|
|
|
|
// in these cases.
|
|
|
|
|
if (!Meta.is_wayland_compositor())
|
2013-11-29 18:17:34 +00:00
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2017-10-31 01:38:18 +01:00
|
|
|
|
|
|
|
|
|
if (!this._touchPressed &&
|
|
|
|
|
event.type() == Clutter.EventType.TOUCH_BEGIN) {
|
|
|
|
|
this._touchPressed = true;
|
|
|
|
|
this._press(key);
|
|
|
|
|
} else if (this._touchPressed &&
|
2019-01-21 20:40:31 +01:00
|
|
|
|
event.type() == Clutter.EventType.TOUCH_END) {
|
2017-10-31 01:38:18 +01:00
|
|
|
|
this._touchPressed = false;
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._release(key);
|
2017-10-31 01:38:18 +01:00
|
|
|
|
}
|
|
|
|
|
return Clutter.EVENT_PROPAGATE;
|
|
|
|
|
});
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
|
|
|
|
return button;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_getExtendedKeys() {
|
2019-12-04 18:51:08 +01:00
|
|
|
|
this._extendedKeyboard = new St.BoxLayout({
|
|
|
|
|
style_class: 'key-container',
|
|
|
|
|
vertical: false,
|
|
|
|
|
});
|
|
|
|
|
for (let i = 0; i < this._extendedKeys.length; ++i) {
|
|
|
|
|
let extendedKey = this._extendedKeys[i];
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let key = this._makeKey(extendedKey);
|
2014-07-22 12:38:44 +02:00
|
|
|
|
|
2019-12-04 18:51:08 +01:00
|
|
|
|
key.extendedKey = extendedKey;
|
|
|
|
|
this._extendedKeyboard.add(key);
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2019-12-04 19:48:36 +01:00
|
|
|
|
key.set_size(...this.keyButton.allocation.get_size());
|
2019-12-04 19:49:17 +01:00
|
|
|
|
this.keyButton.connect('notify::allocation',
|
|
|
|
|
() => key.set_size(...this.keyButton.allocation.get_size()));
|
2011-08-29 11:11:22 -04:00
|
|
|
|
}
|
2019-12-04 18:51:08 +01:00
|
|
|
|
this._boxPointer.bin.add_actor(this._extendedKeyboard);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2012-11-10 22:47:57 +01:00
|
|
|
|
get subkeys() {
|
|
|
|
|
return this._boxPointer;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
setWidth(width) {
|
2018-02-11 16:19:10 +01:00
|
|
|
|
this.keyButton.keyWidth = width;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-02-15 14:48:46 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
setLatched(latched) {
|
2020-02-14 18:09:32 +01:00
|
|
|
|
if (!this._icon)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (latched) {
|
2018-02-15 14:48:46 +01:00
|
|
|
|
this.keyButton.add_style_pseudo_class('latched');
|
2020-02-14 18:09:32 +01:00
|
|
|
|
this._icon.icon_name = 'keyboard-caps-lock-filled-symbolic';
|
|
|
|
|
} else {
|
2018-02-15 14:48:46 +01:00
|
|
|
|
this.keyButton.remove_style_pseudo_class('latched');
|
2020-02-14 18:09:32 +01:00
|
|
|
|
this._icon.icon_name = 'keyboard-shift-filled-symbolic';
|
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2019-07-16 11:24:13 +02:00
|
|
|
|
});
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 02:19:44 +01:00
|
|
|
|
var KeyboardModel = class {
|
|
|
|
|
constructor(groupName) {
|
2019-11-16 13:48:35 +01:00
|
|
|
|
let names = [groupName];
|
2020-03-23 16:14:47 +01:00
|
|
|
|
if (groupName.includes('+'))
|
2019-11-16 13:48:35 +01:00
|
|
|
|
names.push(groupName.replace(/\+.*/, ''));
|
|
|
|
|
names.push('us');
|
|
|
|
|
|
|
|
|
|
for (let i = 0; i < names.length; i++) {
|
|
|
|
|
try {
|
|
|
|
|
this._model = this._loadModel(names[i]);
|
|
|
|
|
break;
|
|
|
|
|
} catch (e) {
|
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_loadModel(groupName) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let file = Gio.File.new_for_uri('resource:///org/gnome/shell/osk-layouts/%s.json'.format(groupName));
|
2019-01-31 15:08:00 +01:00
|
|
|
|
let [success_, contents] = file.load_contents(null);
|
2020-08-13 23:35:27 +02:00
|
|
|
|
contents = ByteArray.toString(contents);
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
|
|
|
|
return JSON.parse(contents);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
getLevels() {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
return this._model.levels;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
getKeysForLevel(levelName) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
return this._model.levels.find(level => level == levelName);
|
2011-08-29 11:11:22 -04:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
};
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
2017-10-31 02:19:44 +01:00
|
|
|
|
var FocusTracker = class {
|
|
|
|
|
constructor() {
|
2018-05-25 11:35:49 +02:00
|
|
|
|
this._currentWindow = null;
|
2018-09-17 21:04:48 +02:00
|
|
|
|
this._rect = null;
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
2018-07-11 18:46:49 +02:00
|
|
|
|
global.display.connect('notify::focus-window', () => {
|
|
|
|
|
this._setCurrentWindow(global.display.focus_window);
|
2018-05-25 11:35:49 +02:00
|
|
|
|
this.emit('window-changed', this._currentWindow);
|
|
|
|
|
});
|
|
|
|
|
|
2018-08-03 16:01:48 +02:00
|
|
|
|
global.display.connect('grab-op-begin', (display, window, op) => {
|
|
|
|
|
if (window == this._currentWindow &&
|
|
|
|
|
(op == Meta.GrabOp.MOVING || op == Meta.GrabOp.KEYBOARD_MOVING))
|
|
|
|
|
this.emit('reset');
|
|
|
|
|
});
|
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
/* Valid for wayland clients */
|
|
|
|
|
Main.inputMethod.connect('cursor-location-changed', (o, rect) => {
|
|
|
|
|
let newRect = { x: rect.get_x(), y: rect.get_y(), width: rect.get_width(), height: rect.get_height() };
|
|
|
|
|
this._setCurrentRect(newRect);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
this._ibusManager = IBusManager.getIBusManager();
|
|
|
|
|
this._ibusManager.connect('set-cursor-location', (manager, rect) => {
|
|
|
|
|
/* Valid for X11 clients only */
|
|
|
|
|
if (Main.inputMethod.currentFocus)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
this._setCurrentRect(rect);
|
|
|
|
|
});
|
2018-09-18 12:54:29 +02:00
|
|
|
|
this._ibusManager.connect('focus-in', () => {
|
|
|
|
|
this.emit('focus-changed', true);
|
|
|
|
|
});
|
|
|
|
|
this._ibusManager.connect('focus-out', () => {
|
|
|
|
|
this.emit('focus-changed', false);
|
|
|
|
|
});
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
|
|
|
|
get currentWindow() {
|
|
|
|
|
return this._currentWindow;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
|
|
|
|
_setCurrentWindow(window) {
|
|
|
|
|
this._currentWindow = window;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
|
|
|
|
_setCurrentRect(rect) {
|
2018-07-17 01:36:38 +02:00
|
|
|
|
if (this._currentWindow) {
|
|
|
|
|
let frameRect = this._currentWindow.get_frame_rect();
|
|
|
|
|
rect.x -= frameRect.x;
|
|
|
|
|
rect.y -= frameRect.y;
|
|
|
|
|
}
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
2018-09-17 21:04:48 +02:00
|
|
|
|
if (this._rect &&
|
|
|
|
|
this._rect.x == rect.x &&
|
|
|
|
|
this._rect.y == rect.y &&
|
|
|
|
|
this._rect.width == rect.width &&
|
|
|
|
|
this._rect.height == rect.height)
|
|
|
|
|
return;
|
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
this._rect = rect;
|
|
|
|
|
this.emit('position-changed');
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
|
|
|
|
getCurrentRect() {
|
2018-07-17 01:36:38 +02:00
|
|
|
|
let rect = { x: this._rect.x, y: this._rect.y,
|
|
|
|
|
width: this._rect.width, height: this._rect.height };
|
|
|
|
|
|
|
|
|
|
if (this._currentWindow) {
|
|
|
|
|
let frameRect = this._currentWindow.get_frame_rect();
|
|
|
|
|
rect.x += frameRect.x;
|
|
|
|
|
rect.y += frameRect.y;
|
|
|
|
|
}
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
|
|
|
|
return rect;
|
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
};
|
2018-05-25 11:35:49 +02:00
|
|
|
|
Signals.addSignalMethods(FocusTracker.prototype);
|
|
|
|
|
|
2019-07-25 18:53:00 +02:00
|
|
|
|
var EmojiPager = GObject.registerClass({
|
2019-07-25 22:13:19 +00:00
|
|
|
|
Properties: {
|
|
|
|
|
'delta': GObject.ParamSpec.int(
|
|
|
|
|
'delta', 'delta', 'delta',
|
|
|
|
|
GObject.ParamFlags.READWRITE,
|
2019-08-20 23:43:54 +02:00
|
|
|
|
GLib.MININT32, GLib.MAXINT32, 0),
|
2019-07-25 22:13:19 +00:00
|
|
|
|
},
|
2019-07-25 18:53:00 +02:00
|
|
|
|
Signals: {
|
|
|
|
|
'emoji': { param_types: [GObject.TYPE_STRING] },
|
|
|
|
|
'page-changed': {
|
2019-08-20 23:43:54 +02:00
|
|
|
|
param_types: [GObject.TYPE_INT, GObject.TYPE_INT, GObject.TYPE_INT],
|
|
|
|
|
},
|
|
|
|
|
},
|
2019-07-25 18:53:00 +02:00
|
|
|
|
}, class EmojiPager extends St.Widget {
|
|
|
|
|
_init(sections, nCols, nRows) {
|
|
|
|
|
super._init({
|
|
|
|
|
layout_manager: new Clutter.BinLayout(),
|
|
|
|
|
reactive: true,
|
2019-08-20 23:43:54 +02:00
|
|
|
|
clip_to_allocation: true,
|
2020-01-21 16:22:34 +01:00
|
|
|
|
y_expand: true,
|
2019-07-25 18:53:00 +02:00
|
|
|
|
});
|
2017-12-22 16:02:51 +01:00
|
|
|
|
this._sections = sections;
|
|
|
|
|
this._nCols = nCols;
|
|
|
|
|
this._nRows = nRows;
|
|
|
|
|
|
|
|
|
|
this._pages = [];
|
|
|
|
|
this._panel = null;
|
|
|
|
|
this._curPage = null;
|
|
|
|
|
this._followingPage = null;
|
|
|
|
|
this._followingPanel = null;
|
|
|
|
|
this._currentKey = null;
|
|
|
|
|
this._delta = 0;
|
|
|
|
|
this._width = null;
|
|
|
|
|
|
|
|
|
|
this._initPagingInfo();
|
|
|
|
|
|
|
|
|
|
let panAction = new Clutter.PanAction({ interpolate: false });
|
|
|
|
|
panAction.connect('pan', this._onPan.bind(this));
|
|
|
|
|
panAction.connect('gesture-begin', this._onPanBegin.bind(this));
|
|
|
|
|
panAction.connect('gesture-cancel', this._onPanCancel.bind(this));
|
|
|
|
|
panAction.connect('gesture-end', this._onPanEnd.bind(this));
|
|
|
|
|
this._panAction = panAction;
|
2019-07-25 18:53:00 +02:00
|
|
|
|
this.add_action(panAction);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
get delta() {
|
|
|
|
|
return this._delta;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
set delta(value) {
|
|
|
|
|
if (value > this._width)
|
|
|
|
|
value = this._width;
|
|
|
|
|
else if (value < -this._width)
|
|
|
|
|
value = -this._width;
|
|
|
|
|
|
2019-07-25 22:13:19 +00:00
|
|
|
|
if (this._delta == value)
|
|
|
|
|
return;
|
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
this._delta = value;
|
2019-07-25 22:13:19 +00:00
|
|
|
|
this.notify('delta');
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
if (value == 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
let relValue = Math.abs(value / this._width);
|
|
|
|
|
let followingPage = this.getFollowingPage();
|
|
|
|
|
|
|
|
|
|
if (this._followingPage != followingPage) {
|
|
|
|
|
if (this._followingPanel) {
|
|
|
|
|
this._followingPanel.destroy();
|
|
|
|
|
this._followingPanel = null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (followingPage != null) {
|
|
|
|
|
this._followingPanel = this._generatePanel(followingPage);
|
|
|
|
|
this._followingPanel.set_pivot_point(0.5, 0.5);
|
2019-07-25 18:53:00 +02:00
|
|
|
|
this.add_child(this._followingPanel);
|
|
|
|
|
this.set_child_below_sibling(this._followingPanel, this._panel);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this._followingPage = followingPage;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this._panel.translation_x = value;
|
|
|
|
|
this._panel.opacity = 255 * (1 - Math.pow(relValue, 3));
|
|
|
|
|
|
|
|
|
|
if (this._followingPanel) {
|
|
|
|
|
this._followingPanel.scale_x = 0.8 + (0.2 * relValue);
|
|
|
|
|
this._followingPanel.scale_y = 0.8 + (0.2 * relValue);
|
|
|
|
|
this._followingPanel.opacity = 255 * relValue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_prevPage(nPage) {
|
|
|
|
|
return (nPage + this._pages.length - 1) % this._pages.length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_nextPage(nPage) {
|
|
|
|
|
return (nPage + 1) % this._pages.length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
getFollowingPage() {
|
|
|
|
|
if (this.delta == 0)
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
if ((this.delta < 0 && global.stage.text_direction == Clutter.TextDirection.LTR) ||
|
|
|
|
|
(this.delta > 0 && global.stage.text_direction == Clutter.TextDirection.RTL))
|
|
|
|
|
return this._nextPage(this._curPage);
|
|
|
|
|
else
|
|
|
|
|
return this._prevPage(this._curPage);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_onPan(action) {
|
2019-01-31 15:08:00 +01:00
|
|
|
|
let [dist_, dx, dy_] = action.get_motion_delta(0);
|
2020-04-04 02:00:35 +02:00
|
|
|
|
this.delta += dx;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
if (this._currentKey != null) {
|
|
|
|
|
this._currentKey.cancel();
|
|
|
|
|
this._currentKey = null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_onPanBegin() {
|
2019-07-25 18:53:00 +02:00
|
|
|
|
this._width = this.width;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_onPanEnd() {
|
2019-07-25 18:53:00 +02:00
|
|
|
|
if (Math.abs(this._delta) < this.width * PANEL_SWITCH_RELATIVE_DISTANCE) {
|
2019-01-29 02:18:52 +01:00
|
|
|
|
this._onPanCancel();
|
2017-12-22 16:02:51 +01:00
|
|
|
|
} else {
|
|
|
|
|
let value;
|
|
|
|
|
if (this._delta > 0)
|
|
|
|
|
value = this._width;
|
|
|
|
|
else if (this._delta < 0)
|
|
|
|
|
value = -this._width;
|
|
|
|
|
|
|
|
|
|
let relDelta = Math.abs(this._delta - value) / this._width;
|
|
|
|
|
let time = PANEL_SWITCH_ANIMATION_TIME * Math.abs(relDelta);
|
|
|
|
|
|
2019-07-25 02:06:05 +02:00
|
|
|
|
this.remove_all_transitions();
|
|
|
|
|
this.ease_property('delta', value, {
|
|
|
|
|
duration: time,
|
|
|
|
|
onComplete: () => {
|
|
|
|
|
this.setCurrentPage(this.getFollowingPage());
|
2019-08-20 23:43:54 +02:00
|
|
|
|
},
|
2019-07-25 02:06:05 +02:00
|
|
|
|
});
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_onPanCancel() {
|
2019-07-25 18:53:00 +02:00
|
|
|
|
let relDelta = Math.abs(this._delta) / this.width;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
let time = PANEL_SWITCH_ANIMATION_TIME * Math.abs(relDelta);
|
|
|
|
|
|
2019-07-25 02:06:05 +02:00
|
|
|
|
this.remove_all_transitions();
|
|
|
|
|
this.ease_property('delta', 0, {
|
|
|
|
|
duration: time,
|
|
|
|
|
});
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_initPagingInfo() {
|
|
|
|
|
for (let i = 0; i < this._sections.length; i++) {
|
|
|
|
|
let section = this._sections[i];
|
|
|
|
|
let itemsPerPage = this._nCols * this._nRows;
|
|
|
|
|
let nPages = Math.ceil(section.keys.length / itemsPerPage);
|
|
|
|
|
let page = -1;
|
|
|
|
|
let pageKeys;
|
|
|
|
|
|
|
|
|
|
for (let j = 0; j < section.keys.length; j++) {
|
|
|
|
|
if (j % itemsPerPage == 0) {
|
|
|
|
|
page++;
|
|
|
|
|
pageKeys = [];
|
|
|
|
|
this._pages.push({ pageKeys, nPages, page, section: this._sections[i] });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pageKeys.push(section.keys[j]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_lookupSection(section, nPage) {
|
|
|
|
|
for (let i = 0; i < this._pages.length; i++) {
|
|
|
|
|
let page = this._pages[i];
|
|
|
|
|
|
|
|
|
|
if (page.section == section && page.page == nPage)
|
|
|
|
|
return i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_generatePanel(nPage) {
|
|
|
|
|
let gridLayout = new Clutter.GridLayout({ orientation: Clutter.Orientation.HORIZONTAL,
|
|
|
|
|
column_homogeneous: true,
|
|
|
|
|
row_homogeneous: true });
|
|
|
|
|
let panel = new St.Widget({ layout_manager: gridLayout,
|
|
|
|
|
style_class: 'emoji-page',
|
|
|
|
|
x_expand: true,
|
|
|
|
|
y_expand: true });
|
|
|
|
|
|
|
|
|
|
/* Set an expander actor so all proportions are right despite the panel
|
|
|
|
|
* not having all rows/cols filled in.
|
|
|
|
|
*/
|
|
|
|
|
let expander = new Clutter.Actor();
|
|
|
|
|
gridLayout.attach(expander, 0, 0, this._nCols, this._nRows);
|
|
|
|
|
|
|
|
|
|
let page = this._pages[nPage];
|
|
|
|
|
let col = 0;
|
|
|
|
|
let row = 0;
|
|
|
|
|
|
|
|
|
|
for (let i = 0; i < page.pageKeys.length; i++) {
|
|
|
|
|
let modelKey = page.pageKeys[i];
|
|
|
|
|
let key = new Key(modelKey.label, modelKey.variants);
|
|
|
|
|
|
|
|
|
|
key.keyButton.set_button_mask(0);
|
|
|
|
|
|
|
|
|
|
key.connect('activated', () => {
|
|
|
|
|
this._currentKey = key;
|
|
|
|
|
});
|
|
|
|
|
key.connect('long-press', () => {
|
|
|
|
|
this._panAction.cancel();
|
|
|
|
|
});
|
|
|
|
|
key.connect('released', (actor, keyval, str) => {
|
|
|
|
|
if (this._currentKey != key)
|
|
|
|
|
return;
|
|
|
|
|
this._currentKey = null;
|
|
|
|
|
this.emit('emoji', str);
|
|
|
|
|
});
|
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
gridLayout.attach(key, col, row, 1, 1);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
col++;
|
|
|
|
|
if (col >= this._nCols) {
|
|
|
|
|
col = 0;
|
|
|
|
|
row++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return panel;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setCurrentPage(nPage) {
|
|
|
|
|
if (this._curPage == nPage)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
this._curPage = nPage;
|
|
|
|
|
|
|
|
|
|
if (this._panel) {
|
|
|
|
|
this._panel.destroy();
|
|
|
|
|
this._panel = null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Reuse followingPage if possible */
|
|
|
|
|
if (nPage == this._followingPage) {
|
|
|
|
|
this._panel = this._followingPanel;
|
|
|
|
|
this._followingPanel = null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (this._followingPanel)
|
|
|
|
|
this._followingPanel.destroy();
|
|
|
|
|
|
|
|
|
|
this._followingPanel = null;
|
|
|
|
|
this._followingPage = null;
|
|
|
|
|
this._delta = 0;
|
|
|
|
|
|
|
|
|
|
if (!this._panel) {
|
|
|
|
|
this._panel = this._generatePanel(nPage);
|
2019-07-25 18:53:00 +02:00
|
|
|
|
this.add_child(this._panel);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let page = this._pages[nPage];
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.emit('page-changed', page.section.label, page.page, page.nPages);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setCurrentSection(section, nPage) {
|
|
|
|
|
for (let i = 0; i < this._pages.length; i++) {
|
|
|
|
|
let page = this._pages[i];
|
|
|
|
|
|
|
|
|
|
if (page.section == section && page.page == nPage) {
|
|
|
|
|
this.setCurrentPage(i);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-07-25 18:53:00 +02:00
|
|
|
|
});
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
var EmojiSelection = GObject.registerClass({
|
|
|
|
|
Signals: {
|
|
|
|
|
'emoji-selected': { param_types: [GObject.TYPE_STRING] },
|
|
|
|
|
'close-request': {},
|
|
|
|
|
'toggle': {},
|
2019-08-20 23:43:54 +02:00
|
|
|
|
},
|
2019-07-16 11:24:13 +02:00
|
|
|
|
}, class EmojiSelection extends St.BoxLayout {
|
|
|
|
|
_init() {
|
|
|
|
|
super._init({
|
|
|
|
|
style_class: 'emoji-panel',
|
|
|
|
|
x_expand: true,
|
|
|
|
|
y_expand: true,
|
2019-08-20 23:43:54 +02:00
|
|
|
|
vertical: true,
|
2019-07-16 11:24:13 +02:00
|
|
|
|
});
|
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
this._sections = [
|
|
|
|
|
{ first: 'grinning face', label: '🙂️' },
|
|
|
|
|
{ first: 'selfie', label: '👍️' },
|
|
|
|
|
{ first: 'monkey face', label: '🌷️' },
|
|
|
|
|
{ first: 'grapes', label: '🍴️' },
|
|
|
|
|
{ first: 'globe showing Europe-Africa', label: '✈️' },
|
|
|
|
|
{ first: 'jack-o-lantern', label: '🏃️' },
|
|
|
|
|
{ first: 'muted speaker', label: '🔔️' },
|
|
|
|
|
{ first: 'ATM sign', label: '❤️' },
|
|
|
|
|
{ first: 'chequered flag', label: '🚩️' },
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
this._populateSections();
|
|
|
|
|
|
|
|
|
|
this._emojiPager = new EmojiPager(this._sections, 11, 3);
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this._emojiPager.connect('page-changed', (pager, sectionLabel, page, nPages) => {
|
|
|
|
|
this._onPageChanged(sectionLabel, page, nPages);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
});
|
|
|
|
|
this._emojiPager.connect('emoji', (pager, str) => {
|
|
|
|
|
this.emit('emoji-selected', str);
|
|
|
|
|
});
|
2019-10-21 20:44:00 +02:00
|
|
|
|
this.add_child(this._emojiPager);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
2019-09-10 07:36:58 +02:00
|
|
|
|
this._pageIndicator = new PageIndicators.PageIndicators(
|
2020-04-04 01:52:29 +02:00
|
|
|
|
Clutter.Orientation.HORIZONTAL);
|
2019-10-21 20:44:00 +02:00
|
|
|
|
this.add_child(this._pageIndicator);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
this._pageIndicator.setReactive(false);
|
|
|
|
|
|
2019-11-22 03:24:46 +05:00
|
|
|
|
this._emojiPager.connect('notify::delta', () => {
|
|
|
|
|
this._updateIndicatorPosition();
|
|
|
|
|
});
|
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
let bottomRow = this._createBottomRow();
|
2019-10-21 20:44:00 +02:00
|
|
|
|
this.add_child(bottomRow);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
2019-11-22 03:24:46 +05:00
|
|
|
|
this._curPage = 0;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
2019-09-10 07:42:48 +02:00
|
|
|
|
vfunc_map() {
|
|
|
|
|
this._emojiPager.setCurrentPage(0);
|
|
|
|
|
super.vfunc_map();
|
|
|
|
|
}
|
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
_onPageChanged(sectionLabel, page, nPages) {
|
2019-11-22 03:24:46 +05:00
|
|
|
|
this._curPage = page;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
this._pageIndicator.setNPages(nPages);
|
2019-11-22 03:24:46 +05:00
|
|
|
|
this._updateIndicatorPosition();
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
for (let i = 0; i < this._sections.length; i++) {
|
|
|
|
|
let sect = this._sections[i];
|
2019-07-16 11:24:13 +02:00
|
|
|
|
sect.button.setLatched(sectionLabel == sect.label);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-22 03:24:46 +05:00
|
|
|
|
_updateIndicatorPosition() {
|
|
|
|
|
this._pageIndicator.setCurrentPosition(this._curPage -
|
|
|
|
|
this._emojiPager.delta / this._emojiPager.width);
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
_findSection(emoji) {
|
|
|
|
|
for (let i = 0; i < this._sections.length; i++) {
|
|
|
|
|
if (this._sections[i].first == emoji)
|
|
|
|
|
return this._sections[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_populateSections() {
|
|
|
|
|
let file = Gio.File.new_for_uri('resource:///org/gnome/shell/osk-layouts/emoji.json');
|
2019-01-31 15:08:00 +01:00
|
|
|
|
let [success_, contents] = file.load_contents(null);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
if (contents instanceof Uint8Array)
|
|
|
|
|
contents = imports.byteArray.toString(contents);
|
|
|
|
|
let emoji = JSON.parse(contents);
|
|
|
|
|
|
|
|
|
|
let variants = [];
|
|
|
|
|
let currentKey = 0;
|
|
|
|
|
let currentSection = null;
|
|
|
|
|
|
|
|
|
|
for (let i = 0; i < emoji.length; i++) {
|
|
|
|
|
/* Group variants of a same emoji so they appear on the key popover */
|
|
|
|
|
if (emoji[i].name.startsWith(emoji[currentKey].name)) {
|
|
|
|
|
variants.push(emoji[i].char);
|
|
|
|
|
if (i < emoji.length - 1)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let newSection = this._findSection(emoji[currentKey].name);
|
|
|
|
|
if (newSection != null) {
|
|
|
|
|
currentSection = newSection;
|
|
|
|
|
currentSection.keys = [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Create the key */
|
|
|
|
|
let label = emoji[currentKey].char + String.fromCharCode(0xFE0F);
|
|
|
|
|
currentSection.keys.push({ label, variants });
|
|
|
|
|
currentKey = i;
|
|
|
|
|
variants = [];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_createBottomRow() {
|
|
|
|
|
let row = new KeyContainer();
|
|
|
|
|
let key;
|
|
|
|
|
|
|
|
|
|
row.appendRow();
|
|
|
|
|
|
|
|
|
|
key = new Key('ABC', []);
|
|
|
|
|
key.keyButton.add_style_class_name('default-key');
|
2019-01-28 01:42:00 +01:00
|
|
|
|
key.connect('released', () => this.emit('toggle'));
|
2019-07-16 11:24:13 +02:00
|
|
|
|
row.appendKey(key, 1.5);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
for (let i = 0; i < this._sections.length; i++) {
|
|
|
|
|
let section = this._sections[i];
|
|
|
|
|
|
|
|
|
|
key = new Key(section.label, []);
|
2019-01-28 01:42:00 +01:00
|
|
|
|
key.connect('released', () => this._emojiPager.setCurrentSection(section, 0));
|
2019-07-16 11:24:13 +02:00
|
|
|
|
row.appendKey(key);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
section.button = key;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-14 18:11:43 +01:00
|
|
|
|
key = new Key(null, [], 'go-down-symbolic');
|
2017-12-22 16:02:51 +01:00
|
|
|
|
key.keyButton.add_style_class_name('default-key');
|
|
|
|
|
key.keyButton.add_style_class_name('hide-key');
|
|
|
|
|
key.connect('released', () => {
|
2019-08-28 23:06:14 +03:00
|
|
|
|
this.emit('close-request');
|
2017-12-22 16:02:51 +01:00
|
|
|
|
});
|
2019-07-16 11:24:13 +02:00
|
|
|
|
row.appendKey(key);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
row.layoutButtons();
|
|
|
|
|
|
2019-03-05 19:34:23 +01:00
|
|
|
|
let actor = new AspectContainer({ layout_manager: new Clutter.BinLayout(),
|
|
|
|
|
x_expand: true, y_expand: true });
|
|
|
|
|
actor.add_child(row);
|
|
|
|
|
/* Regular keyboard layouts are 11.5×4 grids, optimize for that
|
|
|
|
|
* at the moment. Ideally this should be as wide as the current
|
|
|
|
|
* keymap.
|
|
|
|
|
*/
|
|
|
|
|
actor.setRatio(11.5, 1);
|
|
|
|
|
|
|
|
|
|
return actor;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
2019-07-16 11:24:13 +02:00
|
|
|
|
});
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
var Keypad = GObject.registerClass({
|
|
|
|
|
Signals: {
|
|
|
|
|
'keyval': { param_types: [GObject.TYPE_UINT] },
|
2019-08-20 23:43:54 +02:00
|
|
|
|
},
|
2019-07-16 11:24:13 +02:00
|
|
|
|
}, class Keypad extends AspectContainer {
|
|
|
|
|
_init() {
|
2019-01-19 15:58:45 +01:00
|
|
|
|
let keys = [
|
|
|
|
|
{ label: '1', keyval: Clutter.KEY_1, left: 0, top: 0 },
|
|
|
|
|
{ label: '2', keyval: Clutter.KEY_2, left: 1, top: 0 },
|
|
|
|
|
{ label: '3', keyval: Clutter.KEY_3, left: 2, top: 0 },
|
|
|
|
|
{ label: '4', keyval: Clutter.KEY_4, left: 0, top: 1 },
|
|
|
|
|
{ label: '5', keyval: Clutter.KEY_5, left: 1, top: 1 },
|
|
|
|
|
{ label: '6', keyval: Clutter.KEY_6, left: 2, top: 1 },
|
|
|
|
|
{ label: '7', keyval: Clutter.KEY_7, left: 0, top: 2 },
|
|
|
|
|
{ label: '8', keyval: Clutter.KEY_8, left: 1, top: 2 },
|
|
|
|
|
{ label: '9', keyval: Clutter.KEY_9, left: 2, top: 2 },
|
|
|
|
|
{ label: '0', keyval: Clutter.KEY_0, left: 1, top: 3 },
|
2020-08-06 13:28:03 +03:00
|
|
|
|
{ keyval: Clutter.KEY_BackSpace, icon: 'edit-clear-symbolic', left: 3, top: 0 },
|
|
|
|
|
{ keyval: Clutter.KEY_Return, extraClassName: 'enter-key', icon: 'keyboard-enter-symbolic', left: 3, top: 1, height: 2 },
|
2019-01-19 15:58:45 +01:00
|
|
|
|
];
|
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
super._init({
|
|
|
|
|
layout_manager: new Clutter.BinLayout(),
|
|
|
|
|
x_expand: true,
|
2019-08-20 23:43:54 +02:00
|
|
|
|
y_expand: true,
|
2019-07-16 11:24:13 +02:00
|
|
|
|
});
|
2019-01-19 15:58:45 +01:00
|
|
|
|
|
|
|
|
|
let gridLayout = new Clutter.GridLayout({ orientation: Clutter.Orientation.HORIZONTAL,
|
|
|
|
|
column_homogeneous: true,
|
|
|
|
|
row_homogeneous: true });
|
|
|
|
|
this._box = new St.Widget({ layout_manager: gridLayout, x_expand: true, y_expand: true });
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.add_child(this._box);
|
2019-01-19 15:58:45 +01:00
|
|
|
|
|
|
|
|
|
for (let i = 0; i < keys.length; i++) {
|
|
|
|
|
let cur = keys[i];
|
2020-08-06 13:28:03 +03:00
|
|
|
|
let key = new Key(cur.label || "", [], cur.icon);
|
2019-01-19 15:58:45 +01:00
|
|
|
|
|
|
|
|
|
if (keys[i].extraClassName)
|
|
|
|
|
key.keyButton.add_style_class_name(cur.extraClassName);
|
|
|
|
|
|
|
|
|
|
let w, h;
|
|
|
|
|
w = cur.width || 1;
|
|
|
|
|
h = cur.height || 1;
|
2019-07-16 11:24:13 +02:00
|
|
|
|
gridLayout.attach(key, cur.left, cur.top, w, h);
|
2019-01-19 15:58:45 +01:00
|
|
|
|
|
|
|
|
|
key.connect('released', () => {
|
|
|
|
|
this.emit('keyval', cur.keyval);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-07-16 11:24:13 +02:00
|
|
|
|
});
|
2019-01-19 15:58:45 +01:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
var KeyboardManager = class KeyBoardManager {
|
|
|
|
|
constructor() {
|
|
|
|
|
this._keyboard = null;
|
|
|
|
|
this._a11yApplicationsSettings = new Gio.Settings({ schema_id: A11Y_APPLICATIONS_SCHEMA });
|
|
|
|
|
this._a11yApplicationsSettings.connect('changed', this._syncEnabled.bind(this));
|
|
|
|
|
|
2020-02-10 20:03:30 +01:00
|
|
|
|
this._seat = Clutter.get_default_backend().get_default_seat();
|
|
|
|
|
this._seat.connect('notify::touch-mode', this._syncEnabled.bind(this));
|
|
|
|
|
|
2019-10-05 12:34:07 +02:00
|
|
|
|
this._lastDevice = null;
|
2020-04-21 18:04:56 +02:00
|
|
|
|
global.backend.connect('last-device-changed', (backend, device) => {
|
2020-03-28 14:15:09 +01:00
|
|
|
|
if (device.device_type === Clutter.InputDeviceType.KEYBOARD_DEVICE)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
this._lastDevice = device;
|
|
|
|
|
this._syncEnabled();
|
2019-08-28 23:52:52 +02:00
|
|
|
|
});
|
|
|
|
|
this._syncEnabled();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_lastDeviceIsTouchscreen() {
|
2019-10-05 12:34:07 +02:00
|
|
|
|
if (!this._lastDevice)
|
2019-08-28 23:52:52 +02:00
|
|
|
|
return false;
|
|
|
|
|
|
2019-10-05 12:34:07 +02:00
|
|
|
|
let deviceType = this._lastDevice.get_device_type();
|
|
|
|
|
return deviceType == Clutter.InputDeviceType.TOUCHSCREEN_DEVICE;
|
2019-08-28 23:52:52 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_syncEnabled() {
|
|
|
|
|
let enableKeyboard = this._a11yApplicationsSettings.get_boolean(SHOW_KEYBOARD);
|
2020-02-10 20:03:30 +01:00
|
|
|
|
let autoEnabled = this._seat.get_touch_mode() && this._lastDeviceIsTouchscreen();
|
|
|
|
|
let enabled = enableKeyboard || autoEnabled;
|
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
if (!enabled && !this._keyboard)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (enabled && !this._keyboard) {
|
|
|
|
|
this._keyboard = new Keyboard();
|
|
|
|
|
} else if (!enabled && this._keyboard) {
|
|
|
|
|
this._keyboard.setCursorLocation(null);
|
|
|
|
|
this._keyboard.destroy();
|
|
|
|
|
this._keyboard = null;
|
2020-03-28 14:36:20 +01:00
|
|
|
|
Main.layoutManager.hideKeyboard(true);
|
2019-08-28 23:52:52 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
get keyboardActor() {
|
2019-07-16 11:24:13 +02:00
|
|
|
|
return this._keyboard;
|
2019-08-28 23:52:52 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
get visible() {
|
2019-07-16 11:24:13 +02:00
|
|
|
|
return this._keyboard && this._keyboard.visible;
|
2019-08-28 23:52:52 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
open(monitor) {
|
|
|
|
|
if (this._keyboard)
|
|
|
|
|
this._keyboard.open(monitor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
close() {
|
|
|
|
|
if (this._keyboard)
|
|
|
|
|
this._keyboard.close();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
addSuggestion(text, callback) {
|
|
|
|
|
if (this._keyboard)
|
|
|
|
|
this._keyboard.addSuggestion(text, callback);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
resetSuggestions() {
|
|
|
|
|
if (this._keyboard)
|
|
|
|
|
this._keyboard.resetSuggestions();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
shouldTakeEvent(event) {
|
|
|
|
|
if (!this._keyboard)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
let actor = event.get_source();
|
|
|
|
|
return Main.layoutManager.keyboardBox.contains(actor) ||
|
2019-12-04 18:51:08 +01:00
|
|
|
|
!!actor._extendedKeys || !!actor.extendedKey;
|
2019-08-28 23:52:52 +02:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
var Keyboard = GObject.registerClass(
|
|
|
|
|
class Keyboard extends St.BoxLayout {
|
|
|
|
|
_init() {
|
|
|
|
|
super._init({ name: 'keyboard', vertical: true });
|
2012-09-24 20:03:53 +02:00
|
|
|
|
this._focusInExtendedKeys = false;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
this._emojiActive = false;
|
2011-09-01 11:01:48 -04:00
|
|
|
|
|
2018-01-23 16:45:46 +01:00
|
|
|
|
this._languagePopup = null;
|
2017-10-11 09:58:58 +02:00
|
|
|
|
this._currentFocusWindow = null;
|
2018-05-25 11:35:49 +02:00
|
|
|
|
this._animFocusedWindow = null;
|
|
|
|
|
this._delayedAnimFocusWindow = null;
|
2017-01-16 16:21:01 +01:00
|
|
|
|
|
2018-02-15 12:20:51 +01:00
|
|
|
|
this._latched = false; // current level is latched
|
2017-10-09 12:51:08 +02:00
|
|
|
|
|
2017-12-06 13:46:02 +01:00
|
|
|
|
this._suggestions = null;
|
2019-04-26 10:23:28 +02:00
|
|
|
|
this._emojiKeyVisible = Meta.is_wayland_compositor();
|
2015-03-10 17:49:56 +01:00
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
this._focusTracker = new FocusTracker();
|
2019-08-28 23:52:52 +02:00
|
|
|
|
this._connectSignal(this._focusTracker, 'position-changed',
|
|
|
|
|
this._onFocusPositionChanged.bind(this));
|
|
|
|
|
this._connectSignal(this._focusTracker, 'reset', () => {
|
2018-05-25 11:35:49 +02:00
|
|
|
|
this._delayedAnimFocusWindow = null;
|
|
|
|
|
this._animFocusedWindow = null;
|
|
|
|
|
this._oskFocusWindow = null;
|
|
|
|
|
});
|
2019-08-28 23:52:52 +02:00
|
|
|
|
// Valid only for X11
|
|
|
|
|
if (!Meta.is_wayland_compositor()) {
|
|
|
|
|
this._connectSignal(this._focusTracker, 'focus-changed', (_tracker, focused) => {
|
|
|
|
|
if (focused)
|
|
|
|
|
this.open(Main.layoutManager.focusIndex);
|
|
|
|
|
else
|
|
|
|
|
this.close();
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2019-08-28 23:52:52 +02:00
|
|
|
|
}
|
2012-10-30 16:46:18 +01:00
|
|
|
|
|
|
|
|
|
this._showIdleId = 0;
|
2012-11-18 16:17:21 +01:00
|
|
|
|
|
|
|
|
|
this._keyboardVisible = false;
|
2019-08-28 23:52:52 +02:00
|
|
|
|
this._connectSignal(Main.layoutManager, 'keyboard-visible-changed', (_lm, visible) => {
|
2012-11-18 16:17:21 +01:00
|
|
|
|
this._keyboardVisible = visible;
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2012-11-18 16:17:21 +01:00
|
|
|
|
this._keyboardRequested = false;
|
|
|
|
|
this._keyboardRestingId = 0;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
this._connectSignal(Main.layoutManager, 'monitors-changed', this._relayout.bind(this));
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
this._setupKeyboard();
|
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.connect('destroy', this._onDestroy.bind(this));
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-01-16 16:21:01 +01:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
_connectSignal(obj, signal, callback) {
|
|
|
|
|
if (!this._connectionsIDs)
|
|
|
|
|
this._connectionsIDs = [];
|
2015-03-10 17:49:56 +01:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
let id = obj.connect(signal, callback);
|
|
|
|
|
this._connectionsIDs.push([obj, id]);
|
|
|
|
|
return id;
|
|
|
|
|
}
|
2015-03-10 17:49:56 +01:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
get visible() {
|
2019-07-16 11:24:13 +02:00
|
|
|
|
return this._keyboardVisible && super.visible;
|
2019-08-28 23:52:52 +02:00
|
|
|
|
}
|
2015-03-10 17:49:56 +01:00
|
|
|
|
|
2020-04-27 23:45:21 +02:00
|
|
|
|
set visible(visible) {
|
|
|
|
|
super.visible = visible;
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
_onFocusPositionChanged(focusTracker) {
|
|
|
|
|
let rect = focusTracker.getCurrentRect();
|
|
|
|
|
this.setCursorLocation(focusTracker.currentWindow, rect.x, rect.y, rect.width, rect.height);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2015-03-10 17:49:56 +01:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
_onDestroy() {
|
|
|
|
|
for (let [obj, id] of this._connectionsIDs)
|
|
|
|
|
obj.disconnect(id);
|
|
|
|
|
delete this._connectionsIDs;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
this._clearShowIdle();
|
2017-10-09 12:53:04 +02:00
|
|
|
|
|
2020-02-18 15:05:57 +01:00
|
|
|
|
this._keyboardController.destroy();
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
Main.layoutManager.untrackChrome(this);
|
|
|
|
|
Main.layoutManager.keyboardBox.remove_actor(this);
|
2018-01-23 16:45:46 +01:00
|
|
|
|
|
|
|
|
|
if (this._languagePopup) {
|
|
|
|
|
this._languagePopup.destroy();
|
|
|
|
|
this._languagePopup = null;
|
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_setupKeyboard() {
|
2019-07-16 11:24:13 +02:00
|
|
|
|
Main.layoutManager.keyboardBox.add_actor(this);
|
|
|
|
|
Main.layoutManager.trackChrome(this);
|
2011-09-01 11:01:48 -04:00
|
|
|
|
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._keyboardController = new KeyboardController();
|
|
|
|
|
|
2011-08-29 11:11:22 -04:00
|
|
|
|
this._groups = {};
|
2019-05-09 14:37:20 -05:00
|
|
|
|
this._currentPage = null;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-12-06 13:46:02 +01:00
|
|
|
|
this._suggestions = new Suggestions();
|
2019-10-21 20:44:00 +02:00
|
|
|
|
this.add_child(this._suggestions);
|
2017-12-06 13:46:02 +01:00
|
|
|
|
|
2020-01-21 15:00:18 +01:00
|
|
|
|
this._aspectContainer = new AspectContainer({
|
|
|
|
|
layout_manager: new Clutter.BinLayout(),
|
|
|
|
|
y_expand: true,
|
|
|
|
|
});
|
2019-10-21 20:44:00 +02:00
|
|
|
|
this.add_child(this._aspectContainer);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
this._emojiSelection = new EmojiSelection();
|
|
|
|
|
this._emojiSelection.connect('toggle', this._toggleEmoji.bind(this));
|
2019-08-28 23:06:14 +03:00
|
|
|
|
this._emojiSelection.connect('close-request', () => this.close());
|
2017-12-22 16:02:51 +01:00
|
|
|
|
this._emojiSelection.connect('emoji-selected', (selection, emoji) => {
|
|
|
|
|
this._keyboardController.commitString(emoji);
|
|
|
|
|
});
|
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this._aspectContainer.add_child(this._emojiSelection);
|
|
|
|
|
this._emojiSelection.hide();
|
2019-01-21 21:39:27 +01:00
|
|
|
|
|
2019-01-19 15:58:45 +01:00
|
|
|
|
this._keypad = new Keypad();
|
2019-08-28 23:52:52 +02:00
|
|
|
|
this._connectSignal(this._keypad, 'keyval', (_keypad, keyval) => {
|
2019-01-19 15:58:45 +01:00
|
|
|
|
this._keyboardController.keyvalPress(keyval);
|
|
|
|
|
this._keyboardController.keyvalRelease(keyval);
|
|
|
|
|
});
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this._aspectContainer.add_child(this._keypad);
|
|
|
|
|
this._keypad.hide();
|
2019-01-19 15:58:45 +01:00
|
|
|
|
this._keypadVisible = false;
|
|
|
|
|
|
2018-01-23 17:03:02 +01:00
|
|
|
|
this._ensureKeysForGroup(this._keyboardController.getCurrentGroup());
|
|
|
|
|
this._setActiveLayer(0);
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-08-05 12:21:30 +02:00
|
|
|
|
// Keyboard models are defined in LTR, we must override
|
|
|
|
|
// the locale setting in order to avoid flipping the
|
|
|
|
|
// keyboard on RTL locales.
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.text_direction = Clutter.TextDirection.LTR;
|
2012-03-14 12:21:33 +01:00
|
|
|
|
|
2019-08-28 23:52:52 +02:00
|
|
|
|
this._connectSignal(this._keyboardController, 'active-group',
|
|
|
|
|
this._onGroupChanged.bind(this));
|
|
|
|
|
this._connectSignal(this._keyboardController, 'groups-changed',
|
|
|
|
|
this._onKeyboardGroupsChanged.bind(this));
|
|
|
|
|
this._connectSignal(this._keyboardController, 'panel-state',
|
|
|
|
|
this._onKeyboardStateChanged.bind(this));
|
|
|
|
|
this._connectSignal(this._keyboardController, 'keypad-visible',
|
|
|
|
|
this._onKeypadVisible.bind(this));
|
|
|
|
|
this._connectSignal(global.stage, 'notify::key-focus',
|
|
|
|
|
this._onKeyFocusChanged.bind(this));
|
2017-12-20 13:08:50 +01:00
|
|
|
|
|
2019-08-20 02:51:42 +02:00
|
|
|
|
if (Meta.is_wayland_compositor()) {
|
2019-08-28 23:52:52 +02:00
|
|
|
|
this._connectSignal(this._keyboardController, 'emoji-visible',
|
|
|
|
|
this._onEmojiKeyVisible.bind(this));
|
2019-08-20 02:51:42 +02:00
|
|
|
|
}
|
2019-04-26 10:23:28 +02:00
|
|
|
|
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._relayout();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_onKeyFocusChanged() {
|
2011-08-29 11:11:22 -04:00
|
|
|
|
let focus = global.stage.key_focus;
|
|
|
|
|
|
2011-10-11 17:54:05 -04:00
|
|
|
|
// Showing an extended key popup and clicking a key from the extended keys
|
|
|
|
|
// will grab focus, but ignore that
|
2012-09-24 20:03:53 +02:00
|
|
|
|
let extendedKeysWereFocused = this._focusInExtendedKeys;
|
2019-12-04 18:51:08 +01:00
|
|
|
|
this._focusInExtendedKeys = focus && (focus._extendedKeys || focus.extendedKey);
|
2012-09-24 20:03:53 +02:00
|
|
|
|
if (this._focusInExtendedKeys || extendedKeysWereFocused)
|
2011-08-29 11:11:22 -04:00
|
|
|
|
return;
|
|
|
|
|
|
2012-10-30 16:46:18 +01:00
|
|
|
|
if (!(focus instanceof Clutter.Text)) {
|
2019-08-28 23:06:14 +03:00
|
|
|
|
this.close();
|
2012-10-30 16:46:18 +01:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-10 19:26:52 +02:00
|
|
|
|
if (!this._showIdleId) {
|
2019-01-29 20:36:54 +01:00
|
|
|
|
this._showIdleId = GLib.idle_add(GLib.PRIORITY_DEFAULT_IDLE, () => {
|
2019-08-28 23:06:14 +03:00
|
|
|
|
this.open(Main.layoutManager.focusIndex);
|
2019-01-29 20:36:54 +01:00
|
|
|
|
this._showIdleId = 0;
|
|
|
|
|
return GLib.SOURCE_REMOVE;
|
|
|
|
|
});
|
2019-08-28 23:52:52 +02:00
|
|
|
|
GLib.Source.set_name_by_id(this._showIdleId, '[gnome-shell] this.open');
|
2014-04-10 19:26:52 +02:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_createLayersForGroup(groupName) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let keyboardModel = new KeyboardModel(groupName);
|
2013-02-18 12:01:04 +09:00
|
|
|
|
let layers = {};
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let levels = keyboardModel.getLevels();
|
|
|
|
|
for (let i = 0; i < levels.length; i++) {
|
|
|
|
|
let currentLevel = levels[i];
|
|
|
|
|
/* There are keyboard maps which consist of 3 levels (no uppercase,
|
|
|
|
|
* basically). We however make things consistent by skipping that
|
|
|
|
|
* second level.
|
|
|
|
|
*/
|
2019-08-19 21:38:51 +02:00
|
|
|
|
let level = i >= 1 && levels.length == 3 ? i + 1 : i;
|
2017-12-20 13:08:50 +01:00
|
|
|
|
|
|
|
|
|
let layout = new KeyContainer();
|
2018-02-15 14:48:46 +01:00
|
|
|
|
layout.shiftKeys = [];
|
|
|
|
|
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._loadRows(currentLevel, level, levels.length, layout);
|
|
|
|
|
layers[level] = layout;
|
2019-01-21 21:39:27 +01:00
|
|
|
|
this._aspectContainer.add_child(layout);
|
|
|
|
|
layout.layoutButtons(this._aspectContainer);
|
2013-02-18 12:01:04 +09:00
|
|
|
|
|
|
|
|
|
layout.hide();
|
|
|
|
|
}
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
2013-02-18 12:01:04 +09:00
|
|
|
|
return layers;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2013-02-18 12:01:04 +09:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_ensureKeysForGroup(group) {
|
2018-01-23 17:03:02 +01:00
|
|
|
|
if (!this._groups[group])
|
|
|
|
|
this._groups[group] = this._createLayersForGroup(group);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_addRowKeys(keys, layout) {
|
2011-08-29 11:11:22 -04:00
|
|
|
|
for (let i = 0; i < keys.length; ++i) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let key = keys[i];
|
|
|
|
|
let button = new Key(key.shift(), key);
|
|
|
|
|
|
|
|
|
|
/* Space key gets special width, dependent on the number of surrounding keys */
|
|
|
|
|
if (button.key == ' ')
|
|
|
|
|
button.setWidth(keys.length <= 3 ? 5 : 3);
|
|
|
|
|
|
2017-10-31 01:38:18 +01:00
|
|
|
|
button.connect('pressed', (actor, keyval, str) => {
|
2017-12-05 20:05:18 +01:00
|
|
|
|
if (!Main.inputMethod.currentFocus ||
|
|
|
|
|
!this._keyboardController.commitString(str, true)) {
|
|
|
|
|
if (keyval != 0) {
|
|
|
|
|
this._keyboardController.keyvalPress(keyval);
|
|
|
|
|
button._keyvalPress = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2019-01-31 15:08:10 +01:00
|
|
|
|
button.connect('released', (actor, keyval, _str) => {
|
2017-12-05 20:05:18 +01:00
|
|
|
|
if (keyval != 0) {
|
|
|
|
|
if (button._keyvalPress)
|
|
|
|
|
this._keyboardController.keyvalRelease(keyval);
|
|
|
|
|
button._keyvalPress = false;
|
|
|
|
|
}
|
2018-02-15 12:20:51 +01:00
|
|
|
|
|
|
|
|
|
if (!this._latched)
|
|
|
|
|
this._setActiveLayer(0);
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
layout.appendKey(button, button.keyButton.keyWidth);
|
2017-08-05 12:21:30 +02:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2012-11-10 22:47:57 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_popupLanguageMenu(keyActor) {
|
2018-01-23 16:45:46 +01:00
|
|
|
|
if (this._languagePopup)
|
|
|
|
|
this._languagePopup.destroy();
|
|
|
|
|
|
|
|
|
|
this._languagePopup = new LanguageSelectionPopup(keyActor);
|
2019-06-19 00:24:01 +02:00
|
|
|
|
Main.layoutManager.addTopChrome(this._languagePopup.actor);
|
2018-01-23 16:45:46 +01:00
|
|
|
|
this._languagePopup.open(true);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-01-23 16:45:46 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_loadDefaultKeys(keys, layout, numLevels, numKeys) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let extraButton;
|
|
|
|
|
for (let i = 0; i < keys.length; i++) {
|
|
|
|
|
let key = keys[i];
|
|
|
|
|
let keyval = key.keyval;
|
|
|
|
|
let switchToLevel = key.level;
|
|
|
|
|
let action = key.action;
|
2020-02-14 18:09:32 +01:00
|
|
|
|
let icon = key.icon;
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
/* Skip emoji button if necessary */
|
|
|
|
|
if (!this._emojiKeyVisible && action == 'emoji')
|
|
|
|
|
continue;
|
|
|
|
|
|
2020-02-14 18:09:32 +01:00
|
|
|
|
extraButton = new Key(key.label || '', [], icon);
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2018-02-11 16:19:10 +01:00
|
|
|
|
extraButton.keyButton.add_style_class_name('default-key');
|
2017-08-05 12:21:30 +02:00
|
|
|
|
if (key.extraClassName != null)
|
2018-02-11 16:19:10 +01:00
|
|
|
|
extraButton.keyButton.add_style_class_name(key.extraClassName);
|
2017-08-05 12:21:30 +02:00
|
|
|
|
if (key.width != null)
|
|
|
|
|
extraButton.setWidth(key.width);
|
|
|
|
|
|
2018-02-11 16:19:10 +01:00
|
|
|
|
let actor = extraButton.keyButton;
|
2018-01-23 16:45:46 +01:00
|
|
|
|
|
2017-10-31 01:38:18 +01:00
|
|
|
|
extraButton.connect('pressed', () => {
|
2018-02-15 12:20:51 +01:00
|
|
|
|
if (switchToLevel != null) {
|
|
|
|
|
this._setActiveLayer(switchToLevel);
|
|
|
|
|
// Shift only gets latched on long press
|
2019-08-19 21:38:51 +02:00
|
|
|
|
this._latched = switchToLevel != 1;
|
2018-02-15 12:20:51 +01:00
|
|
|
|
} else if (keyval != null) {
|
2018-02-14 12:39:27 +01:00
|
|
|
|
this._keyboardController.keyvalPress(keyval);
|
2018-02-15 12:20:51 +01:00
|
|
|
|
}
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
|
|
|
|
extraButton.connect('released', () => {
|
2018-02-15 12:20:51 +01:00
|
|
|
|
if (keyval != null)
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._keyboardController.keyvalRelease(keyval);
|
|
|
|
|
else if (action == 'hide')
|
2019-08-28 23:06:14 +03:00
|
|
|
|
this.close();
|
2018-01-23 16:45:46 +01:00
|
|
|
|
else if (action == 'languageMenu')
|
|
|
|
|
this._popupLanguageMenu(actor);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
else if (action == 'emoji')
|
|
|
|
|
this._toggleEmoji();
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2018-02-15 14:48:46 +01:00
|
|
|
|
if (switchToLevel == 0) {
|
|
|
|
|
layout.shiftKeys.push(extraButton);
|
|
|
|
|
} else if (switchToLevel == 1) {
|
2017-10-31 01:38:18 +01:00
|
|
|
|
extraButton.connect('long-press', () => {
|
2018-02-15 12:20:51 +01:00
|
|
|
|
this._latched = true;
|
2019-05-09 14:37:20 -05:00
|
|
|
|
this._setCurrentLevelLatched(this._currentPage, this._latched);
|
2017-10-31 01:38:18 +01:00
|
|
|
|
});
|
2018-02-15 12:20:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-05 12:21:30 +02:00
|
|
|
|
/* Fixup default keys based on the number of levels/keys */
|
2018-02-15 13:23:54 +01:00
|
|
|
|
if (switchToLevel == 1 && numLevels == 3) {
|
|
|
|
|
// Hide shift key if the keymap has no uppercase level
|
2017-08-05 12:21:30 +02:00
|
|
|
|
if (key.right) {
|
|
|
|
|
/* Only hide the key actor, so the container still takes space */
|
2018-02-11 16:19:10 +01:00
|
|
|
|
extraButton.keyButton.hide();
|
2017-08-05 12:21:30 +02:00
|
|
|
|
} else {
|
2019-07-16 11:24:13 +02:00
|
|
|
|
extraButton.hide();
|
2017-08-05 12:21:30 +02:00
|
|
|
|
}
|
|
|
|
|
extraButton.setWidth(1.5);
|
|
|
|
|
} else if (key.right && numKeys > 8) {
|
|
|
|
|
extraButton.setWidth(2);
|
2018-02-15 13:23:54 +01:00
|
|
|
|
} else if (keyval == Clutter.KEY_Return && numKeys > 9) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
extraButton.setWidth(1.5);
|
2017-12-22 16:02:51 +01:00
|
|
|
|
} else if (!this._emojiKeyVisible && (action == 'hide' || action == 'languageMenu')) {
|
|
|
|
|
extraButton.setWidth(1.5);
|
2011-08-29 11:11:22 -04:00
|
|
|
|
}
|
2017-12-20 13:08:50 +01:00
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
layout.appendKey(extraButton, extraButton.keyButton.keyWidth);
|
2011-08-29 11:11:22 -04:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-01-19 15:58:45 +01:00
|
|
|
|
_updateCurrentPageVisible() {
|
2019-05-09 14:37:20 -05:00
|
|
|
|
if (this._currentPage)
|
|
|
|
|
this._currentPage.visible = !this._emojiActive && !this._keypadVisible;
|
2019-01-19 15:58:45 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
_setEmojiActive(active) {
|
|
|
|
|
this._emojiActive = active;
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this._emojiSelection.visible = this._emojiActive;
|
2019-01-19 15:58:45 +01:00
|
|
|
|
this._updateCurrentPageVisible();
|
2017-12-22 16:02:51 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_toggleEmoji() {
|
|
|
|
|
this._setEmojiActive(!this._emojiActive);
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_setCurrentLevelLatched(layout, latched) {
|
2019-01-21 21:41:46 +01:00
|
|
|
|
for (let i = 0; i < layout.shiftKeys.length; i++) {
|
2018-02-15 14:48:46 +01:00
|
|
|
|
let key = layout.shiftKeys[i];
|
|
|
|
|
key.setLatched(latched);
|
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2018-02-15 14:48:46 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_getDefaultKeysForRow(row, numRows, level) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
/* The first 2 rows in defaultKeysPre/Post belong together with
|
|
|
|
|
* the first 2 rows on each keymap. On keymaps that have more than
|
|
|
|
|
* 4 rows, the last 2 default key rows must be respectively
|
|
|
|
|
* assigned to the 2 last keymap ones.
|
|
|
|
|
*/
|
|
|
|
|
if (row < 2) {
|
|
|
|
|
return [defaultKeysPre[level][row], defaultKeysPost[level][row]];
|
|
|
|
|
} else if (row >= numRows - 2) {
|
|
|
|
|
let defaultRow = row - (numRows - 2) + 2;
|
|
|
|
|
return [defaultKeysPre[level][defaultRow], defaultKeysPost[level][defaultRow]];
|
|
|
|
|
} else {
|
|
|
|
|
return [null, null];
|
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_mergeRowKeys(layout, pre, row, post, numLevels) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
if (pre != null)
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._loadDefaultKeys(pre, layout, numLevels, row.length);
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._addRowKeys(row, layout);
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
|
|
|
|
if (post != null)
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._loadDefaultKeys(post, layout, numLevels, row.length);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_loadRows(model, level, numLevels, layout) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let rows = model.rows;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
for (let i = 0; i < rows.length; ++i) {
|
2017-12-20 13:08:50 +01:00
|
|
|
|
layout.appendRow();
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let [pre, post] = this._getDefaultKeysForRow(i, rows.length, level);
|
2019-08-19 19:55:49 +02:00
|
|
|
|
this._mergeRowKeys(layout, pre, rows[i], post, numLevels);
|
2017-12-05 22:41:32 +01:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-05 22:41:32 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_getGridSlots() {
|
2017-12-05 22:41:32 +01:00
|
|
|
|
let numOfHorizSlots = 0, numOfVertSlots;
|
2019-05-09 14:37:20 -05:00
|
|
|
|
let rows = this._currentPage.get_children();
|
2017-12-05 22:41:32 +01:00
|
|
|
|
numOfVertSlots = rows.length;
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-12-05 22:41:32 +01:00
|
|
|
|
for (let i = 0; i < rows.length; ++i) {
|
2019-01-31 14:43:52 +01:00
|
|
|
|
let keyboardRow = rows[i];
|
|
|
|
|
let keys = keyboardRow.get_children();
|
2017-12-05 22:41:32 +01:00
|
|
|
|
|
|
|
|
|
numOfHorizSlots = Math.max(numOfHorizSlots, keys.length);
|
2011-08-29 11:11:22 -04:00
|
|
|
|
}
|
|
|
|
|
|
2017-12-05 22:41:32 +01:00
|
|
|
|
return [numOfHorizSlots, numOfVertSlots];
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_relayout() {
|
2012-10-14 18:57:45 +02:00
|
|
|
|
let monitor = Main.layoutManager.keyboardMonitor;
|
2017-10-11 12:59:19 +02:00
|
|
|
|
|
2019-07-16 11:24:13 +02:00
|
|
|
|
if (!monitor)
|
2017-10-11 12:59:19 +02:00
|
|
|
|
return;
|
|
|
|
|
|
2011-08-29 11:11:22 -04:00
|
|
|
|
let maxHeight = monitor.height / 3;
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this.width = monitor.width;
|
2020-01-21 17:01:56 +01:00
|
|
|
|
|
|
|
|
|
if (monitor.width > monitor.height) {
|
|
|
|
|
this.height = maxHeight;
|
|
|
|
|
} else {
|
|
|
|
|
/* In portrait mode, lack of horizontal space means we won't be
|
|
|
|
|
* able to make the OSK that big while keeping size ratio, so
|
|
|
|
|
* we allow the OSK being smaller than 1/3rd of the monitor height
|
|
|
|
|
* there.
|
|
|
|
|
*/
|
2020-07-04 02:36:10 +02:00
|
|
|
|
const forWidth = this.get_theme_node().adjust_for_width(monitor.width);
|
|
|
|
|
const [, natHeight] = this.get_preferred_height(forWidth);
|
2020-07-04 02:22:10 +02:00
|
|
|
|
this.height = Math.min(maxHeight, natHeight);
|
2020-01-21 17:01:56 +01:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_onGroupChanged() {
|
2018-01-23 17:03:02 +01:00
|
|
|
|
this._ensureKeysForGroup(this._keyboardController.getCurrentGroup());
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._setActiveLayer(0);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-02-04 12:30:53 +01:00
|
|
|
|
_onKeyboardGroupsChanged() {
|
2019-07-16 11:24:13 +02:00
|
|
|
|
let nonGroupActors = [this._emojiSelection, this._keypad];
|
2019-04-30 14:08:25 +02:00
|
|
|
|
this._aspectContainer.get_children().filter(c => !nonGroupActors.includes(c)).forEach(c => {
|
|
|
|
|
c.destroy();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
this._groups = {};
|
2018-02-11 16:15:14 +01:00
|
|
|
|
this._onGroupChanged();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2013-02-18 12:01:04 +09:00
|
|
|
|
|
2019-01-19 15:58:45 +01:00
|
|
|
|
_onKeypadVisible(controller, visible) {
|
|
|
|
|
if (visible == this._keypadVisible)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
this._keypadVisible = visible;
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this._keypad.visible = this._keypadVisible;
|
2019-01-19 15:58:45 +01:00
|
|
|
|
this._updateCurrentPageVisible();
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
_onEmojiKeyVisible(controller, visible) {
|
|
|
|
|
if (visible == this._emojiKeyVisible)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
this._emojiKeyVisible = visible;
|
|
|
|
|
/* Rebuild keyboard widgetry to include emoji button */
|
|
|
|
|
this._onKeyboardGroupsChanged();
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_onKeyboardStateChanged(controller, state) {
|
2017-12-05 20:05:18 +01:00
|
|
|
|
let enabled;
|
|
|
|
|
if (state == Clutter.InputPanelState.OFF)
|
|
|
|
|
enabled = false;
|
|
|
|
|
else if (state == Clutter.InputPanelState.ON)
|
|
|
|
|
enabled = true;
|
|
|
|
|
else if (state == Clutter.InputPanelState.TOGGLE)
|
2019-08-19 21:38:51 +02:00
|
|
|
|
enabled = this._keyboardVisible == false;
|
2017-12-05 20:05:18 +01:00
|
|
|
|
else
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (enabled)
|
2019-08-28 23:06:14 +03:00
|
|
|
|
this.open(Main.layoutManager.focusIndex);
|
2017-12-05 20:05:18 +01:00
|
|
|
|
else
|
2019-08-28 23:06:14 +03:00
|
|
|
|
this.close();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-05 20:05:18 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_setActiveLayer(activeLevel) {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
let activeGroupName = this._keyboardController.getCurrentGroup();
|
|
|
|
|
let layers = this._groups[activeGroupName];
|
2019-05-09 14:35:11 -05:00
|
|
|
|
let currentPage = layers[activeLevel];
|
|
|
|
|
|
2019-05-09 14:37:20 -05:00
|
|
|
|
if (this._currentPage == currentPage) {
|
2019-05-09 14:35:11 -05:00
|
|
|
|
this._updateCurrentPageVisible();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-05-09 14:37:20 -05:00
|
|
|
|
if (this._currentPage != null) {
|
|
|
|
|
this._setCurrentLevelLatched(this._currentPage, false);
|
|
|
|
|
this._currentPage.disconnect(this._currentPage._destroyID);
|
|
|
|
|
this._currentPage.hide();
|
|
|
|
|
delete this._currentPage._destroyID;
|
2011-08-29 11:11:22 -04:00
|
|
|
|
}
|
|
|
|
|
|
2019-05-09 14:37:20 -05:00
|
|
|
|
this._currentPage = currentPage;
|
|
|
|
|
this._currentPage._destroyID = this._currentPage.connect('destroy', () => {
|
|
|
|
|
this._currentPage = null;
|
2019-05-09 14:30:21 -05:00
|
|
|
|
});
|
2019-01-19 15:58:45 +01:00
|
|
|
|
this._updateCurrentPageVisible();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_clearKeyboardRestTimer() {
|
2012-11-18 16:17:21 +01:00
|
|
|
|
if (!this._keyboardRestingId)
|
|
|
|
|
return;
|
|
|
|
|
GLib.source_remove(this._keyboardRestingId);
|
|
|
|
|
this._keyboardRestingId = 0;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2012-11-18 16:17:21 +01:00
|
|
|
|
|
2019-08-28 23:06:14 +03:00
|
|
|
|
open(monitor) {
|
2017-01-16 16:36:21 +01:00
|
|
|
|
this._clearShowIdle();
|
2012-11-18 16:17:21 +01:00
|
|
|
|
this._keyboardRequested = true;
|
|
|
|
|
|
|
|
|
|
if (this._keyboardVisible) {
|
|
|
|
|
if (monitor != Main.layoutManager.keyboardIndex) {
|
|
|
|
|
Main.layoutManager.keyboardIndex = monitor;
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._relayout();
|
2012-11-18 16:17:21 +01:00
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this._clearKeyboardRestTimer();
|
|
|
|
|
this._keyboardRestingId = GLib.timeout_add(GLib.PRIORITY_DEFAULT,
|
2019-12-04 20:40:07 +01:00
|
|
|
|
KEYBOARD_REST_TIME,
|
|
|
|
|
() => {
|
|
|
|
|
this._clearKeyboardRestTimer();
|
|
|
|
|
this._open(monitor);
|
|
|
|
|
return GLib.SOURCE_REMOVE;
|
|
|
|
|
});
|
2014-04-10 19:26:52 +02:00
|
|
|
|
GLib.Source.set_name_by_id(this._keyboardRestingId, '[gnome-shell] this._clearKeyboardRestTimer');
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2012-11-18 16:17:21 +01:00
|
|
|
|
|
2019-08-28 23:06:14 +03:00
|
|
|
|
_open(monitor) {
|
2012-11-18 16:17:21 +01:00
|
|
|
|
if (!this._keyboardRequested)
|
|
|
|
|
return;
|
|
|
|
|
|
2012-10-14 18:57:45 +02:00
|
|
|
|
Main.layoutManager.keyboardIndex = monitor;
|
2017-12-20 13:08:50 +01:00
|
|
|
|
this._relayout();
|
2011-08-29 11:11:22 -04:00
|
|
|
|
Main.layoutManager.showKeyboard();
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
2017-12-22 16:02:51 +01:00
|
|
|
|
this._setEmojiActive(false);
|
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
if (this._delayedAnimFocusWindow) {
|
|
|
|
|
this._setAnimationWindow(this._delayedAnimFocusWindow);
|
|
|
|
|
this._delayedAnimFocusWindow = null;
|
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2019-08-28 23:06:14 +03:00
|
|
|
|
close() {
|
2017-01-16 16:36:21 +01:00
|
|
|
|
this._clearShowIdle();
|
2012-11-18 16:17:21 +01:00
|
|
|
|
this._keyboardRequested = false;
|
|
|
|
|
|
|
|
|
|
if (!this._keyboardVisible)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
this._clearKeyboardRestTimer();
|
|
|
|
|
this._keyboardRestingId = GLib.timeout_add(GLib.PRIORITY_DEFAULT,
|
2019-12-04 20:40:07 +01:00
|
|
|
|
KEYBOARD_REST_TIME,
|
|
|
|
|
() => {
|
|
|
|
|
this._clearKeyboardRestTimer();
|
|
|
|
|
this._close();
|
|
|
|
|
return GLib.SOURCE_REMOVE;
|
|
|
|
|
});
|
2014-04-10 19:26:52 +02:00
|
|
|
|
GLib.Source.set_name_by_id(this._keyboardRestingId, '[gnome-shell] this._clearKeyboardRestTimer');
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2012-11-18 16:17:21 +01:00
|
|
|
|
|
2019-08-28 23:06:14 +03:00
|
|
|
|
_close() {
|
2012-11-18 16:17:21 +01:00
|
|
|
|
if (this._keyboardRequested)
|
|
|
|
|
return;
|
|
|
|
|
|
2011-08-29 11:11:22 -04:00
|
|
|
|
Main.layoutManager.hideKeyboard();
|
2017-10-11 09:58:58 +02:00
|
|
|
|
this.setCursorLocation(null);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
resetSuggestions() {
|
2017-12-06 13:46:02 +01:00
|
|
|
|
if (this._suggestions)
|
|
|
|
|
this._suggestions.clear();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-06 13:46:02 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
addSuggestion(text, callback) {
|
2017-12-06 13:46:02 +01:00
|
|
|
|
if (!this._suggestions)
|
|
|
|
|
return;
|
|
|
|
|
this._suggestions.add(text, callback);
|
2019-07-16 11:24:13 +02:00
|
|
|
|
this._suggestions.show();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-06 13:46:02 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_clearShowIdle() {
|
2012-10-30 16:46:18 +01:00
|
|
|
|
if (!this._showIdleId)
|
|
|
|
|
return;
|
|
|
|
|
GLib.source_remove(this._showIdleId);
|
|
|
|
|
this._showIdleId = 0;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2012-10-30 16:46:18 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_windowSlideAnimationComplete(window, delta) {
|
2019-01-31 15:04:56 +01:00
|
|
|
|
// Synchronize window positions again.
|
2017-10-11 09:58:58 +02:00
|
|
|
|
let frameRect = window.get_frame_rect();
|
|
|
|
|
frameRect.y += delta;
|
|
|
|
|
window.move_frame(true, frameRect.x, frameRect.y);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-10-11 09:58:58 +02:00
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
_animateWindow(window, show) {
|
2017-10-11 09:58:58 +02:00
|
|
|
|
let windowActor = window.get_compositor_private();
|
2018-05-25 11:35:49 +02:00
|
|
|
|
let deltaY = Main.layoutManager.keyboardBox.height;
|
2017-10-11 09:58:58 +02:00
|
|
|
|
if (!windowActor)
|
2011-09-27 14:55:38 -04:00
|
|
|
|
return;
|
|
|
|
|
|
2017-10-11 09:58:58 +02:00
|
|
|
|
if (show) {
|
2018-07-20 21:46:19 +02:00
|
|
|
|
windowActor.ease({
|
|
|
|
|
y: windowActor.y - deltaY,
|
|
|
|
|
duration: Layout.KEYBOARD_ANIMATION_TIME,
|
|
|
|
|
mode: Clutter.AnimationMode.EASE_OUT_QUAD,
|
|
|
|
|
onComplete: () => {
|
|
|
|
|
this._windowSlideAnimationComplete(window, -deltaY);
|
2019-08-20 23:43:54 +02:00
|
|
|
|
},
|
2018-07-20 21:46:19 +02:00
|
|
|
|
});
|
2017-10-11 09:58:58 +02:00
|
|
|
|
} else {
|
2018-07-20 21:46:19 +02:00
|
|
|
|
windowActor.ease({
|
|
|
|
|
y: windowActor.y + deltaY,
|
|
|
|
|
duration: Layout.KEYBOARD_ANIMATION_TIME,
|
|
|
|
|
mode: Clutter.AnimationMode.EASE_IN_QUAD,
|
|
|
|
|
onComplete: () => {
|
|
|
|
|
this._windowSlideAnimationComplete(window, deltaY);
|
2019-08-20 23:43:54 +02:00
|
|
|
|
},
|
2018-07-20 21:46:19 +02:00
|
|
|
|
});
|
2017-10-11 09:58:58 +02:00
|
|
|
|
}
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2011-08-29 11:11:22 -04:00
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
_setAnimationWindow(window) {
|
|
|
|
|
if (this._animFocusedWindow == window)
|
2011-09-27 14:55:38 -04:00
|
|
|
|
return;
|
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
if (this._animFocusedWindow)
|
|
|
|
|
this._animateWindow(this._animFocusedWindow, false);
|
|
|
|
|
if (window)
|
|
|
|
|
this._animateWindow(window, true);
|
2017-10-11 09:58:58 +02:00
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
this._animFocusedWindow = window;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-10-11 09:58:58 +02:00
|
|
|
|
|
2019-01-29 02:27:05 +01:00
|
|
|
|
setCursorLocation(window, x, y, w, h) {
|
2018-05-25 11:35:49 +02:00
|
|
|
|
let monitor = Main.layoutManager.keyboardMonitor;
|
2017-10-11 09:58:58 +02:00
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
if (window && monitor) {
|
2017-10-11 09:58:58 +02:00
|
|
|
|
let keyboardHeight = Main.layoutManager.keyboardBox.height;
|
|
|
|
|
|
2018-05-25 11:35:49 +02:00
|
|
|
|
if (y + h >= monitor.y + monitor.height - keyboardHeight) {
|
|
|
|
|
if (this._keyboardVisible)
|
|
|
|
|
this._setAnimationWindow(window);
|
|
|
|
|
else
|
|
|
|
|
this._delayedAnimFocusWindow = window;
|
|
|
|
|
} else if (y < keyboardHeight) {
|
|
|
|
|
this._delayedAnimFocusWindow = null;
|
|
|
|
|
this._setAnimationWindow(null);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
this._setAnimationWindow(null);
|
2017-10-11 09:58:58 +02:00
|
|
|
|
}
|
2018-05-25 11:35:49 +02:00
|
|
|
|
|
|
|
|
|
this._oskFocusWindow = window;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2019-07-16 11:24:13 +02:00
|
|
|
|
});
|
2015-04-03 00:18:07 +02:00
|
|
|
|
|
2017-10-31 02:19:44 +01:00
|
|
|
|
var KeyboardController = class {
|
|
|
|
|
constructor() {
|
2019-10-05 12:35:21 +02:00
|
|
|
|
let seat = Clutter.get_default_backend().get_default_seat();
|
|
|
|
|
this._virtualDevice = seat.create_virtual_device(Clutter.InputDeviceType.KEYBOARD_DEVICE);
|
2016-07-20 17:50:55 +02:00
|
|
|
|
|
|
|
|
|
this._inputSourceManager = InputSourceManager.getInputSourceManager();
|
|
|
|
|
this._sourceChangedId = this._inputSourceManager.connect('current-source-changed',
|
2017-12-02 01:27:35 +01:00
|
|
|
|
this._onSourceChanged.bind(this));
|
2019-08-19 19:55:49 +02:00
|
|
|
|
this._sourcesModifiedId = this._inputSourceManager.connect('sources-changed',
|
|
|
|
|
this._onSourcesModified.bind(this));
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._currentSource = this._inputSourceManager.currentSource;
|
2017-12-05 20:05:18 +01:00
|
|
|
|
|
2020-02-18 15:05:57 +01:00
|
|
|
|
this._notifyContentPurposeId = Main.inputMethod.connect(
|
|
|
|
|
'notify::content-purpose', this._onContentPurposeHintsChanged.bind(this));
|
|
|
|
|
this._notifyContentHintsId = Main.inputMethod.connect(
|
|
|
|
|
'notify::content-hints', this._onContentPurposeHintsChanged.bind(this));
|
|
|
|
|
this._notifyInputPanelStateId = Main.inputMethod.connect(
|
|
|
|
|
'input-panel-state', (o, state) => this.emit('panel-state', state));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
destroy() {
|
|
|
|
|
this._inputSourceManager.disconnect(this._sourceChangedId);
|
|
|
|
|
this._inputSourceManager.disconnect(this._sourcesModifiedId);
|
|
|
|
|
Main.inputMethod.disconnect(this._notifyContentPurposeId);
|
|
|
|
|
Main.inputMethod.disconnect(this._notifyContentHintsId);
|
|
|
|
|
Main.inputMethod.disconnect(this._notifyInputPanelStateId);
|
2020-03-28 11:54:32 +01:00
|
|
|
|
|
|
|
|
|
// Make sure any buttons pressed by the virtual device are released
|
|
|
|
|
// immediately instead of waiting for the next GC cycle
|
|
|
|
|
this._virtualDevice.run_dispose();
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2016-07-20 17:50:55 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_onSourcesModified() {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this.emit('groups-changed');
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2016-07-20 17:50:55 +02:00
|
|
|
|
|
2019-01-31 15:08:10 +01:00
|
|
|
|
_onSourceChanged(inputSourceManager, _oldSource) {
|
2016-07-20 17:50:55 +02:00
|
|
|
|
let source = inputSourceManager.currentSource;
|
2017-08-05 12:21:30 +02:00
|
|
|
|
this._currentSource = source;
|
|
|
|
|
this.emit('active-group', source.id);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2016-07-20 17:50:55 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
_onContentPurposeHintsChanged(method) {
|
2017-12-05 20:05:18 +01:00
|
|
|
|
let purpose = method.content_purpose;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
let emojiVisible = false;
|
2019-01-19 15:58:45 +01:00
|
|
|
|
let keypadVisible = false;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
|
|
|
|
if (purpose == Clutter.InputContentPurpose.NORMAL ||
|
|
|
|
|
purpose == Clutter.InputContentPurpose.ALPHA ||
|
|
|
|
|
purpose == Clutter.InputContentPurpose.PASSWORD ||
|
|
|
|
|
purpose == Clutter.InputContentPurpose.TERMINAL)
|
|
|
|
|
emojiVisible = true;
|
2019-01-19 15:58:45 +01:00
|
|
|
|
if (purpose == Clutter.InputContentPurpose.DIGITS ||
|
|
|
|
|
purpose == Clutter.InputContentPurpose.NUMBER ||
|
|
|
|
|
purpose == Clutter.InputContentPurpose.PHONE)
|
|
|
|
|
keypadVisible = true;
|
2017-12-22 16:02:51 +01:00
|
|
|
|
|
2019-01-29 02:18:52 +01:00
|
|
|
|
this.emit('emoji-visible', emojiVisible);
|
2019-01-19 15:58:45 +01:00
|
|
|
|
this.emit('keypad-visible', keypadVisible);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-05 20:05:18 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
getGroups() {
|
2016-07-20 17:50:55 +02:00
|
|
|
|
let inputSources = this._inputSourceManager.inputSources;
|
2019-01-29 02:18:52 +01:00
|
|
|
|
let groups = [];
|
2016-07-20 17:50:55 +02:00
|
|
|
|
|
|
|
|
|
for (let i in inputSources) {
|
|
|
|
|
let is = inputSources[i];
|
2017-08-05 12:21:30 +02:00
|
|
|
|
groups[is.index] = is.xkbId;
|
2016-07-20 17:50:55 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-08-05 12:21:30 +02:00
|
|
|
|
return groups;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-08-05 12:21:30 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
getCurrentGroup() {
|
2017-08-05 12:21:30 +02:00
|
|
|
|
return this._currentSource.xkbId;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2016-07-20 17:50:55 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
commitString(string, fromKey) {
|
2017-12-05 20:05:18 +01:00
|
|
|
|
if (string == null)
|
|
|
|
|
return false;
|
|
|
|
|
/* Let ibus methods fall through keyval emission */
|
|
|
|
|
if (fromKey && this._currentSource.type == InputSourceManager.INPUT_SOURCE_TYPE_IBUS)
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
Main.inputMethod.commit(string);
|
|
|
|
|
return true;
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2017-12-05 20:05:18 +01:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
keyvalPress(keyval) {
|
2016-07-20 17:50:55 +02:00
|
|
|
|
this._virtualDevice.notify_keyval(Clutter.get_current_event_time(),
|
|
|
|
|
keyval, Clutter.KeyState.PRESSED);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
2015-04-03 00:18:07 +02:00
|
|
|
|
|
2017-10-31 01:03:21 +01:00
|
|
|
|
keyvalRelease(keyval) {
|
2016-07-20 17:50:55 +02:00
|
|
|
|
this._virtualDevice.notify_keyval(Clutter.get_current_event_time(),
|
|
|
|
|
keyval, Clutter.KeyState.RELEASED);
|
2017-10-31 02:19:44 +01:00
|
|
|
|
}
|
|
|
|
|
};
|
2017-08-05 12:21:30 +02:00
|
|
|
|
Signals.addSignalMethods(KeyboardController.prototype);
|