| /** |
| * @license |
| * Copyright (C) 2016 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| import {getBaseUrl} from '../../../utils/url-util'; |
| import {GrAttributeHelper} from '../../plugins/gr-attribute-helper/gr-attribute-helper'; |
| import {GrChangeActionsInterface} from './gr-change-actions-js-api'; |
| import {GrChangeReplyInterface} from './gr-change-reply-js-api'; |
| import {GrDomHooksManager} from '../../plugins/gr-dom-hooks/gr-dom-hooks'; |
| import {GrThemeApi} from '../../plugins/gr-theme-api/gr-theme-api'; |
| import {GrPopupInterface} from '../../plugins/gr-popup-interface/gr-popup-interface'; |
| import {GrAdminApi} from '../../plugins/gr-admin-api/gr-admin-api'; |
| import {GrAnnotationActionsInterface} from './gr-annotation-actions-js-api'; |
| import {GrChangeMetadataApi} from '../../plugins/gr-change-metadata-api/gr-change-metadata-api'; |
| import {GrEventHelper} from '../../plugins/gr-event-helper/gr-event-helper'; |
| import {GrPluginRestApi} from './gr-plugin-rest-api'; |
| import {GrRepoApi} from '../../plugins/gr-repo-api/gr-repo-api'; |
| import {GrSettingsApi} from '../../plugins/gr-settings-api/gr-settings-api'; |
| import {GrStylesApi} from '../../plugins/gr-styles-api/gr-styles-api'; |
| import {getPluginEndpoints} from './gr-plugin-endpoints'; |
| |
| import {getPluginNameFromUrl, PRELOADED_PROTOCOL, send} from './gr-api-utils'; |
| import {GrReportingJsApi} from './gr-reporting-js-api'; |
| import {EventType, PluginApi, TargetElement} from '../../../api/plugin'; |
| import {RequestPayload} from '../../../types/common'; |
| import {HttpMethod} from '../../../constants/constants'; |
| import {GrChangeActions} from '../../change/gr-change-actions/gr-change-actions'; |
| import {GrChecksApi} from '../../plugins/gr-checks-api/gr-checks-api'; |
| import {appContext} from '../../../services/app-context'; |
| import {AdminPluginApi} from '../../../api/admin'; |
| import {AnnotationPluginApi} from '../../../api/annotation'; |
| import {StylesPluginApi} from '../../../api/styles'; |
| import {ThemePluginApi} from '../../../api/theme'; |
| import {EventHelperPluginApi} from '../../../api/event-helper'; |
| import {PopupPluginApi} from '../../../api/popup'; |
| import {SettingsPluginApi} from '../../../api/settings'; |
| import {ReportingPluginApi} from '../../../api/reporting'; |
| import {ChangeActionsPluginApi} from '../../../api/change-actions'; |
| import {ChangeMetadataPluginApi} from '../../../api/change-metadata'; |
| import {RepoPluginApi} from '../../../api/repo'; |
| import {ChangeReplyPluginApi} from '../../../api/change-reply'; |
| import {RestPluginApi} from '../../../api/rest'; |
| import {HookApi, RegisterOptions} from '../../../api/hook'; |
| |
| /** |
| * Plugin-provided custom components can affect content in extension |
| * points using one of following methods: |
| * - DECORATE: custom component is set with `content` attribute and may |
| * decorate (e.g. style) DOM element. |
| * - REPLACE: contents of extension point are replaced with the custom |
| * component. |
| * - STYLE: custom component is a shared styles module that is inserted |
| * into the extension point. |
| */ |
| enum EndpointType { |
| DECORATE = 'decorate', |
| REPLACE = 'replace', |
| STYLE = 'style', |
| } |
| |
| const PLUGIN_NAME_NOT_SET = 'NULL'; |
| |
| export type SendCallback = (response: unknown) => void; |
| |
| export class Plugin implements PluginApi { |
| readonly _url?: URL; |
| |
| private _domHooks: GrDomHooksManager; |
| |
| private readonly _name: string = PLUGIN_NAME_NOT_SET; |
| |
| private readonly jsApi = appContext.jsApiService; |
| |
| constructor(url?: string) { |
| this._domHooks = new GrDomHooksManager(this); |
| |
| if (!url) { |
| console.warn( |
| 'Plugin not being loaded from /plugins base path.', |
| 'Unable to determine name.' |
| ); |
| return this; |
| } |
| |
| this._url = new URL(url); |
| this._name = getPluginNameFromUrl(this._url) ?? 'NULL'; |
| } |
| |
| getPluginName() { |
| return this._name; |
| } |
| |
| registerStyleModule(endpoint: string, moduleName: string) { |
| getPluginEndpoints().registerModule(this, { |
| endpoint, |
| type: EndpointType.STYLE, |
| moduleName, |
| }); |
| } |
| |
| /** |
| * Registers an endpoint for the plugin. |
| */ |
| registerCustomComponent( |
| endpointName: string, |
| moduleName?: string, |
| options?: RegisterOptions |
| ): HookApi { |
| return this._registerCustomComponent(endpointName, moduleName, options); |
| } |
| |
| /** |
| * Registers a dynamic endpoint for the plugin. |
| * |
| * Dynamic plugins are registered by specific prefix, such as |
| * 'change-list-header'. |
| */ |
| registerDynamicCustomComponent( |
| endpointName: string, |
| moduleName?: string, |
| options?: RegisterOptions |
| ): HookApi { |
| const fullEndpointName = `${endpointName}-${this.getPluginName()}`; |
| return this._registerCustomComponent( |
| fullEndpointName, |
| moduleName, |
| options, |
| endpointName |
| ); |
| } |
| |
| _registerCustomComponent( |
| endpoint: string, |
| moduleName?: string, |
| options?: RegisterOptions, |
| dynamicEndpoint?: string |
| ): HookApi { |
| const type = |
| options && options.replace ? EndpointType.REPLACE : EndpointType.DECORATE; |
| const slot = (options && options.slot) || ''; |
| const domHook = this._domHooks.getDomHook(endpoint, moduleName); |
| moduleName = moduleName || domHook.getModuleName(); |
| getPluginEndpoints().registerModule(this, { |
| slot, |
| endpoint, |
| type, |
| moduleName, |
| domHook, |
| dynamicEndpoint, |
| }); |
| return domHook; |
| } |
| |
| /** |
| * Returns instance of DOM hook API for endpoint. Creates a placeholder |
| * element for the first call. |
| */ |
| hook(endpointName: string, options?: RegisterOptions) { |
| return this.registerCustomComponent(endpointName, undefined, options); |
| } |
| |
| getServerInfo() { |
| return appContext.restApiService.getConfig(); |
| } |
| |
| // eslint-disable-next-line @typescript-eslint/no-explicit-any |
| on(eventName: EventType, callback: (...args: any[]) => any) { |
| this.jsApi.addEventCallback(eventName, callback); |
| } |
| |
| url(path?: string) { |
| if (!this._url) throw new Error('plugin url not set'); |
| const relPath = '/plugins/' + this._name + (path || '/'); |
| const sameOriginPath = window.location.origin + `${getBaseUrl()}${relPath}`; |
| if (window.location.origin === this._url.origin) { |
| // Plugin loaded from the same origin as gr-app, getBaseUrl in effect. |
| return sameOriginPath; |
| } else if (this._url.protocol === PRELOADED_PROTOCOL) { |
| // Plugin is preloaded, load plugin with ASSETS_PATH or location.origin |
| return window.ASSETS_PATH |
| ? `${window.ASSETS_PATH}${relPath}` |
| : sameOriginPath; |
| } else { |
| // Plugin loaded from assets bundle, expect assets placed along with it. |
| return this._url.href.split('/plugins/' + this._name)[0] + relPath; |
| } |
| } |
| |
| screenUrl(screenName?: string) { |
| const origin = location.origin; |
| const base = getBaseUrl(); |
| const tokenPart = screenName ? '/' + screenName : ''; |
| return `${origin}${base}/x/${this.getPluginName()}${tokenPart}`; |
| } |
| |
| _send( |
| method: HttpMethod, |
| url: string, |
| callback?: SendCallback, |
| payload?: RequestPayload |
| ) { |
| return send(method, this.url(url), callback, payload); |
| } |
| |
| get(url: string, callback?: SendCallback) { |
| console.warn('.get() is deprecated! Use .restApi().get()'); |
| return this._send(HttpMethod.GET, url, callback); |
| } |
| |
| post(url: string, payload: RequestPayload, callback?: SendCallback) { |
| console.warn('.post() is deprecated! Use .restApi().post()'); |
| return this._send(HttpMethod.POST, url, callback, payload); |
| } |
| |
| put(url: string, payload: RequestPayload, callback?: SendCallback) { |
| console.warn('.put() is deprecated! Use .restApi().put()'); |
| return this._send(HttpMethod.PUT, url, callback, payload); |
| } |
| |
| delete(url: string, callback?: SendCallback) { |
| console.warn('.delete() is deprecated! Use plugin.restApi().delete()'); |
| return this.restApi() |
| .delete(this.url(url)) |
| .then(res => { |
| if (callback) callback(res); |
| return res; |
| }); |
| } |
| |
| annotationApi(): AnnotationPluginApi { |
| return new GrAnnotationActionsInterface(this); |
| } |
| |
| changeActions(): ChangeActionsPluginApi { |
| return new GrChangeActionsInterface( |
| this, |
| (this.jsApi.getElement( |
| TargetElement.CHANGE_ACTIONS |
| ) as unknown) as GrChangeActions |
| ); |
| } |
| |
| changeReply(): ChangeReplyPluginApi { |
| return new GrChangeReplyInterface(this, this.jsApi); |
| } |
| |
| checks(): GrChecksApi { |
| return new GrChecksApi(this); |
| } |
| |
| reporting(): ReportingPluginApi { |
| return new GrReportingJsApi(this); |
| } |
| |
| theme(): ThemePluginApi { |
| return new GrThemeApi(this); |
| } |
| |
| project(): RepoPluginApi { |
| return new GrRepoApi(this); |
| } |
| |
| changeMetadata(): ChangeMetadataPluginApi { |
| return new GrChangeMetadataApi(this); |
| } |
| |
| admin(): AdminPluginApi { |
| return new GrAdminApi(this); |
| } |
| |
| settings(): SettingsPluginApi { |
| return new GrSettingsApi(this); |
| } |
| |
| styles(): StylesPluginApi { |
| return new GrStylesApi(); |
| } |
| |
| restApi(prefix?: string): RestPluginApi { |
| return new GrPluginRestApi(prefix); |
| } |
| |
| attributeHelper(element: HTMLElement) { |
| return new GrAttributeHelper(element); |
| } |
| |
| eventHelper(element: HTMLElement): EventHelperPluginApi { |
| return new GrEventHelper(element); |
| } |
| |
| popup(): Promise<PopupPluginApi>; |
| |
| popup(moduleName: string): Promise<PopupPluginApi>; |
| |
| popup(moduleName?: string): Promise<PopupPluginApi | null> { |
| if (moduleName !== undefined && typeof moduleName !== 'string') { |
| console.error('.popup(element) deprecated, use .popup(moduleName)!'); |
| return Promise.resolve(null); |
| } |
| return new GrPopupInterface(this, moduleName).open(); |
| } |
| |
| screen(screenName: string, moduleName?: string) { |
| if (moduleName && typeof moduleName !== 'string') { |
| console.error( |
| '.screen(pattern, callback) deprecated, use ' + |
| '.screen(screenName, moduleName)!' |
| ); |
| return; |
| } |
| return this.registerCustomComponent( |
| this._getScreenName(screenName), |
| moduleName |
| ); |
| } |
| |
| _getScreenName(screenName: string) { |
| return `${this.getPluginName()}-screen-${screenName}`; |
| } |
| } |