diff --git a/src/modules/apps/builder/AppBuilder.tsx b/src/modules/apps/builder/AppBuilder.tsx
index f52035b..49f9acf 100644
--- a/src/modules/apps/builder/AppBuilder.tsx
+++ b/src/modules/apps/builder/AppBuilder.tsx
@@ -238,7 +238,7 @@ function AppBuilderContent() {
totalCount,
]);
- const handleReportError = useCallback(
+ const handleFixError = useCallback(
async (errorText: string) => {
await sendMessage(
`I have encountered the following error:\n\n\`\`\`error\n${errorText}\n\`\`\`\n\nFix this error please.`,
@@ -364,7 +364,7 @@ function AppBuilderContent() {
diff --git a/src/modules/apps/builder/ArtifactSharedIframe.tsx b/src/modules/apps/builder/ArtifactSharedIframe.tsx
index 58ab9d5..1c82505 100644
--- a/src/modules/apps/builder/ArtifactSharedIframe.tsx
+++ b/src/modules/apps/builder/ArtifactSharedIframe.tsx
@@ -19,7 +19,7 @@ import { createChatCompletion, modulesToPackages } from '@/app/api/apps';
import { ChatCompletionCreateBody } from '@/app/api/apps/types';
import { ApiError } from '@/app/api/errors';
import { useProjectContext } from '@/layout/providers/ProjectProvider';
-import { Theme, useTheme } from '@/layout/providers/ThemeProvider';
+import { useTheme } from '@/layout/providers/ThemeProvider';
import { USERCONTENT_SITE_URL } from '@/utils/constants';
import { removeTrailingSlash } from '@/utils/helpers';
import { Loading } from '@carbon/react';
@@ -29,7 +29,7 @@ import AppPlaceholder from './Placeholder.svg';
interface Props {
sourceCode: string | null;
- onReportError?: (errorText: string) => void;
+ onFixError?: (errorText: string) => void;
}
function getErrorMessage(error: unknown) {
@@ -42,39 +42,35 @@ function getErrorMessage(error: unknown) {
return 'Unknown error when calling LLM function.';
}
-export function ArtifactSharedIframe({ sourceCode, onReportError }: Props) {
+export function ArtifactSharedIframe({ sourceCode, onFixError }: Props) {
const iframeRef = useRef(null);
const [state, setState] = useState(State.LOADING);
const { appliedTheme: theme } = useTheme();
const { project, organization } = useProjectContext();
+ const [iframeLoadCount, setIframeLoadCount] = useState(0);
- const postMessage = (message: PostMessage) => {
+ const postMessage = useCallback((message: PostMessage) => {
+ if(iframeLoadCount === 0) return;
iframeRef.current?.contentWindow?.postMessage(
message,
USERCONTENT_SITE_URL,
);
- };
+ }, [iframeLoadCount])
- const updateTheme = useCallback((theme: Theme) => {
- postMessage({ type: PostMessageType.UPDATE_THEME, theme });
- }, []);
-
- const updateCode = useCallback(
- (code: string | null) => {
- if (!code) {
- return;
- }
-
- postMessage({
- type: PostMessageType.UPDATE_CODE,
+ useEffect(() => {
+ postMessage({
+ type: PostMessageType.UPDATE_STATE,
+ stateChange: {
+ code: sourceCode ?? undefined,
config: {
- can_fix_error: Boolean(onReportError),
+ canFixError: Boolean(onFixError)
},
- code,
- });
- },
- [onReportError],
- );
+ theme: theme ?? 'system',
+ fullscreen: false,
+ ancestorOrigin: window.location.origin,
+ }
+ })
+ }, [sourceCode, onFixError, theme, postMessage]);
const handleMessage = useCallback(
async (event: MessageEvent) => {
@@ -84,15 +80,19 @@ export function ArtifactSharedIframe({ sourceCode, onReportError }: Props) {
return;
}
- if (
- data.type === RecieveMessageType.SCRIPT_RUN_STATE_CHANGED &&
- data.scriptRunState === ScriptRunState.RUNNING
- ) {
+ if (data.type === RecieveMessageType.READY) {
setState(State.READY);
return;
}
if (data.type === RecieveMessageType.REQUEST) {
+ const respond = (payload: unknown = undefined) =>
+ postMessage({
+ type: PostMessageType.RESPONSE,
+ request_id: data.request_id,
+ payload,
+ });
+
try {
switch (data.request_type) {
case 'modules_to_packages':
@@ -101,12 +101,9 @@ export function ArtifactSharedIframe({ sourceCode, onReportError }: Props) {
project.id,
data.payload.modules,
);
- postMessage({
- type: PostMessageType.RESPONSE,
- request_id: data.request_id,
- payload: packagesResponse,
- });
+ respond(packagesResponse);
break;
+
case 'chat_completion':
const response = await createChatCompletion(
organization.id,
@@ -115,49 +112,23 @@ export function ArtifactSharedIframe({ sourceCode, onReportError }: Props) {
);
const message = response?.choices[0]?.message?.content;
if (!message) throw new Error(); // missing completion
- postMessage({
- type: PostMessageType.RESPONSE,
- request_id: data.request_id,
- payload: { message },
- });
+ respond({ message });
+ break;
+
+ case 'fix_error':
+ onFixError?.(data.payload.errorText);
+ respond();
break;
}
} catch (err) {
- postMessage({
- type: PostMessageType.RESPONSE,
- request_id: data.request_id,
- payload: { error: getErrorMessage(err) },
- });
+ respond({ error: getErrorMessage(err) });
}
return;
}
-
- if (data.type === RecieveMessageType.REPORT_ERROR) {
- onReportError?.(data.errorText);
- return;
- }
},
- [project, organization, onReportError],
+ [project, organization, onFixError, postMessage],
);
- const handleIframeLoad = useCallback(() => {
- if (theme) {
- updateTheme(theme);
- }
- }, [theme, updateTheme]);
-
- useEffect(() => {
- if (theme) {
- updateTheme(theme);
- }
- }, [theme, updateTheme]);
-
- useEffect(() => {
- if (state === State.READY) {
- updateCode(sourceCode);
- }
- }, [state, theme, sourceCode, updateCode]);
-
useEffect(() => {
window.addEventListener('message', handleMessage);
@@ -180,7 +151,7 @@ export function ArtifactSharedIframe({ sourceCode, onReportError }: Props) {
'allow-popups-to-escape-sandbox',
].join(' ')}
className={classes.app}
- onLoad={handleIframeLoad}
+ onLoad={() => setIframeLoadCount(i => i + 1)}
/>
{!sourceCode ? (
@@ -188,7 +159,7 @@ export function ArtifactSharedIframe({ sourceCode, onReportError }: Props) {
) : (
- state === State.LOADING && sourceCode &&
+ state === State.LOADING &&
)}
);
@@ -196,15 +167,16 @@ export function ArtifactSharedIframe({ sourceCode, onReportError }: Props) {
type PostMessage =
| {
- type: PostMessageType.UPDATE_CODE;
- code: string;
- config: {
- can_fix_error?: boolean;
- };
- }
- | {
- type: PostMessageType.UPDATE_THEME;
- theme: Theme;
+ type: PostMessageType.UPDATE_STATE;
+ stateChange: Partial<{
+ fullscreen: boolean,
+ theme: 'light' | 'dark' | 'system',
+ code: string,
+ config: {
+ canFixError: boolean
+ },
+ ancestorOrigin: string,
+ }>;
}
| {
type: PostMessageType.RESPONSE;
@@ -213,15 +185,8 @@ type PostMessage =
};
enum PostMessageType {
- UPDATE_CODE = 'bee:updateCode',
- UPDATE_THEME = 'bee:updateTheme',
RESPONSE = 'bee:response',
-}
-
-enum ScriptRunState {
- INITIAL = 'initial',
- NOT_RUNNING = 'notRunning',
- RUNNING = 'running',
+ UPDATE_STATE = 'bee:updateState'
}
enum State {
@@ -230,15 +195,13 @@ enum State {
}
enum RecieveMessageType {
- SCRIPT_RUN_STATE_CHANGED = 'SCRIPT_RUN_STATE_CHANGED',
+ READY = 'bee:ready',
REQUEST = 'bee:request',
- REPORT_ERROR = 'bee:reportError',
}
export type StliteMessage =
| {
- type: RecieveMessageType.SCRIPT_RUN_STATE_CHANGED;
- scriptRunState: ScriptRunState;
+ type: RecieveMessageType.READY;
}
| {
type: RecieveMessageType.REQUEST;
@@ -253,6 +216,8 @@ export type StliteMessage =
payload: ChatCompletionCreateBody;
}
| {
- type: RecieveMessageType.REPORT_ERROR;
- errorText: string;
+ type: RecieveMessageType.REQUEST;
+ request_type: 'fix_error';
+ request_id: string;
+ payload: { errorText: string };
};