| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300 |
- //
- // Copyright (c) 2014-2016 THUNDERBEAST GAMES LLC
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to deal
- // in the Software without restriction, including without limitation the rights
- // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
- // copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
- // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
- // THE SOFTWARE.
- //
- import ClientExtensionEventNames from "./ClientExtensionEventNames";
- import HostInteropType from "../interop";
- // Entry point for web view extensions -- extensions that live inside the web view
- interface EventSubscription {
- eventName: string;
- callback: (data: any) => any;
- }
- /**
- * Implements an event dispatcher for the client services
- */
- export class EventDispatcher implements Editor.Extensions.EventDispatcher {
- private subscriptions: EventSubscription[] = [];
- sendEvent<T extends Atomic.EventCallbackMetaData>(eventCallbackMetaData:T)
- sendEvent(eventType: string, data: any)
- sendEvent(eventTypeOrWrapped: any, data?: any) {
- let eventType: string;
- let eventData: any;
- if (typeof(eventTypeOrWrapped) == "string") {
- eventType = eventTypeOrWrapped;
- eventData = data;
- } else {
- const metaData = eventTypeOrWrapped as Atomic.EventCallbackMetaData;
- eventType = metaData._eventType;
- eventData = metaData._callbackData;
- }
- this.subscriptions.forEach(sub => {
- if (sub.eventName == eventType) {
- sub.callback(eventData);
- }
- });
- }
- subscribeToEvent(eventType: string, callback: (...params) => any);
- subscribeToEvent(wrapped: Atomic.EventMetaData);
- subscribeToEvent(eventTypeOrWrapped: any, callback?: any) {
- if (callback) {
- this.subscriptions.push({
- eventName: eventTypeOrWrapped,
- callback: callback
- });
- } else {
- this.subscriptions.push({
- eventName: eventTypeOrWrapped._eventType,
- callback: eventTypeOrWrapped._callback
- });
- }
- }
- }
- /**
- * Generic registry for storing Editor Extension Services
- */
- class ServicesProvider<T extends Editor.Extensions.ServiceEventListener> implements Editor.Extensions.ServicesProvider<T> {
- registeredServices: T[] = [];
- /**
- * Adds a service to the registered services list for this type of service
- * @param {T} service the service to register
- */
- register(service: T) {
- this.registeredServices.push(service);
- }
- unregister(service: T) {
- var index = this.registeredServices.indexOf(service, 0);
- if (index > -1) {
- this.registeredServices.splice(index, 1);
- }
- }
- }
- export class WebViewServicesProvider extends ServicesProvider<Editor.ClientExtensions.WebViewServiceEventListener> {
- private projectPreferences = {};
- private applicationPreferences = {};
- /**
- * Sets the preferences for the service locator
- * @param {any} projectPreferences
- * @param {any} applicationPreferences
- * @return {[type]}
- */
- setPreferences(projectPreferences?: any, applicationPreferences?: any) {
- this.projectPreferences = projectPreferences || this.projectPreferences;
- this.applicationPreferences = applicationPreferences || this.applicationPreferences;
- }
- /**
- * Allow this service registry to subscribe to events that it is interested in
- * @param {EventDispatcher} eventDispatcher The global event dispatcher
- */
- subscribeToEvents(eventDispatcher: Editor.Extensions.EventDispatcher) {
- eventDispatcher.subscribeToEvent(ClientExtensionEventNames.CodeLoadedEvent, (ev) => this.codeLoaded(ev));
- eventDispatcher.subscribeToEvent(ClientExtensionEventNames.ConfigureEditorEvent, (ev) => this.configureEditor(ev));
- eventDispatcher.subscribeToEvent(ClientExtensionEventNames.ResourceRenamedEvent, (ev) => this.renameResource(ev));
- eventDispatcher.subscribeToEvent(ClientExtensionEventNames.ResourceDeletedEvent, (ev) => this.deleteResource(ev));
- eventDispatcher.subscribeToEvent(ClientExtensionEventNames.CodeSavedEvent, (ev) => this.saveCode(ev));
- eventDispatcher.subscribeToEvent(ClientExtensionEventNames.PreferencesChangedEvent, (ev) => this.preferencesChanged(ev));
- eventDispatcher.subscribeToEvent(ClientExtensionEventNames.FormatCodeEvent, (ev) => this.formatCode());
- }
- /**
- * Called when code is loaded
- * @param {Editor.ClientExtensions.CodeLoadedEvent} ev Event info about the file that is being loaded
- */
- codeLoaded(ev: Editor.ClientExtensions.CodeLoadedEvent) {
- this.registeredServices.forEach((service) => {
- try {
- // Notify services that the project has just been loaded
- if (service.codeLoaded) {
- service.codeLoaded(ev);
- }
- } catch (e) {
- alert(`Extension Error:\n Error detected in extension ${service.name}\n \n ${e.stack}`);
- }
- });
- }
- /**
- * Called after code has been saved
- * @param {Editor.ClientExtensions.SaveResourceEvent} ev
- */
- saveCode(ev: Editor.ClientExtensions.CodeSavedEvent) {
- // run through and find any services that can handle this.
- this.registeredServices.forEach((service) => {
- try {
- // Verify that the service contains the appropriate methods and that it can save
- if (service.save) {
- service.save(ev);
- }
- } catch (e) {
- alert(`Error detected in extension ${service.name}\n \n ${e.stack}`);
- }
- });
- }
- /**
- * Called when a resource has been deleted
- */
- deleteResource(ev: Editor.ClientExtensions.DeleteResourceEvent) {
- this.registeredServices.forEach((service) => {
- try {
- // Verify that the service contains the appropriate methods and that it can delete
- if (service.delete) {
- service.delete(ev);
- }
- } catch (e) {
- alert(`Error detected in extension ${service.name}\n \n ${e.stack}`);
- }
- });
- }
- /**
- * Called when a resource has been renamed
- * @param {Editor.ClientExtensions.RenameResourceEvent} ev
- */
- renameResource(ev: Editor.ClientExtensions.RenameResourceEvent) {
- this.registeredServices.forEach((service) => {
- try {
- // Verify that the service contains the appropriate methods and that it can handle the rename
- if (service.rename) {
- service.rename(ev);
- }
- } catch (e) {
- alert(`Error detected in extension ${service.name}\n \n ${e.stack}`);
- }
- });
- }
- /**
- * Called when the editor code should be formatted
- */
- formatCode() {
- this.registeredServices.forEach((service) => {
- try {
- // Verify that the service contains the appropriate methods and that it can handle the rename
- if (service.formatCode) {
- service.formatCode();
- }
- } catch (e) {
- alert(`Error detected in extension ${service.name}\n \n ${e.stack}`);
- }
- });
- }
- /**
- * Called when the editor is requesting to be configured for a particular file
- * @param {Editor.ClientExtensions.EditorFileEvent} ev
- */
- configureEditor(ev: Editor.ClientExtensions.EditorFileEvent) {
- this.registeredServices.forEach((service) => {
- try {
- // Notify services that the project has just been loaded
- if (service.configureEditor) {
- service.configureEditor(ev);
- }
- } catch (e) {
- alert(`Extension Error:\n Error detected in extension ${service.name}\n \n ${e.stack}`);
- }
- });
- }
- /**
- * Called when preferences changes
- * @param {Editor.EditorEvents.PreferencesChangedEvent} ev
- */
- preferencesChanged(prefs: Editor.ClientExtensions.PreferencesChangedEventData) {
- this.registeredServices.forEach((service) => {
- // Notify services that the project has been unloaded
- try {
- if (service.preferencesChanged) {
- service.preferencesChanged(prefs);
- }
- } catch (e) {
- alert(`Extension Error:\n Error detected in extension ${service.name}\n \n ${e.stack}`);
- }
- });
- }
- /**
- * Returns the Host Interop module
- * @return {Editor.ClientExtensions.HostInterop}
- */
- getHostInterop(): Editor.ClientExtensions.HostInterop {
- return HostInteropType.getInstance();
- }
- /**
- * Return a preference value or the provided default from the user settings file
- * @param {string} settignsGroup name of the group the preference lives under
- * @param {string} preferenceName name of the preference to retrieve
- * @param {number | boolean | string} defaultValue value to return if pref doesn't exist
- * @return {number|boolean|string}
- */
- getUserPreference(settingsGroup: string, preferenceName: string, defaultValue?: number): number;
- getUserPreference(settingsGroup: string, preferenceName: string, defaultValue?: string): string;
- getUserPreference(settingsGroup: string, preferenceName: string, defaultValue?: boolean): boolean;
- getUserPreference(settingsGroup: string, preferenceName: string, defaultValue?: any): any {
- if (this.projectPreferences) {
- let prefs = this.projectPreferences[settingsGroup];
- if (prefs) {
- return prefs[preferenceName] || defaultValue;
- }
- }
- // if all else fails
- return defaultValue;
- }
- /**
- * Return a preference value or the provided default from the user settings file
- * @param {string} settignsGroup name of the group the preference lives under
- * @param {string} preferenceName name of the preference to retrieve
- * @param {number | boolean | string} defaultValue value to return if pref doesn't exist
- * @return {number|boolean|string}
- */
- getApplicationPreference(settingsGroup: string, preferenceName: string, defaultValue?: number): number;
- getApplicationPreference(settingsGroup: string, preferenceName: string, defaultValue?: string): string;
- getApplicationPreference(settingsGroup: string, preferenceName: string, defaultValue?: boolean): boolean;
- getApplicationPreference(settingsGroup: string, preferenceName: string, defaultValue?: any): any {
- if (this.applicationPreferences) {
- let prefs = this.applicationPreferences[settingsGroup];
- if (prefs) {
- return prefs[preferenceName] || defaultValue;
- }
- }
- // if all else fails
- return defaultValue;
- }
- }
|