// (C) Copyright 2015 Moodle Pty Ltd. // // 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 { Injectable } from '@angular/core'; import { Subject } from 'rxjs'; import { CoreSites } from '@services/sites'; import { CoreUtils } from '@services/utils/utils'; import { makeSingleton } from '@singletons'; import { CoreLogger } from '@singletons/logger'; import { CorePushNotificationsNotificationBasicData } from './pushnotifications'; /** * Interface that all click handlers must implement. */ export interface CorePushNotificationsClickHandler { /** * A name to identify the handler. */ name: string; /** * Handler's priority. The highest priority is treated first. */ priority?: number; /** * Name of the feature this handler is related to. * It will be used to check if the feature is disabled (@see CoreSite.isFeatureDisabled). */ featureName?: string; /** * Check if a notification click is handled by this handler. * * @param notification The notification to check. * @returns Whether the notification click is handled by this handler. */ handles(notification: CorePushNotificationsNotificationBasicData): Promise; /** * Handle the notification click. * * @param notification The notification to check. * @returns Promise resolved when done. */ handleClick(notification: CorePushNotificationsNotificationBasicData): Promise; } /** * Service to handle push notifications actions to perform when clicked and received. */ @Injectable({ providedIn: 'root' }) export class CorePushNotificationsDelegateService { protected logger: CoreLogger; protected observables: { [s: string]: Subject } = {}; protected clickHandlers: { [s: string]: CorePushNotificationsClickHandler } = {}; protected counterHandlers: Record = {}; constructor() { this.logger = CoreLogger.getInstance('CorePushNotificationsDelegate'); this.observables['receive'] = new Subject(); } /** * Function called when a push notification is clicked. Sends notification to handlers. * * @param notification Notification clicked. * @returns Promise resolved when done. */ async clicked(notification: CorePushNotificationsNotificationBasicData): Promise { if (!notification) { return; } let handlers: CorePushNotificationsClickHandler[] = []; const promises = Object.values(this.clickHandlers).map(async (handler) => { // Check if the handler is disabled for the site. const disabled = await this.isFeatureDisabled(handler, notification.site); if (disabled) { return; } // Check if the handler handles the notification. const handles = await handler.handles(notification); if (handles) { handlers.push(handler); } }); await CoreUtils.ignoreErrors(CoreUtils.allPromises(promises)); // Sort by priority. handlers = handlers.sort((a, b) => (a.priority || 0) <= (b.priority || 0) ? 1 : -1); // Execute the first one. await handlers[0]?.handleClick(notification); } /** * Check if a handler's feature is disabled for a certain site. * * @param handler Handler to check. * @param siteId The site ID to check. * @returns Promise resolved with boolean: whether the handler feature is disabled. */ protected async isFeatureDisabled(handler: CorePushNotificationsClickHandler, siteId?: string): Promise { if (!siteId) { // Notification doesn't belong to a site. Assume all handlers are enabled. return false; } else if (handler.featureName) { // Check if the feature is disabled. return CoreSites.isFeatureDisabled(handler.featureName, siteId); } else { return false; } } /** * Function called when a push notification is received in foreground (cannot tell when it's received in background). * Sends notification to all handlers. * * @param notification Notification received. */ received(notification: CorePushNotificationsNotificationBasicData): void { this.observables['receive'].next(notification); } /** * Register a push notifications observable for a certain event. Right now, only receive is supported. * let observer = pushNotificationsDelegate.on('receive').subscribe((notification) => { * ... * observer.unsuscribe(); * * @param eventName Only receive is permitted. * @returns Observer to subscribe. */ on(eventName: string): Subject { if (this.observables[eventName] === undefined) { const eventNames = Object.keys(this.observables).join(', '); this.logger.warn(`'${eventName}' event name is not allowed. Use one of the following: '${eventNames}'.`); return new Subject(); } return > this.observables[eventName]; } /** * Register a click handler. * * @param handler The handler to register. * @returns True if registered successfully, false otherwise. */ registerClickHandler(handler: CorePushNotificationsClickHandler): boolean { if (this.clickHandlers[handler.name] !== undefined) { this.logger.log(`Addon '${handler.name}' already registered`); return false; } this.logger.log(`Registered addon '${handler.name}'`); this.clickHandlers[handler.name] = handler; handler.priority = handler.priority || 0; return true; } /** * Register a push notifications handler for update badge counter. * * @param name Handler's name. */ registerCounterHandler(name: string): void { if (this.counterHandlers[name] === undefined) { this.logger.debug(`Registered handler '${name}' as badge counter handler.`); this.counterHandlers[name] = name; } else { this.logger.log(`Handler '${name}' as badge counter handler already registered.`); } } /** * Check if a counter handler is present. * * @param name Handler's name. * @returns If handler name is present. */ isCounterHandlerRegistered(name: string): boolean { return this.counterHandlers[name] !== undefined; } /** * Get all counter badge handlers. * * @returns with all the handler names. */ getCounterHandlers(): Record { return this.counterHandlers; } } export const CorePushNotificationsDelegate = makeSingleton(CorePushNotificationsDelegateService);