ngx-reactify 0.0.1

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.
package/README.md ADDED
@@ -0,0 +1,119 @@
1
+ # ngx-reactify
2
+
3
+ > This project is currently in a beta phase and features will be added upon pull requests.
4
+ I will try to minimize breaking changes between minor version revisions but some may be made until we reach 1.0.0.
5
+
6
+ This project aims to make using Angular components in React not suck, and vice-versa. This is a dependency of my other library, ngx-xyflow.
7
+
8
+ ### Getting started:
9
+
10
+ ##### Install React
11
+
12
+ ```bash
13
+ npm i react react-dom
14
+ ```
15
+
16
+ ```bash
17
+ npm i -D @types/react
18
+ ```
19
+
20
+ ##### Install ngx-reactify
21
+
22
+ ```bash
23
+ npm i ngx-reactify
24
+ ```
25
+
26
+ ### Embed React component in Angular
27
+
28
+ ##### Create component Interface
29
+ Next, you.
30
+ > This step is necessary as it creates Angular bindings that can be used.
31
+ > Important: You can't use the normal stylesheet imports from React. Put your styles in the component.scss file.
32
+
33
+ ```ts
34
+ import { Component, EventEmitter, Input, Output, ViewEncapsulation } from '@angular/core';
35
+ import * as React from 'react';
36
+
37
+ import { ReactDemoWrappableComponent } from './react-demo'; // tsx file
38
+ import { ReactifyNgComponent } from 'ngx-reactify';
39
+
40
+ @Component({
41
+ selector: 'app-react-demo',
42
+ template: ``,
43
+ styleUrls: ['./react-demo.component.scss'],
44
+ standalone: true,
45
+ encapsulation: ViewEncapsulation.None
46
+ })
47
+ export class ReactDemoComponent extends ReactifyNgComponent {
48
+ override readonly ngReactComponent = ReactDemoWrappableComponent;
49
+
50
+ @Output() onNodeClick = new EventEmitter<[MouseEvent, Node]>();
51
+ @Output() onNodeDoubleClick = new EventEmitter<[MouseEvent, Node]>();
52
+
53
+ @Input() nodeTypes?: NodeTypes | undefined;
54
+ @Input() edgeTypes?: EdgeTypes | undefined;
55
+ }
56
+ ```
57
+
58
+ ##### Using the component
59
+
60
+ Last, import and use `ReactDemoComponent` elsewhere in your application.
61
+
62
+ ```ts
63
+
64
+ import { Component } from '@angular/core';
65
+ import { ReactDemoComponent } from './react-demo.component';
66
+
67
+ @Component({
68
+ selector: 'app-demo',
69
+ template: `
70
+ <app-react-demo
71
+ [nodeTypes]="[1,2,3,4]"
72
+ (onNodeClick)="onNodeClick($event)"
73
+ />
74
+ `,
75
+ imports: [ ReactDemoComponent ],
76
+ standalone: true
77
+ })
78
+ export class DemoComponent {
79
+
80
+ onNodeClick(evt) {
81
+ console.log(evt)
82
+ }
83
+ }
84
+ ```
85
+
86
+ ### Embed Angular component in React
87
+
88
+ > :warn: Under construction.
89
+ <!--
90
+ ```ts
91
+ import React from 'react';
92
+ import { ReactifyReactComponent } from 'ngx-reactify';
93
+ import { Component, EventEmitter, Input, Output, ViewEncapsulation } from '@angular/core';
94
+ import { ReactDemoComponent } from './react-demo.component';
95
+
96
+ @Component({
97
+ selector: 'app-demo',
98
+ template: `<div (click)="onNodeClick($event)"></div>`,
99
+ standalone: true
100
+ })
101
+ export class DemoComponent {
102
+ onNodeClick(evt) {
103
+ console.log(evt)
104
+ }
105
+ }
106
+
107
+ const AngularReactComponent = ReactifyReactComponent(DemoComponent)
108
+
109
+ export default function App() {
110
+
111
+ return (
112
+ <img src="https://picsum.photos/200" className={'picture'}></img>
113
+
114
+ <div>
115
+
116
+ </div>
117
+ )
118
+ }
119
+ ``` -->
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmd4LXJlYWN0aWZ5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL25neC1yZWFjdGlmeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUVILGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBHZW5lcmF0ZWQgYnVuZGxlIGluZGV4LiBEbyBub3QgZWRpdC5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3B1YmxpYy1hcGknO1xuIl19
@@ -0,0 +1,3 @@
1
+ export * from './util/angular-to-react';
2
+ export * from './util/react-to-angular';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMseUJBQXlCLENBQUM7QUFDeEMsY0FBYyx5QkFBeUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vdXRpbC9hbmd1bGFyLXRvLXJlYWN0JztcbmV4cG9ydCAqIGZyb20gJy4vdXRpbC9yZWFjdC10by1hbmd1bGFyJztcbiJdfQ==
@@ -0,0 +1,116 @@
1
+ import { createComponent, EventEmitter } from '@angular/core';
2
+ import { createApplication } from '@angular/platform-browser';
3
+ import * as React from 'react';
4
+ import { firstValueFrom } from 'rxjs';
5
+ // declare const Zone;
6
+ // const zone = Zone ? new Zone(Zone.current, { name: "@dotglitch_menu", properties: {} }) : null;
7
+ /**
8
+ * Wrap an angular component inside of a React memo object.
9
+ * Will attempt to bind @Input and @Output properties if provided,
10
+ * and will bind the react arguments directly as @Input properties.
11
+ *
12
+ * @experimental
13
+ * @param componentClass Angular component
14
+ * @param envInjector An `EnvironmentInjector` instance to be used for the component
15
+ * @param injector An `ElementInjector` instance
16
+ * @param _inputs
17
+ * @param _outputs
18
+ * @returns
19
+ */
20
+ export const ReactifyReactComponent = ({ component, appRef, injector, ngZone, staticInputs, staticOutputs, preSiblings, postSiblings, additionalChildren, rootElementName, containerElementName }) => React.memo((args) => {
21
+ const id = Math.random().toString();
22
+ React.useEffect(() => {
23
+ try {
24
+ const componentInstance = createComponent(component, {
25
+ environmentInjector: appRef.injector,
26
+ elementInjector: injector,
27
+ hostElement: document.getElementById(id)
28
+ });
29
+ appRef.attachView(componentInstance.hostView);
30
+ // @ts-ignore
31
+ // component.hostView = hostView;
32
+ Object.assign(staticInputs, args);
33
+ const { inputs, outputs } = component['ɵcmp'];
34
+ // Returns a list of entries that need to be set
35
+ // This makes it so that unnecessary setters are not invoked.
36
+ const updated = Object.entries(inputs).filter(([parentKey, childKey]) => {
37
+ return componentInstance.instance[childKey] != staticInputs[parentKey];
38
+ });
39
+ updated.forEach(([parentKey, childKey]) => {
40
+ if (staticInputs.hasOwnProperty(parentKey))
41
+ componentInstance.instance[childKey] = staticInputs[parentKey];
42
+ });
43
+ const outputSubscriptions = {};
44
+ // Get a list of unregistered outputs
45
+ const newOutputs = Object.entries(outputs).filter(([parentKey, childKey]) => {
46
+ return !outputSubscriptions[parentKey];
47
+ });
48
+ // Reverse bind via subscription
49
+ newOutputs.forEach(([parentKey, childKey]) => {
50
+ if (!staticOutputs.hasOwnProperty(parentKey))
51
+ return;
52
+ const target = componentInstance.instance[childKey];
53
+ const outputs = staticOutputs;
54
+ const sub = target.subscribe((...args) => {
55
+ // Run the callback in the provided zone
56
+ ngZone.run(() => {
57
+ outputs[parentKey](...args);
58
+ });
59
+ }); // Subscription
60
+ outputSubscriptions[parentKey] = sub;
61
+ });
62
+ // Wrap the destroy method to safely release the subscriptions
63
+ const originalDestroy = componentInstance.onDestroy?.bind(componentInstance);
64
+ componentInstance.onDestroy = (cb) => {
65
+ Object.values(outputSubscriptions).forEach(s => s.unsubscribe());
66
+ originalDestroy?.(cb);
67
+ };
68
+ componentInstance.changeDetectorRef.detectChanges();
69
+ }
70
+ catch (err) {
71
+ console.error(err);
72
+ }
73
+ }, []);
74
+ const elements = [
75
+ ...(preSiblings || []),
76
+ React.createElement(containerElementName || "div", { id }),
77
+ ...(postSiblings || []),
78
+ ...(additionalChildren || [])
79
+ ].filter(e => e);
80
+ return React.createElement(rootElementName || "div", {}, ...elements);
81
+ });
82
+ export const ReactifyAngularComponent2 = (component, props) => {
83
+ const inputRef = React.useRef(null);
84
+ const ctx = this;
85
+ React.useEffect(() => {
86
+ // Is there a better way to do this?
87
+ let subscriptions;
88
+ let app;
89
+ (async () => {
90
+ // Code to run when the component mounts
91
+ app = await createApplication({ providers: [] });
92
+ const base = app.bootstrap(component, inputRef.current);
93
+ const { instance } = base;
94
+ await firstValueFrom(app.isStable);
95
+ // App has now bootstrapped fully.
96
+ subscriptions = [];
97
+ Object.entries(instance).filter(([k, v]) => {
98
+ // @Outputs are always Event Emitters (I think)
99
+ if (v instanceof EventEmitter) {
100
+ subscriptions.push(instance[k]?.subscribe(evt => props[k].call(ctx, evt)));
101
+ }
102
+ else {
103
+ instance[k] = props[k];
104
+ }
105
+ });
106
+ })();
107
+ return () => {
108
+ // Code to run when the component unmounts
109
+ subscriptions?.forEach(s => s?.unsubscribe());
110
+ app?.destroy();
111
+ };
112
+ }, []); // Empty dependency array ensures this effect runs only once on mount and cleanup on unmount
113
+ const obj = {};
114
+ return React.createElement("div");
115
+ };
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"angular-to-react.js","sourceRoot":"","sources":["../../../../src/util/angular-to-react.ts"],"names":[],"mappings":"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","sourcesContent":["import { Type, ApplicationRef, Injector, NgZone, createComponent, EventEmitter } from '@angular/core';\nimport { createApplication } from '@angular/platform-browser';\nimport * as React from 'react';\nimport { firstValueFrom, Subscription } from 'rxjs';\n\n\n// declare const Zone;\n// const zone = Zone ? new Zone(Zone.current, { name: \"@dotglitch_menu\", properties: {} }) : null;\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 */\nexport const ReactifyReactComponent = ({\n    component,\n    appRef,\n    injector,\n    ngZone,\n    staticInputs,\n    staticOutputs,\n    preSiblings,\n    postSiblings,\n    additionalChildren,\n    rootElementName,\n    containerElementName\n}: {\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    preSiblings?: React.ReactNode[],\n    postSiblings?: React.ReactNode[],\n    additionalChildren?: React.ReactNode[],\n    rootElementName?: Parameters<typeof React.createElement>[0],\n    containerElementName?: string;\n}) => React.memo((args) => {\n\n    const id = Math.random().toString();\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    const elements = [\n        ...(preSiblings || []),\n        React.createElement(containerElementName || \"div\", { id }),\n        ...(postSiblings || []),\n        ...(additionalChildren || [])\n    ].filter(e => e);\n\n    return React.createElement(rootElementName || \"div\", {}, ...elements);\n});\n\n\nexport const ReactifyAngularComponent2 = (\n    component: Type<any>,\n    props: any\n) => {\n    const inputRef = React.useRef(null);\n    const ctx = this;\n\n    React.useEffect(() => {\n        // Is there a better way to do this?\n        let subscriptions: Subscription[];\n        let app: ApplicationRef;\n        (async () => {\n            // Code to run when the component mounts\n            app = await createApplication({ providers: [] });\n            const base = app.bootstrap(component, inputRef.current);\n            const { instance } = base;\n\n            await firstValueFrom(app.isStable);\n\n            // App has now bootstrapped fully.\n            subscriptions = [];\n            Object.entries(instance).filter(([k, v]) => {\n                // @Outputs are always Event Emitters (I think)\n                if (v instanceof EventEmitter) {\n                    subscriptions.push(\n                        instance[k]?.subscribe(evt => props[k].call(ctx, evt))\n                    );\n                }\n                else {\n                    instance[k] = props[k];\n                }\n            })\n        })()\n\n        return () => {\n            // Code to run when the component unmounts\n            subscriptions?.forEach(s => s?.unsubscribe());\n            app?.destroy();\n        };\n    }, []); // Empty dependency array ensures this effect runs only once on mount and cleanup on unmount\n\n    const obj = {};\n\n    return React.createElement(\"div\")\n}\n"]}
@@ -0,0 +1,75 @@
1
+ import { Component } from '@angular/core';
2
+ import * as React from 'react';
3
+ import { createRoot } from 'react-dom/client';
4
+ import * as i0 from "@angular/core";
5
+ /**
6
+ * Extend this component to automatically generate
7
+ * bindings to a React component.
8
+ *
9
+ * ! You _must_ override the property `ngReactComponent`
10
+ * Failure to do so will result in errors
11
+ * `override readonly ngReactComponent = ReactFlowWrappableComponent;`
12
+ */
13
+ export class ReactifyNgComponent {
14
+ constructor(ngContainer, ngZone) {
15
+ this.ngContainer = ngContainer;
16
+ this.ngZone = ngZone;
17
+ }
18
+ ngOnInit() {
19
+ if (!this.ngReactComponent) {
20
+ throw new Error("ReactMagicWrapperComponent cannot start without a provided ngReactComponent!");
21
+ }
22
+ }
23
+ ngOnChanges(changes) {
24
+ this._render();
25
+ }
26
+ ngAfterViewInit() {
27
+ this._render();
28
+ }
29
+ ngOnDestroy() {
30
+ this._root?.unmount();
31
+ }
32
+ _render() {
33
+ if (!this.ngReactComponent) {
34
+ console.log("Render no component. May be context issue");
35
+ return;
36
+ }
37
+ ;
38
+ this.ngZone.runOutsideAngular(() => {
39
+ try {
40
+ this._root ??= createRoot(this.ngContainer.element.nativeElement);
41
+ // List all keys that do not start with `_` nor `ng`
42
+ const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));
43
+ // Get all property keys from the class
44
+ const propKeys = keys.filter(k => !k.startsWith("on"));
45
+ // Get all event handler keys from the class
46
+ const evtKeys = keys.filter(k => k.startsWith("on"));
47
+ const props = {};
48
+ // Project all key properties onto `props`
49
+ propKeys.forEach(k => props[k] = this[k]);
50
+ // Attempt to ensure no zone is lost during the event emitter fires
51
+ this.ngZone.runGuarded(() => {
52
+ // Bind all event handlers.
53
+ // ! important Angular uses EventEmitter, React uses
54
+ // a different method of event binding
55
+ evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
56
+ });
57
+ this._root.render(React.createElement(this.ngReactComponent, { props: props }));
58
+ }
59
+ catch (err) {
60
+ console.error(err);
61
+ }
62
+ });
63
+ }
64
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactifyNgComponent, deps: [{ token: i0.ViewContainerRef }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
65
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.2", type: ReactifyNgComponent, isStandalone: true, selector: "app-react-magic-wrapper", usesOnChanges: true, ngImport: i0, template: ``, isInline: true }); }
66
+ }
67
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactifyNgComponent, decorators: [{
68
+ type: Component,
69
+ args: [{
70
+ selector: 'app-react-magic-wrapper',
71
+ template: ``,
72
+ standalone: true
73
+ }]
74
+ }], ctorParameters: () => [{ type: i0.ViewContainerRef }, { type: i0.NgZone }] });
75
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVhY3QtdG8tYW5ndWxhci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy91dGlsL3JlYWN0LXRvLWFuZ3VsYXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFpQyxTQUFTLEVBQXdLLE1BQU0sZUFBZSxDQUFDO0FBQy9PLE9BQU8sS0FBSyxLQUFLLE1BQU0sT0FBTyxDQUFDO0FBQy9CLE9BQU8sRUFBRSxVQUFVLEVBQVEsTUFBTSxrQkFBa0IsQ0FBQzs7QUFHcEQ7Ozs7Ozs7R0FPRztBQU1ILE1BQU0sT0FBTyxtQkFBbUI7SUFXNUIsWUFDcUIsV0FBNkIsRUFDN0IsTUFBYztRQURkLGdCQUFXLEdBQVgsV0FBVyxDQUFrQjtRQUM3QixXQUFNLEdBQU4sTUFBTSxDQUFRO0lBRW5DLENBQUM7SUFFRCxRQUFRO1FBQ0osSUFBSSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO1lBQ3pCLE1BQU0sSUFBSSxLQUFLLENBQUMsOEVBQThFLENBQUMsQ0FBQztRQUNwRyxDQUFDO0lBQ0wsQ0FBQztJQUVELFdBQVcsQ0FBQyxPQUF1QjtRQUMvQixJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDbkIsQ0FBQztJQUVELGVBQWU7UUFDWCxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDbkIsQ0FBQztJQUVELFdBQVc7UUFDUCxJQUFJLENBQUMsS0FBSyxFQUFFLE9BQU8sRUFBRSxDQUFDO0lBQzFCLENBQUM7SUFFTyxPQUFPO1FBQ1gsSUFBSSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO1lBQ3pCLE9BQU8sQ0FBQyxHQUFHLENBQUMsMkNBQTJDLENBQUMsQ0FBQTtZQUN4RCxPQUFNO1FBQ1YsQ0FBQztRQUFBLENBQUM7UUFFRixJQUFJLENBQUMsTUFBTSxDQUFDLGlCQUFpQixDQUFDLEdBQUcsRUFBRTtZQUMvQixJQUFJLENBQUM7Z0JBQ0QsSUFBSSxDQUFDLEtBQUssS0FBSyxVQUFVLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxPQUFPLENBQUMsYUFBYSxDQUFDLENBQUM7Z0JBRWxFLG9EQUFvRDtnQkFDcEQsTUFBTSxJQUFJLEdBQUcsTUFBTSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztnQkFFakUsdUNBQXVDO2dCQUN2QyxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsVUFBVSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7Z0JBQ3ZELDRDQUE0QztnQkFDNUMsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztnQkFFckQsTUFBTSxLQUFLLEdBQUcsRUFBRSxDQUFDO2dCQUNqQiwwQ0FBMEM7Z0JBQzFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7Z0JBRTFDLG1FQUFtRTtnQkFDbkUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxVQUFVLENBQUMsR0FBRyxFQUFFO29CQUN4QiwyQkFBMkI7b0JBQzNCLG9EQUFvRDtvQkFDcEQsc0NBQXNDO29CQUN0QyxPQUFPLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsR0FBRyxJQUFJLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztnQkFDckUsQ0FBQyxDQUFDLENBQUE7Z0JBRUYsSUFBSSxDQUFDLEtBQUssQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLEVBQUUsRUFBRSxLQUFLLEVBQUUsS0FBWSxFQUFFLENBQUMsQ0FBQyxDQUFDO1lBQzNGLENBQUM7WUFDRCxPQUFNLEdBQUcsRUFBRSxDQUFDO2dCQUNSLE9BQU8sQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUE7WUFDdEIsQ0FBQztRQUNMLENBQUMsQ0FBQyxDQUFBO0lBQ04sQ0FBQzs4R0F2RVEsbUJBQW1CO2tHQUFuQixtQkFBbUIsd0dBSGxCLEVBQUU7OzJGQUdILG1CQUFtQjtrQkFML0IsU0FBUzttQkFBQztvQkFDUCxRQUFRLEVBQUUseUJBQXlCO29CQUNuQyxRQUFRLEVBQUUsRUFBRTtvQkFDWixVQUFVLEVBQUUsSUFBSTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBZnRlclZpZXdJbml0LCBBcHBsaWNhdGlvblJlZiwgQ29tcG9uZW50LCBDb21wb25lbnRGYWN0b3J5UmVzb2x2ZXIsIEVudmlyb25tZW50SW5qZWN0b3IsIEV2ZW50RW1pdHRlciwgSW5qZWN0b3IsIE5nWm9uZSwgT25DaGFuZ2VzLCBPbkRlc3Ryb3ksIFNpbXBsZUNoYW5nZXMsIFR5cGUsIFZpZXdDb250YWluZXJSZWYsIFZpZXdSZWYsIGNyZWF0ZUNvbXBvbmVudCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnO1xuaW1wb3J0IHsgY3JlYXRlUm9vdCwgUm9vdCB9IGZyb20gJ3JlYWN0LWRvbS9jbGllbnQnO1xuXG5cbi8qKlxuICogRXh0ZW5kIHRoaXMgY29tcG9uZW50IHRvIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVcbiAqIGJpbmRpbmdzIHRvIGEgUmVhY3QgY29tcG9uZW50LlxuICpcbiAqICEgWW91IF9tdXN0XyBvdmVycmlkZSB0aGUgcHJvcGVydHkgYG5nUmVhY3RDb21wb25lbnRgXG4gKiBGYWlsdXJlIHRvIGRvIHNvIHdpbGwgcmVzdWx0IGluIGVycm9yc1xuICogYG92ZXJyaWRlIHJlYWRvbmx5IG5nUmVhY3RDb21wb25lbnQgPSBSZWFjdEZsb3dXcmFwcGFibGVDb21wb25lbnQ7YFxuICovXG5AQ29tcG9uZW50KHtcbiAgICBzZWxlY3RvcjogJ2FwcC1yZWFjdC1tYWdpYy13cmFwcGVyJyxcbiAgICB0ZW1wbGF0ZTogYGAsXG4gICAgc3RhbmRhbG9uZTogdHJ1ZVxufSlcbmV4cG9ydCBjbGFzcyBSZWFjdGlmeU5nQ29tcG9uZW50IGltcGxlbWVudHMgT25DaGFuZ2VzLCBPbkRlc3Ryb3ksIEFmdGVyVmlld0luaXQge1xuXG4gICAgLyoqXG4gICAgICogVGhlIHJlYWN0IGNvbXBvbmVudCB0byBiZSB3cmFwcGVkLlxuICAgICAqICEgTXVzdCBiZSBvdmVycmlkZGVuIGZvciB0aGlzIHdyYXBwZXIgdG8gd29ya1xuICAgICAqL1xuICAgIG5nUmVhY3RDb21wb25lbnQ6IFJlYWN0LkZ1bmN0aW9uQ29tcG9uZW50PGFueT4gfCBSZWFjdC5Db21wb25lbnRDbGFzczxhbnk+O1xuXG4gICAgcHJpdmF0ZSBfcm9vdDogUm9vdDtcbiAgICBwdWJsaWMgdGhlbWU6IHN0cmluZztcblxuICAgIGNvbnN0cnVjdG9yKFxuICAgICAgICBwcml2YXRlIHJlYWRvbmx5IG5nQ29udGFpbmVyOiBWaWV3Q29udGFpbmVyUmVmLFxuICAgICAgICBwcml2YXRlIHJlYWRvbmx5IG5nWm9uZTogTmdab25lXG4gICAgKSB7XG4gICAgfVxuXG4gICAgbmdPbkluaXQoKSB7XG4gICAgICAgIGlmICghdGhpcy5uZ1JlYWN0Q29tcG9uZW50KSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJSZWFjdE1hZ2ljV3JhcHBlckNvbXBvbmVudCBjYW5ub3Qgc3RhcnQgd2l0aG91dCBhIHByb3ZpZGVkIG5nUmVhY3RDb21wb25lbnQhXCIpO1xuICAgICAgICB9XG4gICAgfVxuXG4gICAgbmdPbkNoYW5nZXMoY2hhbmdlcz86IFNpbXBsZUNoYW5nZXMpOiB2b2lkIHtcbiAgICAgICAgdGhpcy5fcmVuZGVyKCk7XG4gICAgfVxuXG4gICAgbmdBZnRlclZpZXdJbml0KCkge1xuICAgICAgICB0aGlzLl9yZW5kZXIoKTtcbiAgICB9XG5cbiAgICBuZ09uRGVzdHJveSgpIHtcbiAgICAgICAgdGhpcy5fcm9vdD8udW5tb3VudCgpO1xuICAgIH1cblxuICAgIHByaXZhdGUgX3JlbmRlcigpIHtcbiAgICAgICAgaWYgKCF0aGlzLm5nUmVhY3RDb21wb25lbnQpIHtcbiAgICAgICAgICAgIGNvbnNvbGUubG9nKFwiUmVuZGVyIG5vIGNvbXBvbmVudC4gTWF5IGJlIGNvbnRleHQgaXNzdWVcIilcbiAgICAgICAgICAgIHJldHVyblxuICAgICAgICB9O1xuXG4gICAgICAgIHRoaXMubmdab25lLnJ1bk91dHNpZGVBbmd1bGFyKCgpID0+IHtcbiAgICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICAgICAgdGhpcy5fcm9vdCA/Pz0gY3JlYXRlUm9vdCh0aGlzLm5nQ29udGFpbmVyLmVsZW1lbnQubmF0aXZlRWxlbWVudCk7XG5cbiAgICAgICAgICAgICAgICAvLyBMaXN0IGFsbCBrZXlzIHRoYXQgZG8gbm90IHN0YXJ0IHdpdGggYF9gIG5vciBgbmdgXG4gICAgICAgICAgICAgICAgY29uc3Qga2V5cyA9IE9iamVjdC5rZXlzKHRoaXMpLmZpbHRlcihrID0+ICEvXig/Ol98bmcpLy50ZXN0KGspKTtcblxuICAgICAgICAgICAgICAgIC8vIEdldCBhbGwgcHJvcGVydHkga2V5cyBmcm9tIHRoZSBjbGFzc1xuICAgICAgICAgICAgICAgIGNvbnN0IHByb3BLZXlzID0ga2V5cy5maWx0ZXIoayA9PiAhay5zdGFydHNXaXRoKFwib25cIikpO1xuICAgICAgICAgICAgICAgIC8vIEdldCBhbGwgZXZlbnQgaGFuZGxlciBrZXlzIGZyb20gdGhlIGNsYXNzXG4gICAgICAgICAgICAgICAgY29uc3QgZXZ0S2V5cyA9IGtleXMuZmlsdGVyKGsgPT4gay5zdGFydHNXaXRoKFwib25cIikpO1xuXG4gICAgICAgICAgICAgICAgY29uc3QgcHJvcHMgPSB7fTtcbiAgICAgICAgICAgICAgICAvLyBQcm9qZWN0IGFsbCBrZXkgcHJvcGVydGllcyBvbnRvIGBwcm9wc2BcbiAgICAgICAgICAgICAgICBwcm9wS2V5cy5mb3JFYWNoKGsgPT4gcHJvcHNba10gPSB0aGlzW2tdKTtcblxuICAgICAgICAgICAgICAgIC8vIEF0dGVtcHQgdG8gZW5zdXJlIG5vIHpvbmUgaXMgbG9zdCBkdXJpbmcgdGhlIGV2ZW50IGVtaXR0ZXIgZmlyZXNcbiAgICAgICAgICAgICAgICB0aGlzLm5nWm9uZS5ydW5HdWFyZGVkKCgpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgLy8gQmluZCBhbGwgZXZlbnQgaGFuZGxlcnMuXG4gICAgICAgICAgICAgICAgICAgIC8vICEgaW1wb3J0YW50IEFuZ3VsYXIgdXNlcyBFdmVudEVtaXR0ZXIsIFJlYWN0IHVzZXNcbiAgICAgICAgICAgICAgICAgICAgLy8gYSBkaWZmZXJlbnQgbWV0aG9kIG9mIGV2ZW50IGJpbmRpbmdcbiAgICAgICAgICAgICAgICAgICAgZXZ0S2V5cy5mb3JFYWNoKGsgPT4gcHJvcHNba10gPSAoLi4uYXJncykgPT4gdGhpc1trXS5uZXh0KGFyZ3MpKTtcbiAgICAgICAgICAgICAgICB9KVxuXG4gICAgICAgICAgICAgICAgdGhpcy5fcm9vdC5yZW5kZXIoUmVhY3QuY3JlYXRlRWxlbWVudCh0aGlzLm5nUmVhY3RDb21wb25lbnQsIHsgcHJvcHM6IHByb3BzIGFzIGFueSB9KSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjYXRjaChlcnIpIHtcbiAgICAgICAgICAgICAgICBjb25zb2xlLmVycm9yKGVycilcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICB9XG59XG4iXX0=
@@ -0,0 +1,196 @@
1
+ import * as i0 from '@angular/core';
2
+ import { createComponent, EventEmitter, Component } from '@angular/core';
3
+ import { createApplication } from '@angular/platform-browser';
4
+ import * as React from 'react';
5
+ import { firstValueFrom } from 'rxjs';
6
+ import { createRoot } from 'react-dom/client';
7
+
8
+ // declare const Zone;
9
+ // const zone = Zone ? new Zone(Zone.current, { name: "@dotglitch_menu", properties: {} }) : null;
10
+ /**
11
+ * Wrap an angular component inside of a React memo object.
12
+ * Will attempt to bind @Input and @Output properties if provided,
13
+ * and will bind the react arguments directly as @Input properties.
14
+ *
15
+ * @experimental
16
+ * @param componentClass Angular component
17
+ * @param envInjector An `EnvironmentInjector` instance to be used for the component
18
+ * @param injector An `ElementInjector` instance
19
+ * @param _inputs
20
+ * @param _outputs
21
+ * @returns
22
+ */
23
+ const ReactifyReactComponent = ({ component, appRef, injector, ngZone, staticInputs, staticOutputs, preSiblings, postSiblings, additionalChildren, rootElementName, containerElementName }) => React.memo((args) => {
24
+ const id = Math.random().toString();
25
+ React.useEffect(() => {
26
+ try {
27
+ const componentInstance = createComponent(component, {
28
+ environmentInjector: appRef.injector,
29
+ elementInjector: injector,
30
+ hostElement: document.getElementById(id)
31
+ });
32
+ appRef.attachView(componentInstance.hostView);
33
+ // @ts-ignore
34
+ // component.hostView = hostView;
35
+ Object.assign(staticInputs, args);
36
+ const { inputs, outputs } = component['ɵcmp'];
37
+ // Returns a list of entries that need to be set
38
+ // This makes it so that unnecessary setters are not invoked.
39
+ const updated = Object.entries(inputs).filter(([parentKey, childKey]) => {
40
+ return componentInstance.instance[childKey] != staticInputs[parentKey];
41
+ });
42
+ updated.forEach(([parentKey, childKey]) => {
43
+ if (staticInputs.hasOwnProperty(parentKey))
44
+ componentInstance.instance[childKey] = staticInputs[parentKey];
45
+ });
46
+ const outputSubscriptions = {};
47
+ // Get a list of unregistered outputs
48
+ const newOutputs = Object.entries(outputs).filter(([parentKey, childKey]) => {
49
+ return !outputSubscriptions[parentKey];
50
+ });
51
+ // Reverse bind via subscription
52
+ newOutputs.forEach(([parentKey, childKey]) => {
53
+ if (!staticOutputs.hasOwnProperty(parentKey))
54
+ return;
55
+ const target = componentInstance.instance[childKey];
56
+ const outputs = staticOutputs;
57
+ const sub = target.subscribe((...args) => {
58
+ // Run the callback in the provided zone
59
+ ngZone.run(() => {
60
+ outputs[parentKey](...args);
61
+ });
62
+ }); // Subscription
63
+ outputSubscriptions[parentKey] = sub;
64
+ });
65
+ // Wrap the destroy method to safely release the subscriptions
66
+ const originalDestroy = componentInstance.onDestroy?.bind(componentInstance);
67
+ componentInstance.onDestroy = (cb) => {
68
+ Object.values(outputSubscriptions).forEach(s => s.unsubscribe());
69
+ originalDestroy?.(cb);
70
+ };
71
+ componentInstance.changeDetectorRef.detectChanges();
72
+ }
73
+ catch (err) {
74
+ console.error(err);
75
+ }
76
+ }, []);
77
+ const elements = [
78
+ ...(preSiblings || []),
79
+ React.createElement(containerElementName || "div", { id }),
80
+ ...(postSiblings || []),
81
+ ...(additionalChildren || [])
82
+ ].filter(e => e);
83
+ return React.createElement(rootElementName || "div", {}, ...elements);
84
+ });
85
+ const ReactifyAngularComponent2 = (component, props) => {
86
+ const inputRef = React.useRef(null);
87
+ const ctx = this;
88
+ React.useEffect(() => {
89
+ // Is there a better way to do this?
90
+ let subscriptions;
91
+ let app;
92
+ (async () => {
93
+ // Code to run when the component mounts
94
+ app = await createApplication({ providers: [] });
95
+ const base = app.bootstrap(component, inputRef.current);
96
+ const { instance } = base;
97
+ await firstValueFrom(app.isStable);
98
+ // App has now bootstrapped fully.
99
+ subscriptions = [];
100
+ Object.entries(instance).filter(([k, v]) => {
101
+ // @Outputs are always Event Emitters (I think)
102
+ if (v instanceof EventEmitter) {
103
+ subscriptions.push(instance[k]?.subscribe(evt => props[k].call(ctx, evt)));
104
+ }
105
+ else {
106
+ instance[k] = props[k];
107
+ }
108
+ });
109
+ })();
110
+ return () => {
111
+ // Code to run when the component unmounts
112
+ subscriptions?.forEach(s => s?.unsubscribe());
113
+ app?.destroy();
114
+ };
115
+ }, []); // Empty dependency array ensures this effect runs only once on mount and cleanup on unmount
116
+ const obj = {};
117
+ return React.createElement("div");
118
+ };
119
+
120
+ /**
121
+ * Extend this component to automatically generate
122
+ * bindings to a React component.
123
+ *
124
+ * ! You _must_ override the property `ngReactComponent`
125
+ * Failure to do so will result in errors
126
+ * `override readonly ngReactComponent = ReactFlowWrappableComponent;`
127
+ */
128
+ class ReactifyNgComponent {
129
+ constructor(ngContainer, ngZone) {
130
+ this.ngContainer = ngContainer;
131
+ this.ngZone = ngZone;
132
+ }
133
+ ngOnInit() {
134
+ if (!this.ngReactComponent) {
135
+ throw new Error("ReactMagicWrapperComponent cannot start without a provided ngReactComponent!");
136
+ }
137
+ }
138
+ ngOnChanges(changes) {
139
+ this._render();
140
+ }
141
+ ngAfterViewInit() {
142
+ this._render();
143
+ }
144
+ ngOnDestroy() {
145
+ this._root?.unmount();
146
+ }
147
+ _render() {
148
+ if (!this.ngReactComponent) {
149
+ console.log("Render no component. May be context issue");
150
+ return;
151
+ }
152
+ ;
153
+ this.ngZone.runOutsideAngular(() => {
154
+ try {
155
+ this._root ??= createRoot(this.ngContainer.element.nativeElement);
156
+ // List all keys that do not start with `_` nor `ng`
157
+ const keys = Object.keys(this).filter(k => !/^(?:_|ng)/.test(k));
158
+ // Get all property keys from the class
159
+ const propKeys = keys.filter(k => !k.startsWith("on"));
160
+ // Get all event handler keys from the class
161
+ const evtKeys = keys.filter(k => k.startsWith("on"));
162
+ const props = {};
163
+ // Project all key properties onto `props`
164
+ propKeys.forEach(k => props[k] = this[k]);
165
+ // Attempt to ensure no zone is lost during the event emitter fires
166
+ this.ngZone.runGuarded(() => {
167
+ // Bind all event handlers.
168
+ // ! important Angular uses EventEmitter, React uses
169
+ // a different method of event binding
170
+ evtKeys.forEach(k => props[k] = (...args) => this[k].next(args));
171
+ });
172
+ this._root.render(React.createElement(this.ngReactComponent, { props: props }));
173
+ }
174
+ catch (err) {
175
+ console.error(err);
176
+ }
177
+ });
178
+ }
179
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactifyNgComponent, deps: [{ token: i0.ViewContainerRef }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
180
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.2", type: ReactifyNgComponent, isStandalone: true, selector: "app-react-magic-wrapper", usesOnChanges: true, ngImport: i0, template: ``, isInline: true }); }
181
+ }
182
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: ReactifyNgComponent, decorators: [{
183
+ type: Component,
184
+ args: [{
185
+ selector: 'app-react-magic-wrapper',
186
+ template: ``,
187
+ standalone: true
188
+ }]
189
+ }], ctorParameters: () => [{ type: i0.ViewContainerRef }, { type: i0.NgZone }] });
190
+
191
+ /**
192
+ * Generated bundle index. Do not edit.
193
+ */
194
+
195
+ export { ReactifyAngularComponent2, ReactifyNgComponent, ReactifyReactComponent };
196
+ //# sourceMappingURL=ngx-reactify.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ngx-reactify.mjs","sources":["../../../src/util/angular-to-react.ts","../../../src/util/react-to-angular.ts","../../../src/ngx-reactify.ts"],"sourcesContent":["import { Type, ApplicationRef, Injector, NgZone, createComponent, EventEmitter } from '@angular/core';\nimport { createApplication } from '@angular/platform-browser';\nimport * as React from 'react';\nimport { firstValueFrom, Subscription } from 'rxjs';\n\n\n// declare const Zone;\n// const zone = Zone ? new Zone(Zone.current, { name: \"@dotglitch_menu\", properties: {} }) : null;\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 */\nexport const ReactifyReactComponent = ({\n component,\n appRef,\n injector,\n ngZone,\n staticInputs,\n staticOutputs,\n preSiblings,\n postSiblings,\n additionalChildren,\n rootElementName,\n containerElementName\n}: {\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 preSiblings?: React.ReactNode[],\n postSiblings?: React.ReactNode[],\n additionalChildren?: React.ReactNode[],\n rootElementName?: Parameters<typeof React.createElement>[0],\n containerElementName?: string;\n}) => React.memo((args) => {\n\n const id = Math.random().toString();\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 const elements = [\n ...(preSiblings || []),\n React.createElement(containerElementName || \"div\", { id }),\n ...(postSiblings || []),\n ...(additionalChildren || [])\n ].filter(e => e);\n\n return React.createElement(rootElementName || \"div\", {}, ...elements);\n});\n\n\nexport const ReactifyAngularComponent2 = (\n component: Type<any>,\n props: any\n) => {\n const inputRef = React.useRef(null);\n const ctx = this;\n\n React.useEffect(() => {\n // Is there a better way to do this?\n let subscriptions: Subscription[];\n let app: ApplicationRef;\n (async () => {\n // Code to run when the component mounts\n app = await createApplication({ providers: [] });\n const base = app.bootstrap(component, inputRef.current);\n const { instance } = base;\n\n await firstValueFrom(app.isStable);\n\n // App has now bootstrapped fully.\n subscriptions = [];\n Object.entries(instance).filter(([k, v]) => {\n // @Outputs are always Event Emitters (I think)\n if (v instanceof EventEmitter) {\n subscriptions.push(\n instance[k]?.subscribe(evt => props[k].call(ctx, evt))\n );\n }\n else {\n instance[k] = props[k];\n }\n })\n })()\n\n return () => {\n // Code to run when the component unmounts\n subscriptions?.forEach(s => s?.unsubscribe());\n app?.destroy();\n };\n }, []); // Empty dependency array ensures this effect runs only once on mount and cleanup on unmount\n\n const obj = {};\n\n return React.createElement(\"div\")\n}\n","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\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 ReactifyNgComponent implements OnChanges, OnDestroy, AfterViewInit {\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>;\n\n private _root: Root;\n public theme: string;\n\n constructor(\n private readonly ngContainer: ViewContainerRef,\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\n ngOnChanges(changes?: SimpleChanges): void {\n this._render();\n }\n\n ngAfterViewInit() {\n this._render();\n }\n\n ngOnDestroy() {\n this._root?.unmount();\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 this._root ??= createRoot(this.ngContainer.element.nativeElement);\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","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["this"],"mappings":"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"}
package/index.d.ts ADDED
@@ -0,0 +1,6 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ /// <amd-module name="ngx-reactify" />
5
+ export * from './public-api';
6
+ //# sourceMappingURL=ngx-reactify.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ngx-reactify.d.ts","sourceRoot":"","sources":["../../src/ngx-reactify.ts"],"names":[],"mappings":"AAAA;;GAEG;;AAEH,cAAc,cAAc,CAAC"}
package/package.json ADDED
@@ -0,0 +1,36 @@
1
+ {
2
+ "name": "ngx-reactify",
3
+ "version": "0.0.1",
4
+ "repository": {
5
+ "url": "https://github.com/knackstedt/ngx-reactify"
6
+ },
7
+ "license": "MIT",
8
+ "author": {
9
+ "email": "andrewk@vivaldi.net",
10
+ "name": "Andrew G. Knackstedt"
11
+ },
12
+ "private": false,
13
+ "peerDependencies": {
14
+ "@angular/common": ">=17.0.0",
15
+ "@angular/core": ">=17.0.0"
16
+ },
17
+ "dependencies": {
18
+ "react": ">=18.0.0",
19
+ "react-dom": ">=18.0.0",
20
+ "tslib": ">=2.6.0"
21
+ },
22
+ "module": "fesm2022/ngx-reactify.mjs",
23
+ "typings": "index.d.ts",
24
+ "exports": {
25
+ "./package.json": {
26
+ "default": "./package.json"
27
+ },
28
+ ".": {
29
+ "types": "./index.d.ts",
30
+ "esm2022": "./esm2022/ngx-reactify.mjs",
31
+ "esm": "./esm2022/ngx-reactify.mjs",
32
+ "default": "./fesm2022/ngx-reactify.mjs"
33
+ }
34
+ },
35
+ "sideEffects": false
36
+ }
@@ -0,0 +1,3 @@
1
+ export * from './util/angular-to-react';
2
+ export * from './util/react-to-angular';
3
+ //# sourceMappingURL=public-api.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"public-api.d.ts","sourceRoot":"","sources":["../../src/public-api.ts"],"names":[],"mappings":"AAAA,cAAc,yBAAyB,CAAC;AACxC,cAAc,yBAAyB,CAAC"}
@@ -0,0 +1,34 @@
1
+ import { Type, ApplicationRef, Injector, NgZone } from '@angular/core';
2
+ import * as React from 'react';
3
+ /**
4
+ * Wrap an angular component inside of a React memo object.
5
+ * Will attempt to bind @Input and @Output properties if provided,
6
+ * and will bind the react arguments directly as @Input properties.
7
+ *
8
+ * @experimental
9
+ * @param componentClass Angular component
10
+ * @param envInjector An `EnvironmentInjector` instance to be used for the component
11
+ * @param injector An `ElementInjector` instance
12
+ * @param _inputs
13
+ * @param _outputs
14
+ * @returns
15
+ */
16
+ export declare const ReactifyReactComponent: ({ component, appRef, injector, ngZone, staticInputs, staticOutputs, preSiblings, postSiblings, additionalChildren, rootElementName, containerElementName }: {
17
+ component: Type<any>;
18
+ appRef: Omit<ApplicationRef, '_runningTick'>;
19
+ injector: Injector;
20
+ ngZone: NgZone;
21
+ staticInputs?: {
22
+ [key: string]: any;
23
+ };
24
+ staticOutputs?: {
25
+ [key: string]: Function;
26
+ };
27
+ preSiblings?: React.ReactNode[];
28
+ postSiblings?: React.ReactNode[];
29
+ additionalChildren?: React.ReactNode[];
30
+ rootElementName?: Parameters<typeof React.createElement>[0];
31
+ containerElementName?: string;
32
+ }) => React.NamedExoticComponent<object>;
33
+ export declare const ReactifyAngularComponent2: (component: Type<any>, props: any) => React.DetailedReactHTMLElement<React.HTMLAttributes<HTMLElement>, HTMLElement>;
34
+ //# sourceMappingURL=angular-to-react.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"angular-to-react.d.ts","sourceRoot":"","sources":["../../../src/util/angular-to-react.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,cAAc,EAAE,QAAQ,EAAE,MAAM,EAAiC,MAAM,eAAe,CAAC;AAEtG,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;AAO/B;;;;;;;;;;;;GAYG;AACH,eAAO,MAAM,sBAAsB;eAapB,KAAK,GAAG,CAAC;YACZ,KAAK,cAAc,EAAE,cAAc,CAAC;cAClC,QAAQ;YACV,MAAM;;;;;;;kBAGA,MAAM,SAAS,EAAE;mBAChB,MAAM,SAAS,EAAE;yBACX,MAAM,SAAS,EAAE;sBACpB,WAAW,OAAO,MAAM,aAAa,CAAC,CAAC,CAAC,CAAC;2BACpC,MAAM;wCA6E/B,CAAC;AAGH,eAAO,MAAM,yBAAyB,cACvB,KAAK,GAAG,CAAC,SACb,GAAG,mFA0Cb,CAAA"}
@@ -0,0 +1,31 @@
1
+ import { AfterViewInit, NgZone, OnChanges, OnDestroy, SimpleChanges, ViewContainerRef } from '@angular/core';
2
+ import * as React from 'react';
3
+ import * as i0 from "@angular/core";
4
+ /**
5
+ * Extend this component to automatically generate
6
+ * bindings to a React component.
7
+ *
8
+ * ! You _must_ override the property `ngReactComponent`
9
+ * Failure to do so will result in errors
10
+ * `override readonly ngReactComponent = ReactFlowWrappableComponent;`
11
+ */
12
+ export declare class ReactifyNgComponent implements OnChanges, OnDestroy, AfterViewInit {
13
+ private readonly ngContainer;
14
+ private readonly ngZone;
15
+ /**
16
+ * The react component to be wrapped.
17
+ * ! Must be overridden for this wrapper to work
18
+ */
19
+ ngReactComponent: React.FunctionComponent<any> | React.ComponentClass<any>;
20
+ private _root;
21
+ theme: string;
22
+ constructor(ngContainer: ViewContainerRef, ngZone: NgZone);
23
+ ngOnInit(): void;
24
+ ngOnChanges(changes?: SimpleChanges): void;
25
+ ngAfterViewInit(): void;
26
+ ngOnDestroy(): void;
27
+ private _render;
28
+ static ɵfac: i0.ɵɵFactoryDeclaration<ReactifyNgComponent, never>;
29
+ static ɵcmp: i0.ɵɵComponentDeclaration<ReactifyNgComponent, "app-react-magic-wrapper", never, {}, {}, never, never, true, never>;
30
+ }
31
+ //# sourceMappingURL=react-to-angular.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"react-to-angular.d.ts","sourceRoot":"","sources":["../../../src/util/react-to-angular.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAoG,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,aAAa,EAAQ,gBAAgB,EAA4B,MAAM,eAAe,CAAC;AAC/O,OAAO,KAAK,KAAK,MAAM,OAAO,CAAC;;AAI/B;;;;;;;GAOG;AACH,qBAKa,mBAAoB,YAAW,SAAS,EAAE,SAAS,EAAE,aAAa;IAYvE,OAAO,CAAC,QAAQ,CAAC,WAAW;IAC5B,OAAO,CAAC,QAAQ,CAAC,MAAM;IAX3B;;;OAGG;IACH,gBAAgB,EAAE,KAAK,CAAC,iBAAiB,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;IAE3E,OAAO,CAAC,KAAK,CAAO;IACb,KAAK,EAAE,MAAM,CAAC;gBAGA,WAAW,EAAE,gBAAgB,EAC7B,MAAM,EAAE,MAAM;IAInC,QAAQ;IAMR,WAAW,CAAC,OAAO,CAAC,EAAE,aAAa,GAAG,IAAI;IAI1C,eAAe;IAIf,WAAW;IAIX,OAAO,CAAC,OAAO;yCAnCN,mBAAmB;2CAAnB,mBAAmB;CAwE/B"}