import { load, questionCircle, saveAs } from "../components/icons"; import { ProjectName } from "../components/ProjectName"; import { ToolButton } from "../components/ToolButton"; import "../components/ToolIcon.scss"; import { Tooltip } from "../components/Tooltip"; import { DarkModeToggle } from "../components/DarkModeToggle"; import { loadFromJSON, saveAsJSON } from "../data"; import { resaveAsImageWithScene } from "../data/resave"; import { t } from "../i18n"; import { useDevice } from "../components/App"; import { KEYS } from "../keys"; import { register } from "./register"; import { CheckboxItem } from "../components/CheckboxItem"; import { getExportSize } from "../scene/export"; import { DEFAULT_EXPORT_PADDING, EXPORT_SCALES, THEME } from "../constants"; import { getSelectedElements, isSomeElementSelected } from "../scene"; import { getNonDeletedElements } from "../element"; import { ActiveFile } from "../components/ActiveFile"; import { isImageFileHandle } from "../data/blob"; import { nativeFileSystemSupported } from "../data/filesystem"; import { Theme } from "../element/types"; export const actionChangeProjectName = register({ name: "changeProjectName", trackEvent: false, perform: (_elements, appState, value) => { return { appState: { ...appState, name: value }, commitToHistory: false }; }, PanelComponent: ({ appState, updateData, appProps }) => ( updateData(name)} isNameEditable={ typeof appProps.name === "undefined" && !appState.viewModeEnabled } /> ), }); export const actionChangeExportScale = register({ name: "changeExportScale", trackEvent: { category: "export", action: "scale" }, perform: (_elements, appState, value) => { return { appState: { ...appState, exportScale: value }, commitToHistory: false, }; }, PanelComponent: ({ elements: allElements, appState, updateData }) => { const elements = getNonDeletedElements(allElements); const exportSelected = isSomeElementSelected(elements, appState); const exportedElements = exportSelected ? getSelectedElements(elements, appState) : elements; return ( <> {EXPORT_SCALES.map((s) => { const [width, height] = getExportSize( exportedElements, DEFAULT_EXPORT_PADDING, s, ); const scaleButtonTitle = `${t( "buttons.scale", )} ${s}x (${width}x${height})`; return ( updateData(s)} /> ); })} ); }, }); export const actionChangeExportBackground = register({ name: "changeExportBackground", trackEvent: { category: "export", action: "toggleBackground" }, perform: (_elements, appState, value) => { return { appState: { ...appState, exportBackground: value }, commitToHistory: false, }; }, PanelComponent: ({ appState, updateData }) => ( updateData(checked)} > {t("labels.withBackground")} ), }); export const actionChangeExportEmbedScene = register({ name: "changeExportEmbedScene", trackEvent: { category: "export", action: "embedScene" }, perform: (_elements, appState, value) => { return { appState: { ...appState, exportEmbedScene: value }, commitToHistory: false, }; }, PanelComponent: ({ appState, updateData }) => ( updateData(checked)} > {t("labels.exportEmbedScene")}
{questionCircle}
), }); export const actionSaveToActiveFile = register({ name: "saveToActiveFile", trackEvent: { category: "export" }, perform: async (elements, appState, value, app) => { const fileHandleExists = !!appState.fileHandle; try { const { fileHandle } = isImageFileHandle(appState.fileHandle) ? await resaveAsImageWithScene(elements, appState, app.files) : await saveAsJSON(elements, appState, app.files); return { commitToHistory: false, appState: { ...appState, fileHandle, toastMessage: fileHandleExists ? fileHandle?.name ? t("toast.fileSavedToFilename").replace( "{filename}", `"${fileHandle.name}"`, ) : t("toast.fileSaved") : null, }, }; } catch (error: any) { if (error?.name !== "AbortError") { console.error(error); } else { console.warn(error); } return { commitToHistory: false }; } }, keyTest: (event) => event.key === KEYS.S && event[KEYS.CTRL_OR_CMD] && !event.shiftKey, PanelComponent: ({ updateData, appState }) => ( updateData(null)} fileName={appState.fileHandle?.name} /> ), }); export const actionSaveFileToDisk = register({ name: "saveFileToDisk", trackEvent: { category: "export" }, perform: async (elements, appState, value, app) => { try { const { fileHandle } = await saveAsJSON( elements, { ...appState, fileHandle: null, }, app.files, ); return { commitToHistory: false, appState: { ...appState, fileHandle } }; } catch (error: any) { if (error?.name !== "AbortError") { console.error(error); } else { console.warn(error); } return { commitToHistory: false }; } }, keyTest: (event) => event.key === KEYS.S && event.shiftKey && event[KEYS.CTRL_OR_CMD], PanelComponent: ({ updateData }) => (