@dotglitch/ngx-common 1.1.43 → 1.1.44

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,4 +1,4 @@
1
- import { AfterViewInit, ApplicationRef, Injector, OnChanges, OnDestroy, SimpleChanges, Type, ViewContainerRef } from '@angular/core';
1
+ import { AfterViewInit, ApplicationRef, Injector, NgZone, OnChanges, OnDestroy, SimpleChanges, Type, ViewContainerRef } from '@angular/core';
2
2
  import * as React from 'react';
3
3
  import { ThemeService } from '../../services/theme.service';
4
4
  import * as i0 from "@angular/core";
@@ -11,8 +11,9 @@ import * as i0 from "@angular/core";
11
11
  * `override readonly ngReactComponent = ReactFlowWrappableComponent;`
12
12
  */
13
13
  export declare class ReactMagicWrapperComponent implements OnChanges, OnDestroy, AfterViewInit {
14
- private ngContainer;
15
- private ngTheme;
14
+ private readonly ngContainer;
15
+ private readonly ngTheme;
16
+ private readonly ngZone;
16
17
  /**
17
18
  * Wrap an angular component inside of a React memo object.
18
19
  * Will attempt to bind @Input and @Output properties if provided,
@@ -39,7 +40,7 @@ export declare class ReactMagicWrapperComponent implements OnChanges, OnDestroy,
39
40
  private _root;
40
41
  theme: string;
41
42
  private ngSubscriptions;
42
- constructor(ngContainer: ViewContainerRef, ngTheme: ThemeService);
43
+ constructor(ngContainer: ViewContainerRef, ngTheme: ThemeService, ngZone: NgZone);
43
44
  ngOnInit(): void;
44
45
  ngOnChanges(changes?: SimpleChanges): void;
45
46
  ngAfterViewInit(): void;
@@ -76,9 +76,10 @@ export class ReactMagicWrapperComponent {
76
76
  // associate things together.
77
77
  return React.createElement('div', {}, React.createElement("div", { id }), ...additionalChildren);
78
78
  }); }
79
- constructor(ngContainer, ngTheme) {
79
+ constructor(ngContainer, ngTheme, ngZone) {
80
80
  this.ngContainer = ngContainer;
81
81
  this.ngTheme = ngTheme;
82
+ this.ngZone = ngZone;
82
83
  this.ngSubscriptions = [
83
84
  this.ngTheme.subscribe(t => {
84
85
  this.theme = t;
@@ -103,25 +104,27 @@ export class ReactMagicWrapperComponent {
103
104
  _render() {
104
105
  if (!this.ngReactComponent)
105
106
  return;
106
- if (!this._root) {
107
- this._root = createRoot(this.ngContainer.element.nativeElement);
108
- }
109
- // List all keys that do not start with `_` nor `ng`
110
- const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));
111
- // Get all property keys from the class
112
- const propKeys = keys.filter(k => !k.startsWith("on"));
113
- // Get all event handler keys from the class
114
- const evtKeys = keys.filter(k => k.startsWith("on"));
115
- const props = {};
116
- // Project all key properties onto `props`
117
- propKeys.forEach(k => props[k] = this[k]);
118
- // Bind all event handlers.
119
- // ! important Angular uses EventEmitter, React uses
120
- // a different method of event binding
121
- evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
122
- this._root.render(React.createElement(this.ngReactComponent, { props: props }));
107
+ this.ngZone.runOutsideAngular(() => {
108
+ if (!this._root) {
109
+ this._root = createRoot(this.ngContainer.element.nativeElement);
110
+ }
111
+ // List all keys that do not start with `_` nor `ng`
112
+ const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));
113
+ // Get all property keys from the class
114
+ const propKeys = keys.filter(k => !k.startsWith("on"));
115
+ // Get all event handler keys from the class
116
+ const evtKeys = keys.filter(k => k.startsWith("on"));
117
+ const props = {};
118
+ // Project all key properties onto `props`
119
+ propKeys.forEach(k => props[k] = this[k]);
120
+ // Bind all event handlers.
121
+ // ! important Angular uses EventEmitter, React uses
122
+ // a different method of event binding
123
+ evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
124
+ this._root.render(React.createElement(this.ngReactComponent, { props: props }));
125
+ });
123
126
  }
124
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactMagicWrapperComponent, deps: [{ token: i0.ViewContainerRef }, { token: i1.ThemeService }], target: i0.ɵɵFactoryTarget.Component }); }
127
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactMagicWrapperComponent, deps: [{ token: i0.ViewContainerRef }, { token: i1.ThemeService }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
125
128
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.2", type: ReactMagicWrapperComponent, isStandalone: true, selector: "app-react-magic-wrapper", usesOnChanges: true, ngImport: i0, template: ``, isInline: true }); }
126
129
  }
127
130
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactMagicWrapperComponent, decorators: [{
@@ -131,5 +134,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImpor
131
134
  template: ``,
132
135
  standalone: true
133
136
  }]
134
- }], ctorParameters: () => [{ type: i0.ViewContainerRef }, { type: i1.ThemeService }] });
135
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"react-magic-wrapper.component.js","sourceRoot":"","sources":["../../../../../packages/common/src/components/react-magic-wrapper/react-magic-wrapper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ApplicationRef, Component, ComponentFactoryResolver, EnvironmentInjector, EventEmitter, Injector, OnChanges, OnDestroy, SimpleChanges, Type, ViewContainerRef, ViewRef, createComponent } from '@angular/core';\nimport * as React from 'react';\nimport { createRoot, Root } from 'react-dom/client';\n\nimport { ThemeService } from '../../services/theme.service';\nimport { ulid } from 'ulidx';\nimport { Subscription } from 'rxjs';\n\n/**\n * Extend this component to automatically generate\n * bindings to a React component.\n *\n * ! You _must_ override the property `ngReactComponent`\n * Failure to do so will result in errors\n * `override readonly ngReactComponent = ReactFlowWrappableComponent;`\n */\n@Component({\n    selector: 'app-react-magic-wrapper',\n    template: ``,\n    standalone: true\n})\nexport class ReactMagicWrapperComponent implements OnChanges, OnDestroy, AfterViewInit {\n\n    /**\n     * Wrap an angular component inside of a React memo object.\n     * Will attempt to bind @Input and @Output properties if provided,\n     * and will bind the react arguments directly as @Input properties.\n     *\n     * @experimental\n     * @param componentClass Angular component\n     * @param envInjector    An `EnvironmentInjector` instance to be used for the component\n     * @param injector       An `ElementInjector` instance\n     * @param _inputs\n     * @param _outputs\n     * @returns\n     */\n    static WrapAngularComponent = (\n        componentClass: Type<any>,\n        appRef: Omit<ApplicationRef, '_runningTick'>,\n        injector: Injector,\n        _inputs: { [key: string]: any } = {},\n        _outputs: { [key: string]: Function } = {},\n        additionalChildren: React.ReactNode[] = []\n    ) => React.memo((args) => {\n\n        const id = ulid();\n        React.useEffect(() => {\n            const component = createComponent(componentClass, {\n                environmentInjector: appRef.injector,\n                elementInjector: injector,\n                hostElement: document.getElementById(id)\n            });\n\n            appRef.attachView(component.hostView);\n            // @ts-ignore\n            // component.hostView = hostView;\n\n            Object.assign(_inputs, args);\n\n            const { inputs, outputs } = componentClass['ɵcmp'];\n\n            // Returns a list of entries that need to be set\n            // This makes it so that unnecessary setters are not invoked.\n            const updated = Object.entries(inputs).filter(([parentKey, childKey]: [string, string]) => {\n                return component.instance[childKey] != _inputs[parentKey];\n            });\n\n            updated.forEach(([parentKey, childKey]: [string, string]) => {\n                if (_inputs.hasOwnProperty(parentKey))\n                    component.instance[childKey] = _inputs[parentKey];\n            });\n\n            const outputSubscriptions: { [key: string]: Subscription } = {};\n            // Get a list of unregistered outputs\n            const newOutputs = Object.entries(outputs).filter(([parentKey, childKey]: [string, string]) => {\n                return !outputSubscriptions[parentKey];\n            });\n\n            // Reverse bind via subscription\n            newOutputs.forEach(([parentKey, childKey]: [string, string]) => {\n                if (_outputs.hasOwnProperty(parentKey)) {\n                    const target: EventEmitter<unknown> = component.instance[childKey];\n                    const outputs = _outputs;\n\n                    const sub = target.subscribe(outputs[parentKey]); // Subscription\n\n                    outputSubscriptions[parentKey] = sub;\n                }\n            });\n\n            // Wrap the destroy method to safely release the subscriptions\n            const originalDestroy = component.onDestroy?.bind(component);\n            component.onDestroy = (cb) => {\n                Object.values(outputSubscriptions).forEach(s => s.unsubscribe());\n                originalDestroy?.(cb);\n            }\n\n            component.changeDetectorRef.detectChanges();\n        }, []);\n\n        // Create a container for the wrapped element and\n        // all of the siblings we will inject into it's container.\n        // The double wrapper is necessary for React's mechanisms to properly\n        // associate things together.\n        return React.createElement('div',\n            {},\n            React.createElement(\"div\", { id }),\n            ...additionalChildren\n        );\n    });\n\n    /**\n     * The react component to be wrapped.\n     * ! Must be overridden for this wrapper to work\n     */\n    ngReactComponent: React.FunctionComponent<any> | React.ComponentClass<any> | string;\n\n    private _root: Root;\n    public theme: string;\n    private ngSubscriptions = [\n        this.ngTheme.subscribe(t => {\n            this.theme = t;\n            this.ngOnChanges();\n        })\n    ];\n\n    constructor(\n        private ngContainer: ViewContainerRef,\n        private ngTheme: ThemeService\n    ) {\n    }\n\n    ngOnInit() {\n        if (!this.ngReactComponent)\n            throw new Error(\"ReactMagicWrapperComponent cannot start without a provided ngReactComponent!\");\n    }\n\n    ngOnChanges(changes?: SimpleChanges): void {\n        this._render();\n    }\n\n    ngAfterViewInit() {\n        this._render();\n    }\n\n    ngOnDestroy() {\n        this._root.unmount();\n        this.ngSubscriptions.forEach(s => s.unsubscribe());\n    }\n\n    private _render() {\n        if (!this.ngReactComponent) return;\n        if (!this._root) {\n            this._root = createRoot(this.ngContainer.element.nativeElement);\n        }\n\n        // List all keys that do not start with `_` nor `ng`\n        const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));\n\n        // Get all property keys from the class\n        const propKeys = keys.filter(k => !k.startsWith(\"on\"));\n        // Get all event handler keys from the class\n        const evtKeys = keys.filter(k => k.startsWith(\"on\"));\n\n        const props = {};\n        // Project all key properties onto `props`\n        propKeys.forEach(k => props[k] = this[k]);\n\n        // Bind all event handlers.\n        // ! important Angular uses EventEmitter, React uses\n        // a different method of event binding\n        evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));\n\n        this._root.render(React.createElement(this.ngReactComponent, { props: props as any }));\n    }\n}\n"]}
137
+ }], ctorParameters: () => [{ type: i0.ViewContainerRef }, { type: i1.ThemeService }, { type: i0.NgZone }] });
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"react-magic-wrapper.component.js","sourceRoot":"","sources":["../../../../../packages/common/src/components/react-magic-wrapper/react-magic-wrapper.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ApplicationRef, Component, ComponentFactoryResolver, EnvironmentInjector, EventEmitter, Injector, NgZone, OnChanges, OnDestroy, SimpleChanges, Type, ViewContainerRef, ViewRef, createComponent } from '@angular/core';\nimport * as React from 'react';\nimport { createRoot, Root } from 'react-dom/client';\n\nimport { ThemeService } from '../../services/theme.service';\nimport { ulid } from 'ulidx';\nimport { Subscription } from 'rxjs';\n\n/**\n * Extend this component to automatically generate\n * bindings to a React component.\n *\n * ! You _must_ override the property `ngReactComponent`\n * Failure to do so will result in errors\n * `override readonly ngReactComponent = ReactFlowWrappableComponent;`\n */\n@Component({\n    selector: 'app-react-magic-wrapper',\n    template: ``,\n    standalone: true\n})\nexport class ReactMagicWrapperComponent implements OnChanges, OnDestroy, AfterViewInit {\n\n    /**\n     * Wrap an angular component inside of a React memo object.\n     * Will attempt to bind @Input and @Output properties if provided,\n     * and will bind the react arguments directly as @Input properties.\n     *\n     * @experimental\n     * @param componentClass Angular component\n     * @param envInjector    An `EnvironmentInjector` instance to be used for the component\n     * @param injector       An `ElementInjector` instance\n     * @param _inputs\n     * @param _outputs\n     * @returns\n     */\n    static WrapAngularComponent = (\n        componentClass: Type<any>,\n        appRef: Omit<ApplicationRef, '_runningTick'>,\n        injector: Injector,\n        _inputs: { [key: string]: any } = {},\n        _outputs: { [key: string]: Function } = {},\n        additionalChildren: React.ReactNode[] = []\n    ) => React.memo((args) => {\n\n        const id = ulid();\n        React.useEffect(() => {\n            const component = createComponent(componentClass, {\n                environmentInjector: appRef.injector,\n                elementInjector: injector,\n                hostElement: document.getElementById(id)\n            });\n\n            appRef.attachView(component.hostView);\n            // @ts-ignore\n            // component.hostView = hostView;\n\n            Object.assign(_inputs, args);\n\n            const { inputs, outputs } = componentClass['ɵcmp'];\n\n            // Returns a list of entries that need to be set\n            // This makes it so that unnecessary setters are not invoked.\n            const updated = Object.entries(inputs).filter(([parentKey, childKey]: [string, string]) => {\n                return component.instance[childKey] != _inputs[parentKey];\n            });\n\n            updated.forEach(([parentKey, childKey]: [string, string]) => {\n                if (_inputs.hasOwnProperty(parentKey))\n                    component.instance[childKey] = _inputs[parentKey];\n            });\n\n            const outputSubscriptions: { [key: string]: Subscription } = {};\n            // Get a list of unregistered outputs\n            const newOutputs = Object.entries(outputs).filter(([parentKey, childKey]: [string, string]) => {\n                return !outputSubscriptions[parentKey];\n            });\n\n            // Reverse bind via subscription\n            newOutputs.forEach(([parentKey, childKey]: [string, string]) => {\n                if (_outputs.hasOwnProperty(parentKey)) {\n                    const target: EventEmitter<unknown> = component.instance[childKey];\n                    const outputs = _outputs;\n\n                    const sub = target.subscribe(outputs[parentKey]); // Subscription\n\n                    outputSubscriptions[parentKey] = sub;\n                }\n            });\n\n            // Wrap the destroy method to safely release the subscriptions\n            const originalDestroy = component.onDestroy?.bind(component);\n            component.onDestroy = (cb) => {\n                Object.values(outputSubscriptions).forEach(s => s.unsubscribe());\n                originalDestroy?.(cb);\n            }\n\n            component.changeDetectorRef.detectChanges();\n        }, []);\n\n        // Create a container for the wrapped element and\n        // all of the siblings we will inject into it's container.\n        // The double wrapper is necessary for React's mechanisms to properly\n        // associate things together.\n        return React.createElement('div',\n            {},\n            React.createElement(\"div\", { id }),\n            ...additionalChildren\n        );\n    });\n\n    /**\n     * The react component to be wrapped.\n     * ! Must be overridden for this wrapper to work\n     */\n    ngReactComponent: React.FunctionComponent<any> | React.ComponentClass<any> | string;\n\n    private _root: Root;\n    public theme: string;\n    private ngSubscriptions = [\n        this.ngTheme.subscribe(t => {\n            this.theme = t;\n            this.ngOnChanges();\n        })\n    ];\n\n    constructor(\n        private readonly ngContainer: ViewContainerRef,\n        private readonly ngTheme: ThemeService,\n        private readonly ngZone: NgZone\n    ) {\n    }\n\n    ngOnInit() {\n        if (!this.ngReactComponent)\n            throw new Error(\"ReactMagicWrapperComponent cannot start without a provided ngReactComponent!\");\n    }\n\n    ngOnChanges(changes?: SimpleChanges): void {\n        this._render();\n    }\n\n    ngAfterViewInit() {\n        this._render();\n    }\n\n    ngOnDestroy() {\n        this._root.unmount();\n        this.ngSubscriptions.forEach(s => s.unsubscribe());\n    }\n\n    private _render() {\n        if (!this.ngReactComponent) return;\n\n        this.ngZone.runOutsideAngular(() => {\n            if (!this._root) {\n                this._root = createRoot(this.ngContainer.element.nativeElement);\n            }\n\n            // List all keys that do not start with `_` nor `ng`\n            const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));\n\n            // Get all property keys from the class\n            const propKeys = keys.filter(k => !k.startsWith(\"on\"));\n            // Get all event handler keys from the class\n            const evtKeys = keys.filter(k => k.startsWith(\"on\"));\n\n            const props = {};\n            // Project all key properties onto `props`\n            propKeys.forEach(k => props[k] = this[k]);\n\n            // Bind all event handlers.\n            // ! important Angular uses EventEmitter, React uses\n            // a different method of event binding\n            evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));\n\n            this._root.render(React.createElement(this.ngReactComponent, { props: props as any }));\n        })\n    }\n}\n"]}
@@ -11671,9 +11671,10 @@ class ReactMagicWrapperComponent {
11671
11671
  // associate things together.
11672
11672
  return React.createElement('div', {}, React.createElement("div", { id }), ...additionalChildren);
11673
11673
  }); }
