import "./App.css"; import styled from "@emotion/styled"; import { useEffect, useState } from "react"; import { FileBar } from "./components/FileBar"; import WelcomeDialog from "./components/WelcomeDialog/WelcomeDialog"; import { get_documentation_model, get_model, uploadFile, } from "./components/rpc"; import { createNewModel, deleteSelectedModel, isStorageEmpty, loadSelectedModelFromStorage, saveModelToStorage, saveSelectedModelInStorage, selectModelFromStorage, } from "./components/storage"; // From IronCalc import { IronCalc, IronCalcIcon, Model, init } from "@ironcalc/workbook"; import { Modal } from "@mui/material"; import TemplatesDialog from "./components/WelcomeDialog/TemplatesDialog"; function App() { const [model, setModel] = useState(null); const [showWelcomeDialog, setShowWelcomeDialog] = useState(false); const [isTemplatesDialogOpen, setTemplatesDialogOpen] = useState(false); useEffect(() => { async function start() { await init(); const queryString = window.location.search; const urlParams = new URLSearchParams(queryString); const modelHash = urlParams.get("model"); const exampleFilename = urlParams.get("example"); // If there is a model name ?model=modelHash we try to load it // if there is not, or the loading failed we load an empty model if (modelHash) { // Get a remote model try { const model_bytes = await get_model(modelHash); const importedModel = Model.from_bytes(model_bytes); localStorage.removeItem("selected"); setModel(importedModel); } catch (e) { alert("Model not found, or failed to load"); } } else if (exampleFilename) { try { const model_bytes = await get_documentation_model(exampleFilename); const importedModel = Model.from_bytes(model_bytes); localStorage.removeItem("selected"); setModel(importedModel); } catch (e) { alert("Example file not found, or failed to load"); } } else { // try to load from local storage const newModel = loadSelectedModelFromStorage(); if (!newModel) { setShowWelcomeDialog(true); const createdModel = new Model("template", "en", "UTC"); setModel(createdModel); } else { setModel(newModel); } } } start(); }, []); if (!model) { return (
Loading IronCalc
); } // We try to save the model every second setInterval(() => { const queue = model.flushSendQueue(); if (queue.length !== 1) { saveSelectedModelInStorage(model); } }, 1000); // We could use context for model, but the problem is that it should initialized to null. // Passing the property down makes sure it is always defined. return ( { const blob = await uploadFile(arrayBuffer, fileName); const bytes = new Uint8Array(await blob.arrayBuffer()); const newModel = Model.from_bytes(bytes); saveModelToStorage(newModel); setModel(newModel); }} newModel={() => { const createdModel = createNewModel(); setModel(createdModel); }} newModelFromTemplate={() => { setTemplatesDialogOpen(true); }} setModel={(uuid: string) => { const newModel = selectModelFromStorage(uuid); if (newModel) { setModel(newModel); } }} onDelete={() => { const newModel = deleteSelectedModel(); if (newModel) { setModel(newModel); } }} /> {showWelcomeDialog && ( { if (isStorageEmpty()) { const createdModel = createNewModel(); setModel(createdModel); } setShowWelcomeDialog(false); }} onSelectTemplate={async (templateId) => { switch (templateId) { case "blank": { const createdModel = createNewModel(); setModel(createdModel); break; } default: { const model_bytes = await get_documentation_model(templateId); const importedModel = Model.from_bytes(model_bytes); saveModelToStorage(importedModel); setModel(importedModel); break; } } setShowWelcomeDialog(false); }} /> )} setTemplatesDialogOpen(false)} aria-labelledby="templates-dialog-title" aria-describedby="templates-dialog-description" > setTemplatesDialogOpen(false)} onSelectTemplate={async (fileName) => { const model_bytes = await get_documentation_model(fileName); const importedModel = Model.from_bytes(model_bytes); saveModelToStorage(importedModel); setModel(importedModel); setTemplatesDialogOpen(false); }} /> ); } const Wrapper = styled("div")` margin: 0px; padding: 0px; width: 100%; height: 100%; display: flex; flex-direction: column; position: absolute; `; const Loading = styled("div")` height: 100%; display: flex; flex-direction: column; align-items: center; justify-content: center; font-family: "Inter"; font-size: 14px; `; export default App;