/* 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/. */ /** * This module exports a urlbar result class, each representing a single result * found by a provider that can be passed from the model to the view through * the controller. It is mainly defined by a result type, and a payload, * containing the data. A few getters allow to retrieve information common to all * the result types. */ const lazy = {}; ChromeUtils.defineESModuleGetters(lazy, { JsonSchemaValidator: "resource://gre/modules/components-utils/JsonSchemaValidator.sys.mjs", ObjectUtils: "resource://gre/modules/ObjectUtils.sys.mjs", UrlbarUtils: "moz-src:///browser/components/urlbar/UrlbarUtils.sys.mjs", }); /** * @typedef UrlbarAutofillData * @property {string} value * The value to insert for autofill. * @property {number} selectionStart * Where to start the selection for the autofill. * @property {number} selectionEnd * Where to end the selection for the autofill. * @property {string} [type] * The type of the autofill. * @property {string} [adaptiveHistoryInput] * The input string associated with this autofill item. */ /** * Class used to create a single result. */ export class UrlbarResult { /** * @typedef {{ [name: string]: any }} Payload * * @typedef {typeof lazy.UrlbarUtils.HIGHLIGHT} HighlightType * @typedef {Array<[number, number]>} HighlightIndexes e.g. [[index, length],,] * @typedef {Record} Highlights */ /** * @param {object} params * @param {Values} params.type * @param {Values} params.source * @param {UrlbarAutofillData} [params.autofill] * @param {number} [params.exposureTelemetry] * @param {Values} [params.group] * @param {boolean} [params.heuristic] * @param {boolean} [params.hideRowLabel] * @param {boolean} [params.isBestMatch] * @param {boolean} [params.isNovaSuggestion] * @param {boolean} [params.isRichSuggestion] * @param {boolean} [params.isSuggestedIndexRelativeToGroup] * @param {string} [params.providerName] * @param {number} [params.resultSpan] * @param {number} [params.richSuggestionIconSize] * @param {string} [params.richSuggestionIconVariation] * @param {string} [params.rowLabel] * @param {boolean} [params.showFeedbackMenu] * @param {number} [params.suggestedIndex] * @param {Payload} [params.payload] * @param {Highlights} [params.highlights] * @param {boolean} [params.testForceNewContent] Used for test only. */ constructor({ type, source, autofill, exposureTelemetry = lazy.UrlbarUtils.EXPOSURE_TELEMETRY.NONE, group, heuristic = false, hideRowLabel = false, isBestMatch = false, isNovaSuggestion = false, isRichSuggestion = false, isSuggestedIndexRelativeToGroup = false, providerName, resultSpan, richSuggestionIconSize, richSuggestionIconVariation, rowLabel, showFeedbackMenu = false, suggestedIndex, payload, highlights = null, testForceNewContent, }) { // Type describes the payload and visualization that should be used for // this result. if (!Object.values(lazy.UrlbarUtils.RESULT_TYPE).includes(type)) { throw new Error("Invalid result type"); } this.#type = type; // Source describes which data has been used to derive this result. In case // multiple sources are involved, use the more privacy restricted. if (!Object.values(lazy.UrlbarUtils.RESULT_SOURCE).includes(source)) { throw new Error("Invalid result source"); } this.#source = source; // The payload contains result data. Some of the data is common across // multiple types, but most of it will vary. if (!payload || typeof payload != "object") { throw new Error("Invalid result payload"); } payload = Object.fromEntries( Object.entries(payload).filter(([_, v]) => v != undefined) ); if (highlights) { this.#highlights = Object.freeze(highlights); } this.#payload = this.#validatePayload(payload); this.#autofill = autofill; this.#exposureTelemetry = exposureTelemetry; this.#group = group; this.#heuristic = heuristic; this.#hideRowLabel = hideRowLabel; this.#isBestMatch = isBestMatch; this.#isNovaSuggestion = isNovaSuggestion; this.#isRichSuggestion = isRichSuggestion; this.#isSuggestedIndexRelativeToGroup = isSuggestedIndexRelativeToGroup; this.#richSuggestionIconSize = richSuggestionIconSize; this.#richSuggestionIconVariation = richSuggestionIconVariation; this.#providerName = providerName; this.#resultSpan = resultSpan; this.#rowLabel = rowLabel; this.#showFeedbackMenu = showFeedbackMenu; this.#suggestedIndex = suggestedIndex; if (this.#type == lazy.UrlbarUtils.RESULT_TYPE.TIP) { this.#isRichSuggestion = true; this.#richSuggestionIconSize = 24; } this.#testForceNewContent = testForceNewContent; } /** * @type {number} * The index of the row where this result is in the suggestions. This is * updated by UrlbarView when new result sets are displayed. */ rowIndex = undefined; get type() { return this.#type; } get source() { return this.#source; } get autofill() { return this.#autofill; } get exposureTelemetry() { return this.#exposureTelemetry; } set exposureTelemetry(value) { this.#exposureTelemetry = value; } get group() { return this.#group; } get heuristic() { return this.#heuristic; } get hideRowLabel() { return this.#hideRowLabel; } get isBestMatch() { return this.#isBestMatch; } get isNovaSuggestion() { return this.#isNovaSuggestion; } get isRichSuggestion() { return this.#isRichSuggestion; } set isRichSuggestion(value) { this.#isRichSuggestion = value; } get isSuggestedIndexRelativeToGroup() { return this.#isSuggestedIndexRelativeToGroup; } set isSuggestedIndexRelativeToGroup(value) { this.#isSuggestedIndexRelativeToGroup = value; } get providerName() { return this.#providerName; } set providerName(value) { this.#providerName = value; } /** * The type of the UrlbarProvider providing the result. * * @type {?Values} */ get providerType() { return this.#providerType; } set providerType(value) { this.#providerType = value; } get resultSpan() { return this.#resultSpan; } get richSuggestionIconSize() { return this.#richSuggestionIconSize; } get richSuggestionIconVariation() { return this.#richSuggestionIconVariation; } set richSuggestionIconSize(value) { this.#richSuggestionIconSize = value; } get rowLabel() { return this.#rowLabel; } get showFeedbackMenu() { return this.#showFeedbackMenu; } get suggestedIndex() { return this.#suggestedIndex; } set suggestedIndex(value) { this.#suggestedIndex = value; } get payload() { return this.#payload; } get testForceNewContent() { return this.#testForceNewContent; } // Used only for test. get testHighlights() { return this.#highlights; } /** * Returns an icon url. * * @returns {string} url of the icon. */ get icon() { return this.payload.icon; } /** * Returns whether the result's `suggestedIndex` property is defined. * `suggestedIndex` is an optional hint to the muxer that can be set to * suggest a specific position among the results. * * @returns {boolean} Whether `suggestedIndex` is defined. */ get hasSuggestedIndex() { return typeof this.suggestedIndex == "number"; } /** * Convenience getter that returns whether the result's exposure telemetry * indicates it should be hidden. * * @returns {boolean} * Whether the result should be hidden. */ get isHiddenExposure() { return this.exposureTelemetry == lazy.UrlbarUtils.EXPOSURE_TELEMETRY.HIDDEN; } /** * Get value and highlights of given payloadName that can display in the view. * * @param {string} payloadName * The payload name to want to get the value. * @param {object} options * @param {object} [options.tokens] * Make highlighting that matches this tokens. * If no specific tokens, this function returns only value. * @param {object} [options.isURL] * If true, the value will be from UrlbarUtils.prepareUrlForDisplay(). */ getDisplayableValueAndHighlights(payloadName, options = {}) { if (!this.#displayValuesCache) { this.#displayValuesCache = new Map(); } if (this.#displayValuesCache.has(payloadName)) { let cached = this.#displayValuesCache.get(payloadName); // If the different options are specified, ignore the cache. // NOTE: If options.tokens is undefined, use cache as it is. if ( options.isURL == cached.options.isURL && (options.tokens == undefined || lazy.ObjectUtils.deepEqual(options.tokens, cached.options.tokens)) ) { return this.#displayValuesCache.get(payloadName); } } let value = this.payload[payloadName]; if (!value) { return {}; } if (options.isURL) { value = lazy.UrlbarUtils.prepareUrlForDisplay(value); } if (typeof value == "string") { value = value.substring(0, lazy.UrlbarUtils.MAX_TEXT_LENGTH); } if (Array.isArray(this.#highlights?.[payloadName])) { return { value, highlights: this.#highlights[payloadName] }; } let highlightType = this.#highlights?.[payloadName]; if (!options.tokens?.length || !highlightType) { let cached = { value, options }; this.#displayValuesCache.set(payloadName, cached); return cached; } let highlights = Array.isArray(value) ? value.map(subval => lazy.UrlbarUtils.getTokenMatches( options.tokens, subval, highlightType ) ) : lazy.UrlbarUtils.getTokenMatches(options.tokens, value, highlightType); let cached = { value, highlights, options }; this.#displayValuesCache.set(payloadName, cached); return cached; } /** * Returns the given payload if it's valid or throws an error if it's not. * The schemas in UrlbarUtils.RESULT_PAYLOAD_SCHEMA are used for validation. * * @param {object} payload The payload object. * @returns {object} `payload` if it's valid. */ #validatePayload(payload) { let schema = lazy.UrlbarUtils.getPayloadSchema(this.type); if (!schema) { throw new Error(`Unrecognized result type: ${this.type}`); } let result = lazy.JsonSchemaValidator.validate(payload, schema, { allowExplicitUndefinedProperties: true, allowNullAsUndefinedProperties: true, allowAdditionalProperties: this.type == lazy.UrlbarUtils.RESULT_TYPE.DYNAMIC, }); if (!result.valid) { throw result.error; } return payload; } static _dynamicResultTypesByName = new Map(); /** * Registers a dynamic result type. Dynamic result types are types that are * created at runtime, for example by an extension. A particular type should * be added only once; if this method is called for a type more than once, the * `type` in the last call overrides those in previous calls. * * @param {string} name * The name of the type. This is used in CSS selectors, so it shouldn't * contain any spaces or punctuation except for -, _, etc. * @param {object} type * An object that describes the type. Currently types do not have any * associated metadata, so this object should be empty. */ static addDynamicResultType(name, type = {}) { if (/[^a-z0-9_-]/i.test(name)) { console.error(`Illegal dynamic type name: ${name}`); return; } this._dynamicResultTypesByName.set(name, type); } /** * Unregisters a dynamic result type. * * @param {string} name * The name of the type. */ static removeDynamicResultType(name) { let type = this._dynamicResultTypesByName.get(name); if (type) { this._dynamicResultTypesByName.delete(name); } } /** * Returns an object describing a registered dynamic result type. * * @param {string} name * The name of the type. * @returns {object} * Currently types do not have any associated metadata, so the return value * is an empty object if the type exists. If the type doesn't exist, * undefined is returned. */ static getDynamicResultType(name) { return this._dynamicResultTypesByName.get(name); } /** * This is useful for logging results. If you need the full payload, then it's * better to JSON.stringify the result object itself. * * @returns {string} string representation of the result. */ toString() { if (this.payload.url) { return this.payload.title + " - " + this.payload.url.substr(0, 100); } if (this.payload.keyword) { return this.payload.keyword + " - " + this.payload.query; } if (this.payload.suggestion) { return this.payload.engine + " - " + this.payload.suggestion; } if (this.payload.engine) { return this.payload.engine + " - " + this.payload.query; } return JSON.stringify(this); } #type; #source; #autofill; #exposureTelemetry; #group; #heuristic; #hideRowLabel; #isBestMatch; #isNovaSuggestion; #isRichSuggestion; #isSuggestedIndexRelativeToGroup; #providerName; #providerType; #resultSpan; #richSuggestionIconSize; #richSuggestionIconVariation; #rowLabel; #showFeedbackMenu; #suggestedIndex; #payload; #highlights; #displayValuesCache; #testForceNewContent; }