ag-grid-angular
Version:
AG Grid Angular Component
115 lines • 18 kB
JavaScript
import { ViewContainerRef } from '@angular/core';
import { Component, Injectable, inject } from '@angular/core';
import { BaseComponentWrapper, _removeFromParent } from 'ag-grid-community';
import * as i0 from "@angular/core";
// To speed up the removal of custom components we create a number of shards to contain them.
// Removing a single component calls a function within Angular called removeFromArray.
// This is a lot faster if the array is smaller.
export class AgComponentContainer {
constructor() {
this.vcr = inject(ViewContainerRef);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AgComponentContainer, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: AgComponentContainer, selector: "ag-component-container", ngImport: i0, template: '', isInline: true }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AgComponentContainer, decorators: [{
type: Component,
args: [{
selector: 'ag-component-container',
template: '',
}]
}] });
const NUM_SHARDS = 16;
let shardIdx = 0;
function createComponentContainers(vcr) {
const containerMap = new Map();
for (let i = 0; i < NUM_SHARDS; i++) {
const container = vcr.createComponent(AgComponentContainer);
containerMap.set(i, container);
_removeFromParent(container.location.nativeElement);
}
return containerMap;
}
/**
* These methods are called on a hot path for every row so we do not want to enter / exit NgZone each time.
* Also these methods should not be used to update the UI, so we don't need to run them inside Angular.
*/
const runOutsideMethods = new Set(['doesFilterPass', 'isFilterActive']);
export class AngularFrameworkComponentWrapper extends BaseComponentWrapper {
setViewContainerRef(viewContainerRef, angularFrameworkOverrides) {
this.viewContainerRef = viewContainerRef;
this.angularFrameworkOverrides = angularFrameworkOverrides;
}
createWrapper(OriginalConstructor) {
const angularFrameworkOverrides = this.angularFrameworkOverrides;
const that = this;
that.compShards ??= createComponentContainers(this.viewContainerRef);
class DynamicAgNg2Component extends BaseGuiComponent {
init(params) {
angularFrameworkOverrides.runInsideAngular(() => {
super.init(params);
this._componentRef.changeDetectorRef.detectChanges();
});
}
createComponent() {
return that.createComponent(OriginalConstructor);
}
hasMethod(name) {
return wrapper.getFrameworkComponentInstance()[name] != null;
}
callMethod(name, args) {
const componentRef = this.getFrameworkComponentInstance();
const methodCall = componentRef[name];
if (runOutsideMethods.has(name)) {
return methodCall.apply(componentRef, args);
}
return angularFrameworkOverrides.runInsideAngular(() => methodCall.apply(componentRef, args));
}
addMethod(name, callback) {
wrapper[name] = callback;
}
}
const wrapper = new DynamicAgNg2Component();
return wrapper;
}
createComponent(componentType) {
shardIdx = (shardIdx + 1) % NUM_SHARDS;
const container = this.compShards.get(shardIdx);
return container.instance.vcr.createComponent(componentType);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AngularFrameworkComponentWrapper, deps: null, target: i0.ɵɵFactoryTarget.Injectable }); }
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AngularFrameworkComponentWrapper }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AngularFrameworkComponentWrapper, decorators: [{
type: Injectable
}] });
class BaseGuiComponent {
init(params) {
this._params = params;
this._componentRef = this.createComponent();
this._agAwareComponent = this._componentRef.instance;
this._frameworkComponentInstance = this._componentRef.instance;
this._eGui = this._componentRef.location.nativeElement;
// Angular appends the component to the DOM, so remove it
_removeFromParent(this._eGui);
this._agAwareComponent.agInit(this._params);
}
getGui() {
return this._eGui;
}
/** `getGui()` returns the `ng-component` element. This returns the actual root element. */
getRootElement() {
const firstChild = this._eGui.firstChild;
return firstChild;
}
destroy() {
if (this._frameworkComponentInstance && typeof this._frameworkComponentInstance.destroy === 'function') {
this._frameworkComponentInstance.destroy();
}
this._componentRef?.destroy();
}
getFrameworkComponentInstance() {
return this._frameworkComponentInstance;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"angularFrameworkComponentWrapper.js","sourceRoot":"","sources":["../../../../projects/ag-grid-angular/src/lib/angularFrameworkComponentWrapper.ts"],"names":[],"mappings":"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","sourcesContent":["import type { ComponentRef } from '@angular/core';\nimport { ViewContainerRef } from '@angular/core';\nimport { Component, Injectable, inject } from '@angular/core';\n\nimport type { FrameworkComponentWrapper, IFilter, WrappableInterface } from 'ag-grid-community';\nimport { BaseComponentWrapper, _removeFromParent } from 'ag-grid-community';\n\nimport type { AngularFrameworkOverrides } from './angularFrameworkOverrides';\nimport type { AgFrameworkComponent } from './interfaces';\n\n// To speed up the removal of custom components we create a number of shards to contain them.\n// Removing a single component calls a function within Angular called removeFromArray.\n// This is a lot faster if the array is smaller.\n@Component({\n    selector: 'ag-component-container',\n    template: '',\n})\nexport class AgComponentContainer {\n    public vcr = inject(ViewContainerRef);\n}\nconst NUM_SHARDS = 16;\nlet shardIdx = 0;\n\nfunction createComponentContainers(vcr: ViewContainerRef): Map<number, ComponentRef<AgComponentContainer>> {\n    const containerMap = new Map<number, ComponentRef<AgComponentContainer>>();\n    for (let i = 0; i < NUM_SHARDS; i++) {\n        const container = vcr.createComponent(AgComponentContainer);\n        containerMap.set(i, container);\n        _removeFromParent(container.location.nativeElement);\n    }\n    return containerMap;\n}\n\n/**\n * These methods are called on a hot path for every row so we do not want to enter / exit NgZone each time.\n * Also these methods should not be used to update the UI, so we don't need to run them inside Angular.\n */\nconst runOutsideMethods = new Set<keyof IFilter>(['doesFilterPass', 'isFilterActive']);\n\n@Injectable()\nexport class AngularFrameworkComponentWrapper\n    extends BaseComponentWrapper<WrappableInterface>\n    implements FrameworkComponentWrapper\n{\n    private viewContainerRef: ViewContainerRef;\n    private angularFrameworkOverrides: AngularFrameworkOverrides;\n    private compShards: Map<number, ComponentRef<AgComponentContainer>>;\n\n    public setViewContainerRef(\n        viewContainerRef: ViewContainerRef,\n        angularFrameworkOverrides: AngularFrameworkOverrides\n    ) {\n        this.viewContainerRef = viewContainerRef;\n        this.angularFrameworkOverrides = angularFrameworkOverrides;\n    }\n\n    protected createWrapper(OriginalConstructor: { new (): any }): WrappableInterface {\n        const angularFrameworkOverrides = this.angularFrameworkOverrides;\n        const that = this;\n        that.compShards ??= createComponentContainers(this.viewContainerRef);\n\n        class DynamicAgNg2Component\n            extends BaseGuiComponent<any, AgFrameworkComponent<any>>\n            implements WrappableInterface\n        {\n            override init(params: any): void {\n                angularFrameworkOverrides.runInsideAngular(() => {\n                    super.init(params);\n                    this._componentRef.changeDetectorRef.detectChanges();\n                });\n            }\n\n            protected createComponent(): ComponentRef<AgFrameworkComponent<any>> {\n                return that.createComponent(OriginalConstructor);\n            }\n\n            hasMethod(name: string): boolean {\n                return wrapper.getFrameworkComponentInstance()[name] != null;\n            }\n\n            callMethod(name: string, args: IArguments): void {\n                const componentRef = this.getFrameworkComponentInstance();\n                const methodCall = componentRef[name];\n\n                if (runOutsideMethods.has(name as any)) {\n                    return methodCall.apply(componentRef, args);\n                }\n                return angularFrameworkOverrides.runInsideAngular(() => methodCall.apply(componentRef, args));\n            }\n\n            addMethod(name: string, callback: (...args: any[]) => any): void {\n                (wrapper as any)[name] = callback;\n            }\n        }\n        const wrapper = new DynamicAgNg2Component();\n        return wrapper;\n    }\n\n    public createComponent<T>(componentType: { new (...args: any[]): T }): ComponentRef<T> {\n        shardIdx = (shardIdx + 1) % NUM_SHARDS;\n        const container = this.compShards.get(shardIdx)!;\n        return container.instance.vcr.createComponent(componentType);\n    }\n}\n\nabstract class BaseGuiComponent<P, T extends AgFrameworkComponent<P>> {\n    protected _params: P;\n    protected _eGui: HTMLElement;\n    protected _componentRef: ComponentRef<T>;\n    protected _agAwareComponent: T;\n    protected _frameworkComponentInstance: any; // the users component - for accessing methods they create\n\n    protected init(params: P): void {\n        this._params = params;\n\n        this._componentRef = this.createComponent();\n        this._agAwareComponent = this._componentRef.instance;\n        this._frameworkComponentInstance = this._componentRef.instance;\n        this._eGui = this._componentRef.location.nativeElement;\n        // Angular appends the component to the DOM, so remove it\n        _removeFromParent(this._eGui);\n\n        this._agAwareComponent.agInit(this._params);\n    }\n\n    public getGui(): HTMLElement {\n        return this._eGui;\n    }\n\n    /** `getGui()` returns the `ng-component` element. This returns the actual root element. */\n    public getRootElement(): HTMLElement {\n        const firstChild = this._eGui.firstChild;\n        return firstChild as HTMLElement;\n    }\n\n    public destroy(): void {\n        if (this._frameworkComponentInstance && typeof this._frameworkComponentInstance.destroy === 'function') {\n            this._frameworkComponentInstance.destroy();\n        }\n        this._componentRef?.destroy();\n    }\n\n    public getFrameworkComponentInstance(): any {\n        return this._frameworkComponentInstance;\n    }\n\n    protected abstract createComponent(): ComponentRef<T>;\n}\n"]}