gecko-dev/toolkit/content/widgets/text.js
Brian Grinstead 4663e6262a Bug 1448213 - Migrate label-control to a Custom Element r=surkov
Differential Revision: https://phabricator.services.mozilla.com/D5686

--HG--
extra : moz-landing-system : lando
2019-04-23 20:16:46 +00:00

361 lines
10 KiB
JavaScript

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
"use strict";
// This is loaded into all XUL windows. Wrap in a block to prevent
// leaking to window scope.
{
const {Services} = ChromeUtils.import("resource://gre/modules/Services.jsm");
const MozXULTextElement = MozElements.MozElementMixin(XULTextElement);
let gInsertSeparator = false;
let gAlwaysAppendAccessKey = false;
let gUnderlineAccesskey = Services.prefs.getIntPref("ui.key.menuAccessKey") != 0;
if (gUnderlineAccesskey) {
try {
const nsIPrefLocalizedString = Ci.nsIPrefLocalizedString;
const prefNameInsertSeparator =
"intl.menuitems.insertseparatorbeforeaccesskeys";
const prefNameAlwaysAppendAccessKey =
"intl.menuitems.alwaysappendaccesskeys";
let val = Services.prefs.getComplexValue(prefNameInsertSeparator,
nsIPrefLocalizedString).data;
gInsertSeparator = val == "true";
val = Services.prefs.getComplexValue(prefNameAlwaysAppendAccessKey,
nsIPrefLocalizedString).data;
gAlwaysAppendAccessKey = val == "true";
} catch (e) {
gInsertSeparator = gAlwaysAppendAccessKey = true;
}
}
class MozTextLabel extends MozXULTextElement {
constructor() {
super();
this._lastFormattedAccessKey = null;
this.addEventListener("click", this._onClick);
}
static get observedAttributes() {
return ["accesskey", "text"];
}
set textContent(val) {
super.textContent = val;
this._lastFormattedAccessKey = null;
this.formatAccessKey();
}
get textContent() {
return super.textContent;
}
attributeChangedCallback(name, oldValue, newValue) {
if (!this.isConnectedAndReady || oldValue == newValue) {
return;
}
// As long as we are still hosted within XBL anonymous content and the [text]
// attribute inheritance, we need to handle changes this way, in addition to textContent:
if (name == "text") {
this._lastFormattedAccessKey = null;
}
// Note that this is only happening when "accesskey" or "text" attributes change:
this.formatAccessKey();
}
_onClick(event) {
let controlElement = this.labeledControlElement;
if (!controlElement || this.disabled) {
return;
}
controlElement.focus();
const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
if (controlElement.namespaceURI != XUL_NS) {
return;
}
if ((controlElement.localName == "checkbox" ||
controlElement.localName == "radio") &&
controlElement.getAttribute("disabled") == "true") {
return;
}
if (controlElement.localName == "checkbox") {
controlElement.checked = !controlElement.checked;
} else if (controlElement.localName == "radio") {
controlElement.control.selectedItem = controlElement;
}
}
connectedCallback() {
if (this.delayConnectedCallback()) {
return;
}
this.formatAccessKey();
}
set accessKey(val) {
this.setAttribute("accesskey", val);
var control = this.labeledControlElement;
if (control) {
control.setAttribute("accesskey", val);
}
}
get accessKey() {
let accessKey = this.getAttribute("accesskey");
return accessKey ? accessKey[0] : null;
}
get labeledControlElement() {
let control = this.control;
return control ? document.getElementById(control) : null;
}
set control(val) {
this.setAttribute("control", val);
}
get control() {
return this.getAttribute("control");
}
// This is used to match the rendering of accesskeys from nsTextBoxFrame.cpp (i.e. when the
// label uses [value]). So this is just for when we have textContent.
formatAccessKey() {
// Skip doing any DOM manipulation whenever possible:
let accessKey = this.accessKey;
if (!gUnderlineAccesskey ||
!this.isConnectedAndReady ||
this._lastFormattedAccessKey == accessKey ||
!this.textContent) {
return;
}
this._lastFormattedAccessKey = accessKey;
if (this.accessKeySpan) { // Clear old accesskey
mergeElement(this.accessKeySpan);
this.accessKeySpan = null;
}
if (this.hiddenColon) {
mergeElement(this.hiddenColon);
this.hiddenColon = null;
}
if (this.accessKeyParens) {
this.accessKeyParens.remove();
this.accessKeyParens = null;
}
// If we used to have an accessKey but not anymore, we're done here
if (!accessKey) {
return;
}
let labelText = this.textContent;
let accessKeyIndex = -1;
if (!gAlwaysAppendAccessKey) {
accessKeyIndex = labelText.indexOf(accessKey);
if (accessKeyIndex < 0) { // Try again in upper case
accessKeyIndex =
labelText.toUpperCase().indexOf(accessKey.toUpperCase());
}
} else if (labelText.endsWith(`(${accessKey.toUpperCase()})`)) {
accessKeyIndex = labelText.length - (1 + accessKey.length); // = index of accessKey.
}
const HTML_NS = "http://www.w3.org/1999/xhtml";
this.accessKeySpan = document.createElementNS(HTML_NS, "span");
this.accessKeySpan.className = "accesskey";
// Note that if you change the following code, see the comment of
// nsTextBoxFrame::UpdateAccessTitle.
// If accesskey is in the string, underline it:
if (accessKeyIndex >= 0) {
wrapChar(this, this.accessKeySpan, accessKeyIndex);
return;
}
// If accesskey is not in string, append in parentheses
// If end is colon, we should insert before colon.
// i.e., "label:" -> "label(X):"
let colonHidden = false;
if (/:$/.test(labelText)) {
labelText = labelText.slice(0, -1);
this.hiddenColon = document.createElementNS(HTML_NS, "span");
this.hiddenColon.className = "hiddenColon";
this.hiddenColon.style.display = "none";
// Hide the last colon by using span element.
// I.e., label<span style="display:none;">:</span>
wrapChar(this, this.hiddenColon, labelText.length);
colonHidden = true;
}
// If end is space(U+20),
// we should not add space before parentheses.
let endIsSpace = false;
if (/ $/.test(labelText)) {
endIsSpace = true;
}
this.accessKeyParens = document.createElementNS("http://www.w3.org/1999/xhtml", "span");
this.appendChild(this.accessKeyParens);
if (gInsertSeparator && !endIsSpace)
this.accessKeyParens.textContent = " (";
else
this.accessKeyParens.textContent = "(";
this.accessKeySpan.textContent = accessKey.toUpperCase();
this.accessKeyParens.appendChild(this.accessKeySpan);
if (!colonHidden) {
this.accessKeyParens.appendChild(document.createTextNode(")"));
} else {
this.accessKeyParens.appendChild(document.createTextNode("):"));
}
}
}
customElements.define("label", MozTextLabel);
function mergeElement(element) {
// If the element has been removed already, return:
if (!element.isConnected) {
return;
}
if (element.previousSibling instanceof Text) {
element.previousSibling.appendData(element.textContent);
} else {
element.parentNode.insertBefore(element.firstChild, element);
}
element.remove();
}
function wrapChar(parent, element, index) {
let treeWalker = document.createNodeIterator(parent, NodeFilter.SHOW_TEXT, null);
let node = treeWalker.nextNode();
while (index >= node.length) {
index -= node.length;
node = treeWalker.nextNode();
}
if (index) {
node = node.splitText(index);
}
node.parentNode.insertBefore(element, node);
if (node.length > 1) {
node.splitText(1);
}
element.appendChild(node);
}
class MozTextLink extends MozXULTextElement {
constructor() {
super();
this.addEventListener("click", (event) => {
if (event.button == 0 || event.button == 1) {
this.open(event);
}
}, true);
this.addEventListener("keypress", (event) => {
if (event.keyCode != KeyEvent.DOM_VK_RETURN) {
return;
}
this.click();
});
}
connectedCallback() {
this.classList.add("text-link");
}
set href(val) {
this.setAttribute("href", val);
return val;
}
get href() {
return this.getAttribute("href");
}
open(aEvent) {
var href = this.href;
if (!href || this.disabled || aEvent.defaultPrevented)
return;
var uri = null;
try {
const nsISSM = Ci.nsIScriptSecurityManager;
const secMan =
Cc["@mozilla.org/scriptsecuritymanager;1"]
.getService(nsISSM);
uri = Services.io.newURI(href);
let principal;
if (this.getAttribute("useoriginprincipal") == "true") {
principal = this.nodePrincipal;
} else {
principal = secMan.createNullPrincipal({});
}
try {
secMan.checkLoadURIWithPrincipal(principal, uri,
nsISSM.DISALLOW_INHERIT_PRINCIPAL);
} catch (ex) {
var msg = "Error: Cannot open a " + uri.scheme + ": link using \
the text-link binding.";
Cu.reportError(msg);
return;
}
const cID = "@mozilla.org/uriloader/external-protocol-service;1";
const nsIEPS = Ci.nsIExternalProtocolService;
var protocolSvc = Cc[cID].getService(nsIEPS);
// if the scheme is not an exposed protocol, then opening this link
// should be deferred to the system's external protocol handler
if (!protocolSvc.isExposedProtocol(uri.scheme)) {
protocolSvc.loadURI(uri);
aEvent.preventDefault();
return;
}
} catch (ex) {
Cu.reportError(ex);
}
aEvent.preventDefault();
href = uri ? uri.spec : href;
// Try handing off the link to the host application, e.g. for
// opening it in a tabbed browser.
var linkHandled = Cc["@mozilla.org/supports-PRBool;1"]
.createInstance(Ci.nsISupportsPRBool);
linkHandled.data = false;
let { shiftKey, ctrlKey, metaKey, altKey, button } = aEvent;
let data = { shiftKey, ctrlKey, metaKey, altKey, button, href };
Services.obs.notifyObservers(linkHandled,
"handle-xul-text-link",
JSON.stringify(data));
if (linkHandled.data)
return;
// otherwise, fall back to opening the anchor directly
var win = window;
if (window.isChromeWindow) {
while (win.opener && !win.opener.closed)
win = win.opener;
}
win.open(href);
}
}
customElements.define("text-link", MozTextLink, { extends: "label" });
}