/* 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/. */ /** * Provide unit converter. */ import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs"; import { UnitConverterSimple } from "moz-src:///browser/components/urlbar/unitconverters/UnitConverterSimple.sys.mjs"; import { UnitConverterTemperature } from "moz-src:///browser/components/urlbar/unitconverters/UnitConverterTemperature.sys.mjs"; import { UnitConverterTimezone } from "moz-src:///browser/components/urlbar/unitconverters/UnitConverterTimezone.sys.mjs"; import { UrlbarProvider, UrlbarUtils, } from "moz-src:///browser/components/urlbar/UrlbarUtils.sys.mjs"; const lazy = {}; ChromeUtils.defineESModuleGetters(lazy, { UrlbarPrefs: "moz-src:///browser/components/urlbar/UrlbarPrefs.sys.mjs", UrlbarResult: "moz-src:///browser/components/urlbar/UrlbarResult.sys.mjs", UrlbarView: "moz-src:///browser/components/urlbar/UrlbarView.sys.mjs", }); XPCOMUtils.defineLazyServiceGetter( lazy, "ClipboardHelper", "@mozilla.org/widget/clipboardhelper;1", Ci.nsIClipboardHelper ); const CONVERTERS = [ new UnitConverterSimple(), new UnitConverterTemperature(), new UnitConverterTimezone(), ]; const DYNAMIC_RESULT_TYPE = "unitConversion"; const VIEW_TEMPLATE = { attributes: { selectable: true, }, children: [ { name: "content", tag: "span", classList: ["urlbarView-no-wrap"], children: [ { name: "icon", tag: "img", classList: ["urlbarView-favicon"], attributes: { src: "chrome://global/skin/icons/edit-copy.svg", }, }, { name: "output", tag: "strong", attributes: { dir: "ltr", }, }, { name: "action", tag: "span", }, ], }, ], }; /** * Provide a feature that converts given units. */ export class UrlbarProviderUnitConversion extends UrlbarProvider { constructor() { super(); lazy.UrlbarResult.addDynamicResultType(DYNAMIC_RESULT_TYPE); lazy.UrlbarView.addDynamicViewTemplate(DYNAMIC_RESULT_TYPE, VIEW_TEMPLATE); } /** * @returns {Values} */ get type() { return UrlbarUtils.PROVIDER_TYPE.PROFILE; } /** * Whether the provider should be invoked for the given context. If this * method returns false, the providers manager won't start a query with this * provider, to save on resources. * * @param {UrlbarQueryContext} queryContext * The query context object. */ async isActive({ searchString }) { if (!lazy.UrlbarPrefs.get("unitConversion.enabled")) { return false; } for (const converter of CONVERTERS) { const result = converter.convert(searchString); if (result) { this._activeResult = result; return true; } } this._activeResult = null; return false; } /** * This is called only for dynamic result types, when the urlbar view updates * the view of one of the results of the provider. It should return an object * describing the view update. * * @param {UrlbarResult} result The result whose view will be updated. * @returns {object} An object describing the view update. */ getViewUpdate(result) { return { output: { textContent: result.payload.output, }, action: { l10n: { id: "urlbar-result-action-copy-to-clipboard" }, }, }; } /** * Starts querying. * * @param {UrlbarQueryContext} queryContext * @param {(provider: UrlbarProvider, result: UrlbarResult) => void} addCallback * Callback invoked by the provider to add a new result. */ startQuery(queryContext, addCallback) { const result = new lazy.UrlbarResult({ type: UrlbarUtils.RESULT_TYPE.DYNAMIC, source: UrlbarUtils.RESULT_SOURCE.OTHER_LOCAL, suggestedIndex: lazy.UrlbarPrefs.get("unitConversion.suggestedIndex"), payload: { dynamicType: DYNAMIC_RESULT_TYPE, output: this._activeResult, input: queryContext.searchString, }, }); addCallback(this, result); } onEngagement(queryContext, controller, details) { let { element } = details; const { textContent } = element.querySelector( ".urlbarView-dynamic-unitConversion-output" ); lazy.ClipboardHelper.copyString(textContent); } }