@dotglitch/ngx-common 1.1.46 → 1.1.48
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.
|
@@ -27,11 +27,19 @@ export declare class ReactMagicWrapperComponent implements OnChanges, OnDestroy,
|
|
|
27
27
|
* @param _outputs
|
|
28
28
|
* @returns
|
|
29
29
|
*/
|
|
30
|
-
static WrapAngularComponent: (
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
30
|
+
static WrapAngularComponent: ({ component, appRef, injector, ngZone, staticInputs, staticOutputs, additionalChildren: siblings }: {
|
|
31
|
+
component: Type<any>;
|
|
32
|
+
appRef: Omit<ApplicationRef, '_runningTick'>;
|
|
33
|
+
injector: Injector;
|
|
34
|
+
ngZone: NgZone;
|
|
35
|
+
staticInputs?: {
|
|
36
|
+
[key: string]: any;
|
|
37
|
+
};
|
|
38
|
+
staticOutputs?: {
|
|
39
|
+
[key: string]: Function;
|
|
40
|
+
};
|
|
41
|
+
additionalChildren?: React.ReactNode[];
|
|
42
|
+
}) => React.NamedExoticComponent<object>;
|
|
35
43
|
/**
|
|
36
44
|
* The react component to be wrapped.
|
|
37
45
|
* ! Must be overridden for this wrapper to work
|
|
@@ -26,55 +26,61 @@ export class ReactMagicWrapperComponent {
|
|
|
26
26
|
* @param _outputs
|
|
27
27
|
* @returns
|
|
28
28
|
*/
|
|
29
|
-
static { this.WrapAngularComponent = (
|
|
29
|
+
static { this.WrapAngularComponent = ({ component, appRef, injector, ngZone, staticInputs, staticOutputs, additionalChildren: siblings }) => React.memo((args) => {
|
|
30
30
|
const id = ulid();
|
|
31
31
|
React.useEffect(() => {
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
const
|
|
32
|
+
try {
|
|
33
|
+
const componentInstance = createComponent(component, {
|
|
34
|
+
environmentInjector: appRef.injector,
|
|
35
|
+
elementInjector: injector,
|
|
36
|
+
hostElement: document.getElementById(id)
|
|
37
|
+
});
|
|
38
|
+
appRef.attachView(componentInstance.hostView);
|
|
39
|
+
// @ts-ignore
|
|
40
|
+
// component.hostView = hostView;
|
|
41
|
+
Object.assign(staticInputs, args);
|
|
42
|
+
const { inputs, outputs } = component['ɵcmp'];
|
|
43
|
+
// Returns a list of entries that need to be set
|
|
44
|
+
// This makes it so that unnecessary setters are not invoked.
|
|
45
|
+
const updated = Object.entries(inputs).filter(([parentKey, childKey]) => {
|
|
46
|
+
return componentInstance.instance[childKey] != staticInputs[parentKey];
|
|
47
|
+
});
|
|
48
|
+
updated.forEach(([parentKey, childKey]) => {
|
|
49
|
+
if (staticInputs.hasOwnProperty(parentKey))
|
|
50
|
+
componentInstance.instance[childKey] = staticInputs[parentKey];
|
|
51
|
+
});
|
|
52
|
+
const outputSubscriptions = {};
|
|
53
|
+
// Get a list of unregistered outputs
|
|
54
|
+
const newOutputs = Object.entries(outputs).filter(([parentKey, childKey]) => {
|
|
55
|
+
return !outputSubscriptions[parentKey];
|
|
56
|
+
});
|
|
57
|
+
// Reverse bind via subscription
|
|
58
|
+
newOutputs.forEach(([parentKey, childKey]) => {
|
|
59
|
+
if (!staticOutputs.hasOwnProperty(parentKey))
|
|
60
|
+
return;
|
|
61
|
+
const target = componentInstance.instance[childKey];
|
|
62
|
+
const outputs = staticOutputs;
|
|
63
|
+
const sub = target.subscribe((...args) => {
|
|
64
|
+
// Run the callback in the provided zone
|
|
65
|
+
ngZone.run(() => {
|
|
66
|
+
outputs[parentKey](...args);
|
|
67
|
+
});
|
|
68
|
+
}); // Subscription
|
|
62
69
|
outputSubscriptions[parentKey] = sub;
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
70
|
+
});
|
|
71
|
+
// Wrap the destroy method to safely release the subscriptions
|
|
72
|
+
const originalDestroy = componentInstance.onDestroy?.bind(componentInstance);
|
|
73
|
+
componentInstance.onDestroy = (cb) => {
|
|
74
|
+
Object.values(outputSubscriptions).forEach(s => s.unsubscribe());
|
|
75
|
+
originalDestroy?.(cb);
|
|
76
|
+
};
|
|
77
|
+
componentInstance.changeDetectorRef.detectChanges();
|
|
78
|
+
}
|
|
79
|
+
catch (err) {
|
|
80
|
+
console.error(err);
|
|
81
|
+
}
|
|
72
82
|
}, []);
|
|
73
|
-
|
|
74
|
-
// all of the siblings we will inject into it's container.
|
|
75
|
-
// The double wrapper is necessary for React's mechanisms to properly
|
|
76
|
-
// associate things together.
|
|
77
|
-
return React.createElement('div', {}, React.createElement("div", { id }), ...additionalChildren);
|
|
83
|
+
return React.createElement("div", {}, React.createElement("div", { id }), ...siblings);
|
|
78
84
|
}); }
|
|
79
85
|
constructor(ngContainer, ngTheme, ngZone) {
|
|
80
86
|
this.ngContainer = ngContainer;
|
|
@@ -88,8 +94,10 @@ export class ReactMagicWrapperComponent {
|
|
|
88
94
|
];
|
|
89
95
|
}
|
|
90
96
|
ngOnInit() {
|
|
91
|
-
if (!this.ngReactComponent)
|
|
97
|
+
if (!this.ngReactComponent) {
|
|
98
|
+
console.error("NO");
|
|
92
99
|
throw new Error("ReactMagicWrapperComponent cannot start without a provided ngReactComponent!");
|
|
100
|
+
}
|
|
93
101
|
}
|
|
94
102
|
ngOnChanges(changes) {
|
|
95
103
|
this._render();
|
|
@@ -102,29 +110,37 @@ export class ReactMagicWrapperComponent {
|
|
|
102
110
|
this.ngSubscriptions.forEach(s => s.unsubscribe());
|
|
103
111
|
}
|
|
104
112
|
_render() {
|
|
105
|
-
if (!this.ngReactComponent)
|
|
113
|
+
if (!this.ngReactComponent) {
|
|
114
|
+
console.log("Render no component. May be context issue");
|
|
106
115
|
return;
|
|
116
|
+
}
|
|
117
|
+
;
|
|
107
118
|
this.ngZone.runOutsideAngular(() => {
|
|
108
|
-
|
|
109
|
-
|
|
119
|
+
try {
|
|
120
|
+
if (!this._root) {
|
|
121
|
+
this._root = createRoot(this.ngContainer.element.nativeElement);
|
|
122
|
+
}
|
|
123
|
+
// List all keys that do not start with `_` nor `ng`
|
|
124
|
+
const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));
|
|
125
|
+
// Get all property keys from the class
|
|
126
|
+
const propKeys = keys.filter(k => !k.startsWith("on"));
|
|
127
|
+
// Get all event handler keys from the class
|
|
128
|
+
const evtKeys = keys.filter(k => k.startsWith("on"));
|
|
129
|
+
const props = {};
|
|
130
|
+
// Project all key properties onto `props`
|
|
131
|
+
propKeys.forEach(k => props[k] = this[k]);
|
|
132
|
+
// Attempt to ensure no zone is lost during the event emitter fires
|
|
133
|
+
this.ngZone.runGuarded(() => {
|
|
134
|
+
// Bind all event handlers.
|
|
135
|
+
// ! important Angular uses EventEmitter, React uses
|
|
136
|
+
// a different method of event binding
|
|
137
|
+
evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
|
|
138
|
+
});
|
|
139
|
+
this._root.render(React.createElement(this.ngReactComponent, { props: props }));
|
|
140
|
+
}
|
|
141
|
+
catch (err) {
|
|
142
|
+
console.error(err);
|
|
110
143
|
}
|
|
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
|
-
// Attempt to ensure no zone is lost during the event emitter fires
|
|
121
|
-
this.ngZone.run(() => {
|
|
122
|
-
// Bind all event handlers.
|
|
123
|
-
// ! important Angular uses EventEmitter, React uses
|
|
124
|
-
// a different method of event binding
|
|
125
|
-
evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
|
|
126
|
-
});
|
|
127
|
-
this._root.render(React.createElement(this.ngReactComponent, { props: props }));
|
|
128
144
|
});
|
|
129
145
|
}
|
|
130
146
|
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 }); }
|
|
@@ -138,4 +154,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImpor
|
|
|
138
154
|
standalone: true
|
|
139
155
|
}]
|
|
140
156
|
}], ctorParameters: () => [{ type: i0.ViewContainerRef }, { type: i1.ThemeService }, { type: i0.NgZone }] });
|
|
141
|
-
//# 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            // Attempt to ensure no zone is lost during the event emitter fires\n            this.ngZone.run(() => {\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\n            this._root.render(React.createElement(this.ngReactComponent, { props: props as any }));\n        })\n    }\n}\n"]}
|
|
157
|
+
//# 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":"AAAA,OAAO,EAAiC,SAAS,EAAuJ,eAAe,EAAE,MAAM,eAAe,CAAC;AAC/O,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAC/B,OAAO,EAAE,UAAU,EAAQ,MAAM,kBAAkB,CAAC;AAGpD,OAAO,EAAE,IAAI,EAAE,MAAM,OAAO,CAAC;;;AAG7B;;;;;;;GAOG;AAMH,MAAM,OAAO,0BAA0B;IAEnC;;;;;;;;;;;;OAYG;aACI,yBAAoB,GAAG,CAAC,EAAE,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,kBAAkB,EAAE,QAAQ,EAQ9H,EAAE,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,EAAE;QAEtB,MAAM,EAAE,GAAG,IAAI,EAAE,CAAC;QAClB,KAAK,CAAC,SAAS,CAAC,GAAG,EAAE;YACjB,IAAI,CAAC;gBAED,MAAM,iBAAiB,GAAG,eAAe,CAAC,SAAS,EAAE;oBACjD,mBAAmB,EAAE,MAAM,CAAC,QAAQ;oBACpC,eAAe,EAAE,QAAQ;oBACzB,WAAW,EAAE,QAAQ,CAAC,cAAc,CAAC,EAAE,CAAC;iBAC3C,CAAC,CAAC;gBAEH,MAAM,CAAC,UAAU,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;gBAC9C,aAAa;gBACb,iCAAiC;gBAEjC,MAAM,CAAC,MAAM,CAAC,YAAY,EAAE,IAAI,CAAC,CAAC;gBAElC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;gBAE9C,gDAAgD;gBAChD,6DAA6D;gBAC7D,MAAM,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS,EAAE,QAAQ,CAAmB,EAAE,EAAE;oBACtF,OAAO,iBAAiB,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,YAAY,CAAC,SAAS,CAAC,CAAC;gBAC3E,CAAC,CAAC,CAAC;gBAEH,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EAAE,QAAQ,CAAmB,EAAE,EAAE;oBACxD,IAAI,YAAY,CAAC,cAAc,CAAC,SAAS,CAAC;wBACtC,iBAAiB,CAAC,QAAQ,CAAC,QAAQ,CAAC,GAAG,YAAY,CAAC,SAAS,CAAC,CAAC;gBACvE,CAAC,CAAC,CAAC;gBAEH,MAAM,mBAAmB,GAAoC,EAAE,CAAC;gBAChE,qCAAqC;gBACrC,MAAM,UAAU,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS,EAAE,QAAQ,CAAmB,EAAE,EAAE;oBAC1F,OAAO,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC;gBAC3C,CAAC,CAAC,CAAC;gBAEH,gCAAgC;gBAChC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EAAE,QAAQ,CAAmB,EAAE,EAAE;oBAC3D,IAAI,CAAC,aAAa,CAAC,cAAc,CAAC,SAAS,CAAC;wBAAE,OAAO;oBAErD,MAAM,MAAM,GAA0B,iBAAiB,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;oBAC3E,MAAM,OAAO,GAAG,aAAa,CAAC;oBAE9B,MAAM,GAAG,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC,GAAG,IAAI,EAAE,EAAE;wBACrC,wCAAwC;wBACxC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;4BACZ,OAAO,CAAC,SAAS,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC;wBAChC,CAAC,CAAC,CAAA;oBACN,CAAC,CAAC,CAAC,CAAC,eAAe;oBAEnB,mBAAmB,CAAC,SAAS,CAAC,GAAG,GAAG,CAAC;gBACzC,CAAC,CAAC,CAAC;gBAEH,8DAA8D;gBAC9D,MAAM,eAAe,GAAG,iBAAiB,CAAC,SAAS,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBAC7E,iBAAiB,CAAC,SAAS,GAAG,CAAC,EAAE,EAAE,EAAE;oBACjC,MAAM,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;oBACjE,eAAe,EAAE,CAAC,EAAE,CAAC,CAAC;gBAC1B,CAAC,CAAA;gBAED,iBAAiB,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;YACxD,CAAC;YACD,OAAM,GAAG,EAAE,CAAC;gBACR,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAA;YACtB,CAAC;QACL,CAAC,EAAE,EAAE,CAAC,CAAC;QAEP,OAAO,KAAK,CAAC,aAAa,CAAC,KAAK,EAAE,EAAG,EACjC,KAAK,CAAC,aAAa,CAAC,KAAK,EAAE,EAAE,EAAE,EAAE,CAAC,EAClC,GAAG,QAAQ,CACd,CAAA;IACL,CAAC,CAAC,AAhFyB,CAgFxB;IAiBH,YACqB,WAA6B,EAC7B,OAAqB,EACrB,MAAc;QAFd,gBAAW,GAAX,WAAW,CAAkB;QAC7B,YAAO,GAAP,OAAO,CAAc;QACrB,WAAM,GAAN,MAAM,CAAQ;QAV3B,oBAAe,GAAG;YACtB,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE;gBACvB,IAAI,CAAC,KAAK,GAAG,CAAC,CAAC;gBACf,IAAI,CAAC,WAAW,EAAE,CAAC;YACvB,CAAC,CAAC;SACL,CAAC;IAOF,CAAC;IAED,QAAQ;QAEJ,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACzB,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;YACnB,MAAM,IAAI,KAAK,CAAC,8EAA8E,CAAC,CAAC;QACpG,CAAC;IACL,CAAC;IAED,WAAW,CAAC,OAAuB;QAC/B,IAAI,CAAC,OAAO,EAAE,CAAC;IACnB,CAAC;IAED,eAAe;QACX,IAAI,CAAC,OAAO,EAAE,CAAC;IACnB,CAAC;IAED,WAAW;QACP,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;QACrB,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;IACvD,CAAC;IAEO,OAAO;QACX,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACzB,OAAO,CAAC,GAAG,CAAC,2CAA2C,CAAC,CAAA;YACxD,OAAM;QACV,CAAC;QAAA,CAAC;QAEF,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,GAAG,EAAE;YAC/B,IAAI,CAAC;gBAED,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;oBACd,IAAI,CAAC,KAAK,GAAG,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;gBACpE,CAAC;gBAED,oDAAoD;gBACpD,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;gBAEjE,uCAAuC;gBACvC,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC;gBACvD,4CAA4C;gBAC5C,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC;gBAErD,MAAM,KAAK,GAAG,EAAE,CAAC;gBACjB,0CAA0C;gBAC1C,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC;gBAE1C,mEAAmE;gBACnE,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;oBACxB,2BAA2B;oBAC3B,oDAAoD;oBACpD,sCAAsC;oBACtC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;gBACrE,CAAC,CAAC,CAAA;gBAEF,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,aAAa,CAAC,IAAI,CAAC,gBAAgB,EAAE,EAAE,KAAK,EAAE,KAAY,EAAE,CAAC,CAAC,CAAC;YAC3F,CAAC;YACD,OAAM,GAAG,EAAE,CAAC;gBACR,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAA;YACtB,CAAC;QACL,CAAC,CAAC,CAAA;IACN,CAAC;8GAnLQ,0BAA0B;kGAA1B,0BAA0B,wGAHzB,EAAE;;2FAGH,0BAA0B;kBALtC,SAAS;mBAAC;oBACP,QAAQ,EAAE,yBAAyB;oBACnC,QAAQ,EAAE,EAAE;oBACZ,UAAU,EAAE,IAAI;iBACnB","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 = ({ component, appRef, injector, ngZone, staticInputs, staticOutputs, additionalChildren: siblings}: {\n        component: Type<any>,\n        appRef: Omit<ApplicationRef, '_runningTick'>,\n        injector: Injector,\n        ngZone: NgZone,\n        staticInputs?: { [key: string]: any },\n        staticOutputs?: { [key: string]: Function },\n        additionalChildren?: React.ReactNode[]\n    }) => React.memo((args) => {\n\n        const id = ulid();\n        React.useEffect(() => {\n            try {\n\n                const componentInstance = createComponent(component, {\n                    environmentInjector: appRef.injector,\n                    elementInjector: injector,\n                    hostElement: document.getElementById(id)\n                });\n\n                appRef.attachView(componentInstance.hostView);\n                // @ts-ignore\n                // component.hostView = hostView;\n\n                Object.assign(staticInputs, args);\n\n                const { inputs, outputs } = component['ɵ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 componentInstance.instance[childKey] != staticInputs[parentKey];\n                });\n\n                updated.forEach(([parentKey, childKey]: [string, string]) => {\n                    if (staticInputs.hasOwnProperty(parentKey))\n                        componentInstance.instance[childKey] = staticInputs[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 (!staticOutputs.hasOwnProperty(parentKey)) return;\n\n                    const target: EventEmitter<unknown> = componentInstance.instance[childKey];\n                    const outputs = staticOutputs;\n\n                    const sub = target.subscribe((...args) => {\n                        // Run the callback in the provided zone\n                        ngZone.run(() => {\n                            outputs[parentKey](...args);\n                        })\n                    }); // Subscription\n\n                    outputSubscriptions[parentKey] = sub;\n                });\n\n                // Wrap the destroy method to safely release the subscriptions\n                const originalDestroy = componentInstance.onDestroy?.bind(componentInstance);\n                componentInstance.onDestroy = (cb) => {\n                    Object.values(outputSubscriptions).forEach(s => s.unsubscribe());\n                    originalDestroy?.(cb);\n                }\n\n                componentInstance.changeDetectorRef.detectChanges();\n            }\n            catch(err) {\n                console.error(err)\n            }\n        }, []);\n\n        return React.createElement(\"div\", { },\n            React.createElement(\"div\", { id }),\n            ...siblings\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\n        if (!this.ngReactComponent) {\n            console.error(\"NO\")\n            throw new Error(\"ReactMagicWrapperComponent cannot start without a provided ngReactComponent!\");\n        }\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) {\n            console.log(\"Render no component. May be context issue\")\n            return\n        };\n\n        this.ngZone.runOutsideAngular(() => {\n            try {\n\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                // Attempt to ensure no zone is lost during the event emitter fires\n                this.ngZone.runGuarded(() => {\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\n                this._root.render(React.createElement(this.ngReactComponent, { props: props as any }));\n            }\n            catch(err) {\n                console.error(err)\n            }\n        })\n    }\n}\n"]}
|
|
@@ -11621,55 +11621,61 @@ class ReactMagicWrapperComponent {
|
|
|
11621
11621
|
* @param _outputs
|
|
11622
11622
|
* @returns
|
|
11623
11623
|
*/
|
|
11624
|
-
static { this.WrapAngularComponent = (
|
|
11624
|
+
static { this.WrapAngularComponent = ({ component, appRef, injector, ngZone, staticInputs, staticOutputs, additionalChildren: siblings }) => React.memo((args) => {
|
|
11625
11625
|
const id = ulid();
|
|
11626
11626
|
React.useEffect(() => {
|
|
11627
|
-
|
|
11628
|
-
|
|
11629
|
-
|
|
11630
|
-
|
|
11631
|
-
|
|
11632
|
-
|
|
11633
|
-
|
|
11634
|
-
|
|
11635
|
-
|
|
11636
|
-
|
|
11637
|
-
|
|
11638
|
-
|
|
11639
|
-
|
|
11640
|
-
|
|
11641
|
-
|
|
11642
|
-
|
|
11643
|
-
|
|
11644
|
-
|
|
11645
|
-
|
|
11646
|
-
|
|
11647
|
-
|
|
11648
|
-
|
|
11649
|
-
|
|
11650
|
-
|
|
11651
|
-
|
|
11652
|
-
|
|
11653
|
-
|
|
11654
|
-
|
|
11655
|
-
|
|
11656
|
-
const
|
|
11627
|
+
try {
|
|
11628
|
+
const componentInstance = createComponent(component, {
|
|
11629
|
+
environmentInjector: appRef.injector,
|
|
11630
|
+
elementInjector: injector,
|
|
11631
|
+
hostElement: document.getElementById(id)
|
|
11632
|
+
});
|
|
11633
|
+
appRef.attachView(componentInstance.hostView);
|
|
11634
|
+
// @ts-ignore
|
|
11635
|
+
// component.hostView = hostView;
|
|
11636
|
+
Object.assign(staticInputs, args);
|
|
11637
|
+
const { inputs, outputs } = component['ɵcmp'];
|
|
11638
|
+
// Returns a list of entries that need to be set
|
|
11639
|
+
// This makes it so that unnecessary setters are not invoked.
|
|
11640
|
+
const updated = Object.entries(inputs).filter(([parentKey, childKey]) => {
|
|
11641
|
+
return componentInstance.instance[childKey] != staticInputs[parentKey];
|
|
11642
|
+
});
|
|
11643
|
+
updated.forEach(([parentKey, childKey]) => {
|
|
11644
|
+
if (staticInputs.hasOwnProperty(parentKey))
|
|
11645
|
+
componentInstance.instance[childKey] = staticInputs[parentKey];
|
|
11646
|
+
});
|
|
11647
|
+
const outputSubscriptions = {};
|
|
11648
|
+
// Get a list of unregistered outputs
|
|
11649
|
+
const newOutputs = Object.entries(outputs).filter(([parentKey, childKey]) => {
|
|
11650
|
+
return !outputSubscriptions[parentKey];
|
|
11651
|
+
});
|
|
11652
|
+
// Reverse bind via subscription
|
|
11653
|
+
newOutputs.forEach(([parentKey, childKey]) => {
|
|
11654
|
+
if (!staticOutputs.hasOwnProperty(parentKey))
|
|
11655
|
+
return;
|
|
11656
|
+
const target = componentInstance.instance[childKey];
|
|
11657
|
+
const outputs = staticOutputs;
|
|
11658
|
+
const sub = target.subscribe((...args) => {
|
|
11659
|
+
// Run the callback in the provided zone
|
|
11660
|
+
ngZone.run(() => {
|
|
11661
|
+
outputs[parentKey](...args);
|
|
11662
|
+
});
|
|
11663
|
+
}); // Subscription
|
|
11657
11664
|
outputSubscriptions[parentKey] = sub;
|
|
11658
|
-
}
|
|
11659
|
-
|
|
11660
|
-
|
|
11661
|
-
|
|
11662
|
-
|
|
11663
|
-
|
|
11664
|
-
|
|
11665
|
-
|
|
11666
|
-
|
|
11665
|
+
});
|
|
11666
|
+
// Wrap the destroy method to safely release the subscriptions
|
|
11667
|
+
const originalDestroy = componentInstance.onDestroy?.bind(componentInstance);
|
|
11668
|
+
componentInstance.onDestroy = (cb) => {
|
|
11669
|
+
Object.values(outputSubscriptions).forEach(s => s.unsubscribe());
|
|
11670
|
+
originalDestroy?.(cb);
|
|
11671
|
+
};
|
|
11672
|
+
componentInstance.changeDetectorRef.detectChanges();
|
|
11673
|
+
}
|
|
11674
|
+
catch (err) {
|
|
11675
|
+
console.error(err);
|
|
11676
|
+
}
|
|
11667
11677
|
}, []);
|
|
11668
|
-
|
|
11669
|
-
// all of the siblings we will inject into it's container.
|
|
11670
|
-
// The double wrapper is necessary for React's mechanisms to properly
|
|
11671
|
-
// associate things together.
|
|
11672
|
-
return React.createElement('div', {}, React.createElement("div", { id }), ...additionalChildren);
|
|
11678
|
+
return React.createElement("div", {}, React.createElement("div", { id }), ...siblings);
|
|
11673
11679
|
}); }
|
|
11674
11680
|
constructor(ngContainer, ngTheme, ngZone) {
|
|
11675
11681
|
this.ngContainer = ngContainer;
|
|
@@ -11683,8 +11689,10 @@ class ReactMagicWrapperComponent {
|
|
|
11683
11689
|
];
|
|
11684
11690
|
}
|
|
11685
11691
|
ngOnInit() {
|
|
11686
|
-
if (!this.ngReactComponent)
|
|
11692
|
+
if (!this.ngReactComponent) {
|
|
11693
|
+
console.error("NO");
|
|
11687
11694
|
throw new Error("ReactMagicWrapperComponent cannot start without a provided ngReactComponent!");
|
|
11695
|
+
}
|
|
11688
11696
|
}
|
|
11689
11697
|
ngOnChanges(changes) {
|
|
11690
11698
|
this._render();
|
|
@@ -11697,29 +11705,37 @@ class ReactMagicWrapperComponent {
|
|
|
11697
11705
|
this.ngSubscriptions.forEach(s => s.unsubscribe());
|
|
11698
11706
|
}
|
|
11699
11707
|
_render() {
|
|
11700
|
-
if (!this.ngReactComponent)
|
|
11708
|
+
if (!this.ngReactComponent) {
|
|
11709
|
+
console.log("Render no component. May be context issue");
|
|
11701
11710
|
return;
|
|
11711
|
+
}
|
|
11712
|
+
;
|
|
11702
11713
|
this.ngZone.runOutsideAngular(() => {
|
|
11703
|
-
|
|
11704
|
-
|
|
11714
|
+
try {
|
|
11715
|
+
if (!this._root) {
|
|
11716
|
+
this._root = createRoot(this.ngContainer.element.nativeElement);
|
|
11717
|
+
}
|
|
11718
|
+
// List all keys that do not start with `_` nor `ng`
|
|
11719
|
+
const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));
|
|
11720
|
+
// Get all property keys from the class
|
|
11721
|
+
const propKeys = keys.filter(k => !k.startsWith("on"));
|
|
11722
|
+
// Get all event handler keys from the class
|
|
11723
|
+
const evtKeys = keys.filter(k => k.startsWith("on"));
|
|
11724
|
+
const props = {};
|
|
11725
|
+
// Project all key properties onto `props`
|
|
11726
|
+
propKeys.forEach(k => props[k] = this[k]);
|
|
11727
|
+
// Attempt to ensure no zone is lost during the event emitter fires
|
|
11728
|
+
this.ngZone.runGuarded(() => {
|
|
11729
|
+
// Bind all event handlers.
|
|
11730
|
+
// ! important Angular uses EventEmitter, React uses
|
|
11731
|
+
// a different method of event binding
|
|
11732
|
+
evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
|
|
11733
|
+
});
|
|
11734
|
+
this._root.render(React.createElement(this.ngReactComponent, { props: props }));
|
|
11735
|
+
}
|
|
11736
|
+
catch (err) {
|
|
11737
|
+
console.error(err);
|
|
11705
11738
|
}
|
|
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
|
-
// Attempt to ensure no zone is lost during the event emitter fires
|
|
11716
|
-
this.ngZone.run(() => {
|
|
11717
|
-
// Bind all event handlers.
|
|
11718
|
-
// ! important Angular uses EventEmitter, React uses
|
|
11719
|
-
// a different method of event binding
|
|
11720
|
-
evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
|
|
11721
|
-
});
|
|
11722
|
-
this._root.render(React.createElement(this.ngReactComponent, { props: props }));
|
|
11723
11739
|
});
|
|
11724
11740
|
}
|
|
11725
11741
|
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 }); }
|