11674
- constructor(ngContainer, ngTheme) {
11674
+ constructor(ngContainer, ngTheme, ngZone) {
11675
11675
  this.ngContainer = ngContainer;
11676
11676
  this.ngTheme = ngTheme;
11677
+ this.ngZone = ngZone;
11677
11678
  this.ngSubscriptions = [
11678
11679
  this.ngTheme.subscribe(t => {
11679
11680
  this.theme = t;
@@ -11698,25 +11699,27 @@ class ReactMagicWrapperComponent {
11698
11699
  _render() {
11699
11700
  if (!this.ngReactComponent)
11700
11701
  return;
11701
- if (!this._root) {
11702
- this._root = createRoot(this.ngContainer.element.nativeElement);
11703
- }
11704
- // List all keys that do not start with `_` nor `ng`
11705
- const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));
11706
- // Get all property keys from the class
11707
- const propKeys = keys.filter(k => !k.startsWith("on"));
11708
- // Get all event handler keys from the class
11709
- const evtKeys = keys.filter(k => k.startsWith("on"));
11710
- const props = {};
11711
- // Project all key properties onto `props`
11712
- propKeys.forEach(k => props[k] = this[k]);
11713
- // Bind all event handlers.
11714
- // ! important Angular uses EventEmitter, React uses
11715
- // a different method of event binding
11716
- evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
11717
- this._root.render(React.createElement(this.ngReactComponent, { props: props }));
11718
- }
11719
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactMagicWrapperComponent, deps: [{ token: i0.ViewContainerRef }, { token: ThemeService }], target: i0.ɵɵFactoryTarget.Component }); }
11702
+ this.ngZone.runOutsideAngular(() => {
11703
+ if (!this._root) {
11704
+ this._root = createRoot(this.ngContainer.element.nativeElement);
11705
+ }
11706
+ // List all keys that do not start with `_` nor `ng`
11707
+ const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));
11708
+ // Get all property keys from the class
11709
+ const propKeys = keys.filter(k => !k.startsWith("on"));
11710
+ // Get all event handler keys from the class
11711
+ const evtKeys = keys.filter(k => k.startsWith("on"));
11712
+ const props = {};
11713
+ // Project all key properties onto `props`
11714
+ propKeys.forEach(k => props[k] = this[k]);
11715
+ // Bind all event handlers.
11716
+ // ! important Angular uses EventEmitter, React uses
11717
+ // a different method of event binding
11718
+ evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
11719
+ this._root.render(React.createElement(this.ngReactComponent, { props: props }));
11720
+ });
11721
+ }
11722
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactMagicWrapperComponent, deps: [{ token: i0.ViewContainerRef }, { token: ThemeService }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
11720
11723
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.2", type: ReactMagicWrapperComponent, isStandalone: true, selector: "app-react-magic-wrapper", usesOnChanges: true, ngImport: i0, template: ``, isInline: true }); }
11721
11724
  }
11722
11725
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactMagicWrapperComponent, decorators: [{
@@ -11726,7 +11729,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImpor
11726
11729
  template: ``,
11727
11730
  standalone: true
11728
11731
  }]
11729
- }], ctorParameters: () => [{ type: i0.ViewContainerRef }, { type: ThemeService }] });
11732
+ }], ctorParameters: () => [{ type: i0.ViewContainerRef }, { type: ThemeService }, { type: i0.NgZone }] });
11730
11733
 
11731
11734
  /*
11732
11735
  * Public API Surface of package