/* 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"; const { BrowserLoader } = ChromeUtils.import( "resource://devtools/shared/loader/browser-loader.js" ); loader.lazyRequireGetter( this, "openDocLink", "resource://devtools/client/shared/link.js", true ); class CssCompatibilityTooltipHelper { constructor() { this.addTab = this.addTab.bind(this); } #currentTooltip = null; #currentUrl = null; #createElement(doc, tag, classList = [], attributeList = {}) { const XHTML_NS = "http://www.w3.org/1999/xhtml"; const newElement = doc.createElementNS(XHTML_NS, tag); for (const elementClass of classList) { newElement.classList.add(elementClass); } for (const key in attributeList) { newElement.setAttribute(key, attributeList[key]); } return newElement; } /* * Attach the UnsupportedBrowserList component to the * ".compatibility-browser-list-wrapper" div to render the * unsupported browser list */ #renderUnsupportedBrowserList(container, unsupportedBrowsers) { // Mount the ReactDOM only if the unsupported browser // list is not empty. Else "compatibility-browser-list-wrapper" // is not defined. For example, for property clip, // unsupportedBrowsers is an empty array if (!unsupportedBrowsers.length) { return; } const { require } = BrowserLoader({ baseURI: "resource://devtools/client/shared/widgets/tooltip/", window: this.#currentTooltip.doc.defaultView, }); const { createFactory, createElement, } = require("resource://devtools/client/shared/vendor/react.js"); const ReactDOM = require("resource://devtools/client/shared/vendor/react-dom.js"); const UnsupportedBrowserList = createFactory( require("resource://devtools/client/inspector/compatibility/components/UnsupportedBrowserList.js") ); const unsupportedBrowserList = createElement(UnsupportedBrowserList, { browsers: unsupportedBrowsers, }); ReactDOM.render( unsupportedBrowserList, container.querySelector(".compatibility-browser-list-wrapper") ); } /* * Get the first paragraph for the compatibility tooltip * Return a subtree similar to: *

*

*/ #getCompatibilityMessage(doc, data) { const { msgId, property } = data; return this.#createElement(doc, "p", [], { "data-l10n-id": msgId, "data-l10n-args": JSON.stringify({ property }), }); } /** * Gets the paragraph elements related to the browserList. * This returns an array with following subtree: * [ *

, *

*

*

* ] * The first element is the message and the second element is the * unsupported browserList itself. * If the unsupportedBrowser is an empty array, we return an empty * array back. */ #getBrowserListContainer(doc, unsupportedBrowsers) { if (!unsupportedBrowsers.length) { return null; } const browserList = this.#createElement(doc, "p"); const browserListWrapper = this.#createElement(doc, "div", [ "compatibility-browser-list-wrapper", ]); browserList.appendChild(browserListWrapper); return browserList; } /* * This is the learn more message element linking to the MDN documentation * for the particular incompatible CSS declaration. * The element returned is: *

* *

*/ #getLearnMoreMessage(doc, { rootProperty }) { const learnMoreMessage = this.#createElement(doc, "p", [], { "data-l10n-id": "css-compatibility-learn-more-message", "data-l10n-args": JSON.stringify({ rootProperty }), }); learnMoreMessage.appendChild( this.#createElement(doc, "span", ["link"], { "data-l10n-name": "link", }) ); return learnMoreMessage; } /** * Fill the tooltip with inactive CSS information. * * @param {Object} data * An object in the following format: { * // Type of compatibility issue * type: , * // The CSS declaration that has compatibility issues * // The raw CSS declaration name that has compatibility issues * declaration: , * property: , * // Alias to the given CSS property * alias: , * // Link to MDN documentation for the particular CSS rule * url: , * deprecated: , * experimental: , * // An array of all the browsers that don't support the given CSS rule * unsupportedBrowsers: , * } * @param {HTMLTooltip} tooltip * The tooltip we are targetting. */ async setContent(data, tooltip) { const fragment = this.getTemplate(data, tooltip); const { doc } = tooltip; tooltip.panel.innerHTML = ""; tooltip.panel.addEventListener("click", this.addTab); tooltip.once("hidden", () => { tooltip.panel.removeEventListener("click", this.addTab); }); // Because Fluent is async we need to manually translate the fragment and // then insert it into the tooltip. This is needed in order for the tooltip // to size to the contents properly and for tests. await doc.l10n.translateFragment(fragment); doc.l10n.pauseObserving(); tooltip.panel.appendChild(fragment); doc.l10n.resumeObserving(); // Size the content. tooltip.setContentSize({ width: 267, height: Infinity }); } /** * Get the template that the Fluent string will be merged with. This template * looks like this: * *
*

* *

* *

* * *

*
* * @param {Object} data * An object in the following format: { * // Type of compatibility issue * type: , * // The CSS declaration that has compatibility issues * // The raw CSS declaration name that has compatibility issues * declaration: , * property: , * // Alias to the given CSS property * alias: , * // Link to MDN documentation for the particular CSS rule * url: , * // Link to the spec for the particular CSS rule * specUrl: , * deprecated: , * experimental: , * // An array of all the browsers that don't support the given CSS rule * unsupportedBrowsers: , * } * @param {HTMLTooltip} tooltip * The tooltip we are targetting. */ getTemplate(data, tooltip) { const { doc } = tooltip; const { specUrl, url, unsupportedBrowsers } = data; this.#currentTooltip = tooltip; this.#currentUrl = url ? `${url}?utm_source=devtools&utm_medium=inspector-css-compatibility&utm_campaign=default` : specUrl; const templateNode = this.#createElement(doc, "template"); const tooltipContainer = this.#createElement(doc, "div", [ "devtools-tooltip-css-compatibility", ]); tooltipContainer.appendChild(this.#getCompatibilityMessage(doc, data)); const browserListContainer = this.#getBrowserListContainer( doc, unsupportedBrowsers ); if (browserListContainer) { tooltipContainer.appendChild(browserListContainer); this.#renderUnsupportedBrowserList(tooltipContainer, unsupportedBrowsers); } if (this.#currentUrl) { tooltipContainer.appendChild(this.#getLearnMoreMessage(doc, data)); } templateNode.content.appendChild(tooltipContainer); return doc.importNode(templateNode.content, true); } /** * Hide the tooltip, open `this.#currentUrl` in a new tab and focus it. * * @param {DOMEvent} event * The click event originating from the tooltip. * */ addTab(event) { // The XUL panel swallows click events so handlers can't be added directly // to the link span. As a workaround we listen to all click events in the // panel and if a link span is clicked we proceed. if (event.target.className !== "link") { return; } const tooltip = this.#currentTooltip; tooltip.hide(); const isMacOS = Services.appinfo.OS === "Darwin"; openDocLink(this.#currentUrl, { relatedToCurrent: true, inBackground: isMacOS ? event.metaKey : event.ctrlKey, }); } destroy() { this.#currentTooltip = null; this.#currentUrl = null; } } module.exports = CssCompatibilityTooltipHelper;