2013-02-18 15:55:30 -05:00
|
|
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
2019-12-13 03:06:14 -05:00
|
|
|
/* exported addContextMenu CapsLockWarning */
|
2013-02-18 15:55:30 -05:00
|
|
|
|
2019-12-13 03:06:14 -05:00
|
|
|
const { Clutter, GObject, Pango, Shell, St } = imports.gi;
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2013-02-18 15:55:44 -05:00
|
|
|
const BoxPointer = imports.ui.boxpointer;
|
2011-09-14 12:47:09 -04:00
|
|
|
const Main = imports.ui.main;
|
|
|
|
const Params = imports.misc.params;
|
|
|
|
const PopupMenu = imports.ui.popupMenu;
|
|
|
|
|
2017-10-30 21:19:44 -04:00
|
|
|
var EntryMenu = class extends PopupMenu.PopupMenu {
|
|
|
|
constructor(entry) {
|
|
|
|
super(entry, 0, St.Side.TOP);
|
2011-09-14 12:47:09 -04:00
|
|
|
|
|
|
|
this._entry = entry;
|
|
|
|
this._clipboard = St.Clipboard.get_default();
|
|
|
|
|
|
|
|
// Populate menu
|
|
|
|
let item;
|
|
|
|
item = new PopupMenu.PopupMenuItem(_("Copy"));
|
2017-12-01 19:27:35 -05:00
|
|
|
item.connect('activate', this._onCopyActivated.bind(this));
|
2011-09-14 12:47:09 -04:00
|
|
|
this.addMenuItem(item);
|
|
|
|
this._copyItem = item;
|
|
|
|
|
|
|
|
item = new PopupMenu.PopupMenuItem(_("Paste"));
|
2017-12-01 19:27:35 -05:00
|
|
|
item.connect('activate', this._onPasteActivated.bind(this));
|
2011-09-14 12:47:09 -04:00
|
|
|
this.addMenuItem(item);
|
|
|
|
this._pasteItem = item;
|
|
|
|
|
2019-12-12 04:08:42 -05:00
|
|
|
if (entry instanceof St.PasswordEntry)
|
|
|
|
this._makePasswordItem();
|
2011-09-14 12:47:09 -04:00
|
|
|
|
|
|
|
Main.uiGroup.add_actor(this.actor);
|
|
|
|
this.actor.hide();
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_makePasswordItem() {
|
2012-05-27 13:06:08 -04:00
|
|
|
let item = new PopupMenu.PopupMenuItem('');
|
2017-12-01 19:27:35 -05:00
|
|
|
item.connect('activate', this._onPasswordActivated.bind(this));
|
2012-05-27 13:06:08 -04:00
|
|
|
this.addMenuItem(item);
|
|
|
|
this._passwordItem = item;
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2012-05-27 13:06:08 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
open(animate) {
|
2011-09-14 12:47:09 -04:00
|
|
|
this._updatePasteItem();
|
|
|
|
this._updateCopyItem();
|
|
|
|
if (this._passwordItem)
|
|
|
|
this._updatePasswordItem();
|
|
|
|
|
2017-10-30 21:19:44 -04:00
|
|
|
super.open(animate);
|
2013-02-18 16:06:53 -05:00
|
|
|
this._entry.add_style_pseudo_class('focus');
|
|
|
|
|
2018-11-27 07:58:25 -05:00
|
|
|
let direction = St.DirectionType.TAB_FORWARD;
|
2011-09-14 12:47:09 -04:00
|
|
|
if (!this.actor.navigate_focus(null, direction, false))
|
|
|
|
this.actor.grab_key_focus();
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2012-09-06 15:35:54 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_updateCopyItem() {
|
2011-09-14 12:47:09 -04:00
|
|
|
let selection = this._entry.clutter_text.get_selection();
|
2013-03-04 10:22:16 -05:00
|
|
|
this._copyItem.setSensitive(!this._entry.clutter_text.password_char &&
|
|
|
|
selection && selection != '');
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_updatePasteItem() {
|
2017-10-30 20:38:18 -04:00
|
|
|
this._clipboard.get_text(St.ClipboardType.CLIPBOARD,
|
|
|
|
(clipboard, text) => {
|
2011-09-14 12:47:09 -04:00
|
|
|
this._pasteItem.setSensitive(text && text != '');
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_updatePasswordItem() {
|
2019-12-12 04:08:42 -05:00
|
|
|
if (!this._entry.password_visible)
|
2011-09-14 12:47:09 -04:00
|
|
|
this._passwordItem.label.set_text(_("Show Text"));
|
|
|
|
else
|
|
|
|
this._passwordItem.label.set_text(_("Hide Text"));
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_onCopyActivated() {
|
2011-09-14 12:47:09 -04:00
|
|
|
let selection = this._entry.clutter_text.get_selection();
|
2011-11-09 11:19:47 -05:00
|
|
|
this._clipboard.set_text(St.ClipboardType.CLIPBOARD, selection);
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_onPasteActivated() {
|
2017-10-30 20:38:18 -04:00
|
|
|
this._clipboard.get_text(St.ClipboardType.CLIPBOARD,
|
|
|
|
(clipboard, text) => {
|
2011-09-14 12:47:09 -04:00
|
|
|
if (!text)
|
|
|
|
return;
|
|
|
|
this._entry.clutter_text.delete_selection();
|
|
|
|
let pos = this._entry.clutter_text.get_cursor_position();
|
|
|
|
this._entry.clutter_text.insert_text(text, pos);
|
2017-10-30 20:38:18 -04:00
|
|
|
});
|
2017-10-30 21:19:44 -04:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2017-10-30 20:03:21 -04:00
|
|
|
_onPasswordActivated() {
|
2019-12-12 04:08:42 -05:00
|
|
|
this._entry.password_visible = !this._entry.password_visible;
|
2011-09-14 12:47:09 -04:00
|
|
|
}
|
2017-10-30 21:19:44 -04:00
|
|
|
};
|
2011-09-14 12:47:09 -04:00
|
|
|
|
|
|
|
function _setMenuAlignment(entry, stageX) {
|
2019-02-01 08:41:55 -05:00
|
|
|
let [success, entryX] = entry.transform_stage_point(stageX, 0);
|
2011-09-14 12:47:09 -04:00
|
|
|
if (success)
|
2012-05-27 13:06:08 -04:00
|
|
|
entry.menu.setSourceAlignment(entryX / entry.width);
|
2019-01-28 20:18:52 -05:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2012-09-06 15:14:32 -04:00
|
|
|
function _onButtonPressEvent(actor, event, entry) {
|
2012-05-27 13:06:08 -04:00
|
|
|
if (entry.menu.isOpen) {
|
2013-02-18 15:55:44 -05:00
|
|
|
entry.menu.close(BoxPointer.PopupAnimation.FULL);
|
2013-11-29 13:17:34 -05:00
|
|
|
return Clutter.EVENT_STOP;
|
2012-09-06 15:14:32 -04:00
|
|
|
} else if (event.get_button() == 3) {
|
2019-02-01 08:41:55 -05:00
|
|
|
let [stageX] = event.get_coords();
|
2011-09-14 12:47:09 -04:00
|
|
|
_setMenuAlignment(entry, stageX);
|
2013-02-18 15:55:44 -05:00
|
|
|
entry.menu.open(BoxPointer.PopupAnimation.FULL);
|
2013-11-29 13:17:34 -05:00
|
|
|
return Clutter.EVENT_STOP;
|
2011-09-14 12:47:09 -04:00
|
|
|
}
|
2013-11-29 13:17:34 -05:00
|
|
|
return Clutter.EVENT_PROPAGATE;
|
2019-01-28 20:18:52 -05:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2012-09-06 15:14:32 -04:00
|
|
|
function _onPopup(actor, entry) {
|
2019-12-08 15:56:46 -05:00
|
|
|
let cursorPosition = entry.clutter_text.get_cursor_position();
|
|
|
|
let [success, textX, textY_, lineHeight_] = entry.clutter_text.position_to_coords(cursorPosition);
|
2011-09-14 12:47:09 -04:00
|
|
|
if (success)
|
2012-05-27 13:06:08 -04:00
|
|
|
entry.menu.setSourceAlignment(textX / entry.width);
|
2013-02-18 15:55:44 -05:00
|
|
|
entry.menu.open(BoxPointer.PopupAnimation.FULL);
|
2019-01-28 20:18:52 -05:00
|
|
|
}
|
2011-09-14 12:47:09 -04:00
|
|
|
|
|
|
|
function addContextMenu(entry, params) {
|
2012-05-27 13:06:08 -04:00
|
|
|
if (entry.menu)
|
2011-09-14 12:47:09 -04:00
|
|
|
return;
|
|
|
|
|
2019-12-16 13:41:49 -05:00
|
|
|
params = Params.parse(params, { actionMode: Shell.ActionMode.POPUP });
|
2013-05-09 21:35:05 -04:00
|
|
|
|
|
|
|
entry.menu = new EntryMenu(entry);
|
2019-04-09 19:23:59 -04:00
|
|
|
entry._menuManager = new PopupMenu.PopupMenuManager(entry,
|
2019-01-23 15:59:42 -05:00
|
|
|
{ actionMode: params.actionMode });
|
2012-05-27 13:06:08 -04:00
|
|
|
entry._menuManager.addMenu(entry.menu);
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2012-09-06 15:14:32 -04:00
|
|
|
// Add an event handler to both the entry and its clutter_text; the former
|
2011-09-14 12:47:09 -04:00
|
|
|
// so padding is included in the clickable area, the latter because the
|
|
|
|
// event processing of ClutterText prevents event-bubbling.
|
2017-12-01 19:27:35 -05:00
|
|
|
entry.clutter_text.connect('button-press-event', (actor, event) => {
|
|
|
|
_onButtonPressEvent(actor, event, entry);
|
|
|
|
});
|
|
|
|
entry.connect('button-press-event', (actor, event) => {
|
|
|
|
_onButtonPressEvent(actor, event, entry);
|
|
|
|
});
|
2011-09-14 12:47:09 -04:00
|
|
|
|
2019-01-27 19:42:00 -05:00
|
|
|
entry.connect('popup-menu', actor => _onPopup(actor, entry));
|
2013-04-04 11:35:50 -04:00
|
|
|
|
2017-10-30 20:38:18 -04:00
|
|
|
entry.connect('destroy', () => {
|
2013-04-04 11:35:50 -04:00
|
|
|
entry.menu.destroy();
|
|
|
|
entry.menu = null;
|
|
|
|
entry._menuManager = null;
|
|
|
|
});
|
2011-09-14 12:47:09 -04:00
|
|
|
}
|
2019-12-13 03:06:14 -05:00
|
|
|
|
|
|
|
var CapsLockWarning = GObject.registerClass(
|
|
|
|
class CapsLockWarning extends St.Label {
|
|
|
|
_init(params) {
|
|
|
|
let defaultParams = { style_class: 'prompt-dialog-error-label' };
|
|
|
|
super._init(Object.assign(defaultParams, params));
|
|
|
|
|
|
|
|
this.text = _('Caps lock is on.');
|
|
|
|
|
|
|
|
this._keymap = Clutter.get_default_backend().get_keymap();
|
|
|
|
|
|
|
|
this.connect('notify::mapped', () => {
|
|
|
|
if (this.is_mapped()) {
|
|
|
|
this.stateChangedId = this._keymap.connect('state-changed',
|
|
|
|
this._updateCapsLockWarningOpacity.bind(this));
|
|
|
|
} else {
|
|
|
|
this._keymap.disconnect(this.stateChangedId);
|
|
|
|
this.stateChangedId = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._updateCapsLockWarningOpacity();
|
|
|
|
});
|
|
|
|
|
|
|
|
this.connect('destroy', () => {
|
|
|
|
if (this.stateChangedId > 0)
|
|
|
|
this._keymap.disconnect(this.stateChangedId);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.clutter_text.ellipsize = Pango.EllipsizeMode.NONE;
|
|
|
|
this.clutter_text.line_wrap = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
_updateCapsLockWarningOpacity() {
|
|
|
|
let capsLockOn = this._keymap.get_caps_lock_state();
|
|
|
|
this.opacity = capsLockOn ? 255 : 0;
|
|
|
|
}
|
|
|
|
});
|