mirror of
https://github.com/twentyhq/twenty
synced 2026-04-21 13:37:22 +00:00
132 lines
3.8 KiB
TypeScript
132 lines
3.8 KiB
TypeScript
import { ThreadWebWorker, release, retain } from '@quilted/threads';
|
|
import { RemoteReceiver } from '@remote-dom/core/receivers';
|
|
import { useEffect, useRef } from 'react';
|
|
import { type ActionConfirmationModalResult } from '../../../sdk/front-component-api/globals/frontComponentHostCommunicationApi';
|
|
import { type FrontComponentHostCommunicationApi } from '../../types/FrontComponentHostCommunicationApi';
|
|
import { type WorkerExports } from '../../types/WorkerExports';
|
|
import { createRemoteWorker } from '../worker/utils/createRemoteWorker';
|
|
|
|
// Must match ACTION_MENU_CONFIRMATION_MODAL_RESULT_BROWSER_EVENT_NAME in twenty-front
|
|
const ACTION_MENU_CONFIRMATION_MODAL_RESULT_BROWSER_EVENT_NAME =
|
|
'action-menu-confirmation-modal-result';
|
|
|
|
type ActionMenuConfirmationModalResultBrowserEventDetail = {
|
|
frontComponentId: string;
|
|
confirmationResult: ActionConfirmationModalResult;
|
|
};
|
|
|
|
type FrontComponentWorkerEffectProps = {
|
|
componentUrl: string;
|
|
applicationAccessToken?: string;
|
|
apiUrl?: string;
|
|
frontComponentId: string;
|
|
frontComponentHostCommunicationApi: FrontComponentHostCommunicationApi;
|
|
setReceiver: React.Dispatch<React.SetStateAction<RemoteReceiver | null>>;
|
|
setThread: React.Dispatch<
|
|
React.SetStateAction<ThreadWebWorker<
|
|
WorkerExports,
|
|
FrontComponentHostCommunicationApi
|
|
> | null>
|
|
>;
|
|
setError: React.Dispatch<React.SetStateAction<Error | null>>;
|
|
};
|
|
|
|
export const FrontComponentWorkerEffect = ({
|
|
componentUrl,
|
|
applicationAccessToken,
|
|
apiUrl,
|
|
frontComponentId,
|
|
frontComponentHostCommunicationApi,
|
|
setReceiver,
|
|
setThread,
|
|
setError,
|
|
}: FrontComponentWorkerEffectProps) => {
|
|
const isInitializedRef = useRef(false);
|
|
|
|
useEffect(() => {
|
|
if (isInitializedRef.current) {
|
|
return;
|
|
}
|
|
|
|
const newReceiver = new RemoteReceiver({ retain, release });
|
|
|
|
const worker = createRemoteWorker();
|
|
|
|
worker.onerror = (event: ErrorEvent) => {
|
|
const workerError =
|
|
event.error ?? new Error(event.message || 'Unknown worker error');
|
|
|
|
console.error('[FrontComponentRenderer] Worker error:', workerError);
|
|
setError(workerError);
|
|
};
|
|
|
|
const thread = new ThreadWebWorker<
|
|
WorkerExports,
|
|
FrontComponentHostCommunicationApi
|
|
>(worker, {
|
|
exports: frontComponentHostCommunicationApi,
|
|
});
|
|
|
|
const handleActionMenuConfirmationModalResultBrowserEvent = (
|
|
event: CustomEvent<ActionMenuConfirmationModalResultBrowserEventDetail>,
|
|
) => {
|
|
const actionMenuConfirmationModalResultBrowserEventDetail = event.detail;
|
|
|
|
if (
|
|
actionMenuConfirmationModalResultBrowserEventDetail.frontComponentId !==
|
|
frontComponentId
|
|
) {
|
|
return;
|
|
}
|
|
|
|
thread.imports
|
|
.onConfirmationModalResult(
|
|
actionMenuConfirmationModalResultBrowserEventDetail.confirmationResult,
|
|
)
|
|
.catch((error: Error) => {
|
|
setError(error);
|
|
});
|
|
};
|
|
|
|
window.addEventListener(
|
|
ACTION_MENU_CONFIRMATION_MODAL_RESULT_BROWSER_EVENT_NAME,
|
|
handleActionMenuConfirmationModalResultBrowserEvent as EventListener,
|
|
);
|
|
|
|
setThread(thread);
|
|
|
|
thread.imports
|
|
.render(newReceiver.connection, {
|
|
componentUrl,
|
|
applicationAccessToken,
|
|
apiUrl,
|
|
})
|
|
.catch((error: Error) => {
|
|
setError(error);
|
|
});
|
|
|
|
setReceiver(newReceiver);
|
|
isInitializedRef.current = true;
|
|
|
|
return () => {
|
|
window.removeEventListener(
|
|
ACTION_MENU_CONFIRMATION_MODAL_RESULT_BROWSER_EVENT_NAME,
|
|
handleActionMenuConfirmationModalResultBrowserEvent as EventListener,
|
|
);
|
|
setThread(null);
|
|
worker.terminate();
|
|
isInitializedRef.current = false;
|
|
};
|
|
}, [
|
|
componentUrl,
|
|
applicationAccessToken,
|
|
apiUrl,
|
|
frontComponentId,
|
|
setError,
|
|
setReceiver,
|
|
setThread,
|
|
frontComponentHostCommunicationApi,
|
|
]);
|
|
|
|
return null;
|
|
};
|