diff --git a/packages/engine/src/HedronEngine.ts b/packages/engine/src/HedronEngine.ts index 975084cd..410fac43 100644 --- a/packages/engine/src/HedronEngine.ts +++ b/packages/engine/src/HedronEngine.ts @@ -1,10 +1,11 @@ import { listenToStore } from './storeListener' import { importSketchModule } from './importSketchModule' +import { Result } from './types' import { stripForSave } from '@utils/stripForSave' import { Renderer } from '@world/Renderer' import { SketchManager } from '@world/SketchManager' import { createDebugScene } from '@world/debugScene' -import { EngineData } from '@store/types' +import { EngineData, SketchModuleItem } from '@store/types' import { getSketchesOfModuleId } from '@store/selectors/getSketchesOfModuleId' import { createEngineStore, EngineStore } from '@store/engineStore' @@ -42,21 +43,34 @@ export class HedronEngine { this.store.setState({ isSketchModulesReady: true }) } - public async addSketchModule(moduleId: string) { + public async addSketchModule(moduleId: string): Promise> { if (!this.sketchesUrl) throw new Error('Sketches URL not ready') - const moduleItem = await importSketchModule(this.sketchesUrl, moduleId) + const result = await importSketchModule(this.sketchesUrl, moduleId) + + if (!result.success) { + // TODO: Show UI error here (engine needs to have some "error" state slice) + return result + } + + const moduleItem = result.data this.store.getState().setSketchModuleItem(moduleItem) - return moduleItem + return result } public removeSketchModule = async (moduleId: string): Promise => { this.store.getState().deleteSketchModule(moduleId) } - public async reimportSketchModuleAndReloadSketches(moduleId: string) { - const moduleItem = await this.addSketchModule(moduleId) + public async reimportSketchModuleAndReloadSketches(moduleId: string): Promise { + const result = await this.addSketchModule(moduleId) + + if (!result.success) { + return + } + + const moduleItem = result.data const sketchesToRefresh = getSketchesOfModuleId(this.store.getState(), moduleId) diff --git a/packages/engine/src/importSketchModule.ts b/packages/engine/src/importSketchModule.ts index ec10aa8e..f0fe821f 100644 --- a/packages/engine/src/importSketchModule.ts +++ b/packages/engine/src/importSketchModule.ts @@ -1,46 +1,61 @@ +import { Result } from './types' import { SketchConfig, SketchModule, SketchModuleItem } from '@store/types' import { createUniqueId } from '@utils/createUniqueId' export const importSketchModule = async ( baseUrl: string, moduleId: string, -): Promise => { - const cacheBust = createUniqueId() +): Promise> => { + try { + const cacheBust = createUniqueId() - // Get the sketch module - const sketchPath = `${baseUrl}/${moduleId}/index.js?${cacheBust}` - if ((await fetch(sketchPath)).status !== 200) { - return Promise.reject(`Sketch module not found: ${sketchPath}`) - } - const sketchModule = await import(/* @vite-ignore */ sketchPath) - const module: SketchModule = sketchModule.default + // Get the sketch module + const sketchPath = `${baseUrl}/${moduleId}/index.js?${cacheBust}` + if ((await fetch(sketchPath)).status !== 200) { + return Promise.reject(`Sketch module not found: ${sketchPath}`) + } + const sketchModule = await import(/* @vite-ignore */ sketchPath) + const module: SketchModule = sketchModule.default - // Get the sketch config - const configPath = `${baseUrl}/${moduleId}/config.js?${cacheBust}` - let config: SketchConfig - if ((await fetch(configPath)).status !== 200) { - // No config file found - // Try instancing the sketch, and call getConfig() on it - const tempModule = new module() - config = tempModule.getConfig?.() - if (!config) { - return Promise.reject( - `Sketch config not found: ${configPath} and no valid getConfig() function found in sketch`, - ) + // Get the sketch config + const configPath = `${baseUrl}/${moduleId}/config.js?${cacheBust}` + let config: SketchConfig + if ((await fetch(configPath)).status !== 200) { + // No config file found + // Try instancing the sketch, and call getConfig() on it + const tempModule = new module() + config = tempModule.getConfig?.() + if (!config) { + return Promise.reject( + `Sketch config not found: ${configPath} and no valid getConfig() function found in sketch`, + ) + } + } else { + const configModule = await import(/* @vite-ignore */ configPath) + config = configModule.default } - } else { - const configModule = await import(/* @vite-ignore */ configPath) - config = configModule.default - } - // A config could be missing a title, but it is a required parameter - if (!config.title) { - config.title = moduleId - } + // A config could be missing a title, but it is a required parameter + if (!config.title) { + config.title = moduleId + } + + return { + success: true, + error: undefined, + data: { + moduleId, + config, + module, + }, + } + } catch (error) { + console.error(error) - return { - moduleId, - config, - module, + return { + data: undefined, + success: false, + error: `[HEDRON] Sketch module failed to import: ${moduleId}`, + } } } diff --git a/packages/engine/src/types.ts b/packages/engine/src/types.ts new file mode 100644 index 00000000..5a29ba95 --- /dev/null +++ b/packages/engine/src/types.ts @@ -0,0 +1,3 @@ +export type Result = + | { success: true; data: T; error: undefined } + | { success: false; error: string; data: undefined }