214 lines
6.5 KiB
TypeScript
214 lines
6.5 KiB
TypeScript
// (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, Type } from '@angular/core';
|
|
import { CoreSites } from '@services/sites';
|
|
import { CoreDelegate, CoreDelegateHandler } from '@classes/delegate';
|
|
import { CoreSite } from '@classes/sites/site';
|
|
import { Subject } from 'rxjs';
|
|
import { CoreCourseBlock } from '@features/course/services/course';
|
|
import { Params } from '@angular/router';
|
|
import { makeSingleton } from '@singletons';
|
|
import { CoreBlockDefaultHandler } from './handlers/default-block';
|
|
import { CoreNavigationOptions } from '@services/navigator';
|
|
import type { ICoreBlockComponent } from '@features/block/classes/base-block-component';
|
|
|
|
/**
|
|
* Interface that all blocks must implement.
|
|
*/
|
|
export interface CoreBlockHandler extends CoreDelegateHandler {
|
|
/**
|
|
* Name of the block the handler supports. E.g. 'activity_modules'.
|
|
*/
|
|
blockName: string;
|
|
|
|
/**
|
|
* Returns the data needed to render the block.
|
|
*
|
|
* @param block The block to render.
|
|
* @param contextLevel The context where the block will be used.
|
|
* @param instanceId The instance ID associated with the context level.
|
|
* @returns Data or promise resolved with the data.
|
|
*/
|
|
getDisplayData?(
|
|
block: CoreCourseBlock,
|
|
contextLevel: string,
|
|
instanceId: number,
|
|
): undefined | CoreBlockHandlerData | Promise<CoreBlockHandlerData>;
|
|
}
|
|
|
|
/**
|
|
* Data needed to render a block. It's returned by the handler.
|
|
*/
|
|
export interface CoreBlockHandlerData {
|
|
/**
|
|
* Title to display for the block.
|
|
*/
|
|
title: string;
|
|
|
|
/**
|
|
* Class to add to the displayed block.
|
|
*/
|
|
class?: string;
|
|
|
|
/**
|
|
* The component to render the contents of the block.
|
|
* It's recommended to return the class of the component, but you can also return an instance of the component.
|
|
*/
|
|
component: Type<ICoreBlockComponent>;
|
|
|
|
/**
|
|
* Data to pass to the component. All the properties in this object will be passed to the component as inputs.
|
|
*/
|
|
componentData?: Record<string | number, unknown>;
|
|
|
|
/**
|
|
* Link to go when showing only title.
|
|
*/
|
|
link?: string;
|
|
|
|
/**
|
|
* Params of the link.
|
|
*/
|
|
linkParams?: Params;
|
|
|
|
/**
|
|
* Navigation options.
|
|
*/
|
|
navOptions?: CoreNavigationOptions;
|
|
}
|
|
|
|
/**
|
|
* Delegate to register block handlers.
|
|
*/
|
|
@Injectable({ providedIn: 'root' })
|
|
export class CoreBlockDelegateService extends CoreDelegate<CoreBlockHandler> {
|
|
|
|
protected handlerNameProperty = 'blockName';
|
|
|
|
protected featurePrefix = 'CoreBlockDelegate_';
|
|
|
|
blocksUpdateObservable: Subject<void>;
|
|
|
|
constructor(
|
|
protected defaultHandler: CoreBlockDefaultHandler,
|
|
) {
|
|
super('CoreBlockDelegate', true);
|
|
|
|
this.blocksUpdateObservable = new Subject<void>();
|
|
}
|
|
|
|
/**
|
|
* Check if blocks are disabled in a certain site.
|
|
*
|
|
* @param site Site. If not defined, use current site.
|
|
* @returns Whether it's disabled.
|
|
*/
|
|
areBlocksDisabledInSite(site?: CoreSite): boolean {
|
|
site = site || CoreSites.getCurrentSite();
|
|
|
|
return !!site && site.isFeatureDisabled('NoDelegate_SiteBlocks');
|
|
}
|
|
|
|
/**
|
|
* Check if blocks are disabled in a certain site for courses.
|
|
*
|
|
* @param site Site. If not defined, use current site.
|
|
* @returns Whether it's disabled.
|
|
*/
|
|
areBlocksDisabledInCourses(site?: CoreSite): boolean {
|
|
site = site || CoreSites.getCurrentSite();
|
|
|
|
return !!site && site.isFeatureDisabled('NoDelegate_CourseBlocks');
|
|
}
|
|
|
|
/**
|
|
* Check if blocks are disabled in a certain site.
|
|
*
|
|
* @param siteId Site Id. If not defined, use current site.
|
|
* @returns Promise resolved with true if disabled, rejected or resolved with false otherwise.
|
|
*/
|
|
async areBlocksDisabled(siteId?: string): Promise<boolean> {
|
|
const site = await CoreSites.getSite(siteId);
|
|
|
|
return this.areBlocksDisabledInSite(site);
|
|
}
|
|
|
|
/**
|
|
* Get the display data for a certain block.
|
|
*
|
|
* @param block The block to render.
|
|
* @param contextLevel The context where the block will be used.
|
|
* @param instanceId The instance ID associated with the context level.
|
|
* @returns Promise resolved with the display data.
|
|
*/
|
|
async getBlockDisplayData(
|
|
block: CoreCourseBlock,
|
|
contextLevel: string,
|
|
instanceId: number,
|
|
): Promise<CoreBlockHandlerData | undefined> {
|
|
return this.executeFunctionOnEnabled(
|
|
block.name,
|
|
'getDisplayData',
|
|
[block, contextLevel, instanceId],
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Check if any of the blocks in a list is supported.
|
|
*
|
|
* @param blocks The list of blocks.
|
|
* @returns Whether any of the blocks is supported.
|
|
*/
|
|
hasSupportedBlock(blocks: CoreCourseBlock[]): boolean {
|
|
blocks = blocks || [];
|
|
|
|
return !!blocks.find((block) => this.isBlockSupported(block.name));
|
|
}
|
|
|
|
/**
|
|
* Check if a block is supported.
|
|
*
|
|
* @param name Block "name". E.g. 'activity_modules'.
|
|
* @returns Whether it's supported.
|
|
*/
|
|
isBlockSupported(name: string): boolean {
|
|
return this.hasHandler(name, true);
|
|
}
|
|
|
|
/**
|
|
* Check if feature is enabled or disabled in the site, depending on the feature prefix and the handler name.
|
|
*
|
|
* @param handler Handler to check.
|
|
* @param site Site to check.
|
|
* @returns Whether is enabled or disabled in site.
|
|
*/
|
|
protected isFeatureDisabled(handler: CoreBlockHandler, site: CoreSite): boolean {
|
|
// Allow displaying my overview even if all blocks are disabled, to avoid having an empty My Courses.
|
|
return (this.areBlocksDisabledInSite(site) && handler.blockName !== 'myoverview') ||
|
|
super.isFeatureDisabled(handler, site);
|
|
}
|
|
|
|
/**
|
|
* Called when there are new block handlers available. Informs anyone who subscribed to the
|
|
* observable.
|
|
*/
|
|
updateData(): void {
|
|
this.blocksUpdateObservable.next();
|
|
}
|
|
|
|
}
|
|
|
|
export const CoreBlockDelegate = makeSingleton(CoreBlockDelegateService);
|