6334bd832f
* Placed eraser into shape switcher (top toolbar). Redesigned top toolbar. * Redesigned zoom and undo-redo buttons. * Started redesigning left toolbar. * Redesigned help dialog. * Colour picker now somewhat in line with new design * [WIP] Changed a bunch of icons. TODO: organise new icons. * [WIP] Organised a bunch of icons. Still some to do * [WIP] Started working on hamburger menu. * Fixed some bugs with hamburger menu. * Menu and left toolbar positioning. * Added some more items to hamburger menu. * Changed some icons. * Modal/dialog styling & bunch of fixes. * Some more dialog improvements & fixes. * Mobile menu changes. * Menu can now be closed with outside click. * Collab avatars and button changes. * Icon sizing. Left toolbar positioning. * Implemented welcome screen rendering logic. * [WIP] Welcome screen content + design. * Some more welcome screen content and design. * Merge fixes. * Tweaked icon set. * Welcome screen darkmode fix. * Content updates. * Various small fixes & adjustments. Moved language selection into menu. Fixed some problematic icons. Slightly moved encryption icon. * Sidebar header redesign. * Libraries content rendering logic + some styling. * Somem more library sidebar styling. * Publish library dialog styling. * scroll-back-to-content btn styling * ColorPicker positioning. * Library button styling. * ColorPicker positioning "fix". * Misc adjustments. * PenMode button changes. * Trying to make mobile somewhat usable. * Added a couple of icons. * Added some shortcuts. * Prevent welcome screen flickering. Fix issue with welcome screen interactivity. Don't show sidebar button when docked. * Icon sizing on smaller screens. * Sidebar styling changes. * Alignment button... well... alignments. * Fix inconsistent padding in left toolbar. * HintViewer changes. * Hamburger menu changes. * Move encryption badge back to its original pos. * Arrowhead changes. Active state, colours + stronger shadow. * Added new custom font. * Fixed bug with library button not rendering. * Fixed issue with lang selection colours. * Add tooltips for undo, redo. * Address some dark mode contrast issues. * (Re)introduce counter for selectedItems in sidebar * [WIP] Tweaked bounding box colour & padding. * Dashed bounding box for remote clients. * Some more bounding box tweaks. * Removed docking animation for now... * Address some RTL issues. * Welcome screen responsiveness. * use lighter selection color in dark mode & align naming * use rounded corners for transform handles * use lighter gray for welcomeScreen text in dark mode * disable selection on dialog buttons * change selection button icon * fix library item width being flexible * library: visually align spinner with first section heading * lint * fix scrollbar color in dark mode & make thinner * adapt properties panel max-height * add shrotcut label to save-to-current-file * fix unrelated `useOutsideClick` firing for active modal * add promo color to e+ menu item * fix type * lowered button size * fix transform handles raidus not accounting for zoom * attempt fix for excal logo on safari * final fix for excal logo on safari * fixing fhd resolution button sized * remove TODO shortcut * Collab related styling changes. Expanding avatar list no longer offsets top toolbar. Added active state & collaborator count badge for collab button. * Tweaked collab button active colours. * Added active style to collab btn in hamburger menu * Remove unnecessary comment. * Added back promo link for non (signed in) E+ users * Go to E+ button added for signed in E+ users. * Close menu & dropdown on modal close. * tweak icons & fix rendering on smaller sizes [part one] * align welcomeScreen icons with other UI * switch icon resize mq to `device-width` * disable welcomeScreen items `:hover` when selecting on canvas * change selection box color and style * reduce selection padding and fix group selection styling * improve collab cursor styling - make name borders round - hide status when "active" - remove black/gray colors * add Twitter to hamburger menu * align collab button * add shortcut for image export dialog * revert yarn.lock * fix more tabler icons * slightly better-looking penMode button * change penMode button & tooltip * revert hamburger menu icon * align padding on lang picker & canvas bg * updated robot txt to allow twitter bot and fb bot * added new OG and tweaked the OG state * add tooltip to collab button * align style for scroll-to-content button * fix pointer-events around toolbar * fix decor arrow positioning and RTL * fix welcomeScreen-item active state in dark mode * change `load` button copy * prevent shadow anim when opening a docked sidebar * update E+ links ga params * show redirect-to-eplus welcomeScreen subheading for signed-in users * make more generic * add ga for eplus redirect button * change copy and icons for hamburger export buttons * update snaps * trim the username to account for trailing spaces * tweaks around decor breakpoints * fix linear element editor test * remove .env change * remove `it.only` Co-authored-by: dwelle <luzar.david@gmail.com> Co-authored-by: Maielo <maielo.mv@gmail.com> Co-authored-by: Aakansha Doshi <aakansha1216@gmail.com>
231 lines
9.1 KiB
TypeScript
231 lines
9.1 KiB
TypeScript
import oc from "open-color";
|
|
import {
|
|
DEFAULT_FONT_FAMILY,
|
|
DEFAULT_FONT_SIZE,
|
|
DEFAULT_TEXT_ALIGN,
|
|
EXPORT_SCALES,
|
|
THEME,
|
|
} from "./constants";
|
|
import { t } from "./i18n";
|
|
import { AppState, NormalizedZoomValue } from "./types";
|
|
import { getDateTime } from "./utils";
|
|
|
|
const defaultExportScale = EXPORT_SCALES.includes(devicePixelRatio)
|
|
? devicePixelRatio
|
|
: 1;
|
|
|
|
export const getDefaultAppState = (): Omit<
|
|
AppState,
|
|
"offsetTop" | "offsetLeft" | "width" | "height"
|
|
> => {
|
|
return {
|
|
showWelcomeScreen: false,
|
|
theme: THEME.LIGHT,
|
|
collaborators: new Map(),
|
|
currentChartType: "bar",
|
|
currentItemBackgroundColor: "transparent",
|
|
currentItemEndArrowhead: "arrow",
|
|
currentItemFillStyle: "hachure",
|
|
currentItemFontFamily: DEFAULT_FONT_FAMILY,
|
|
currentItemFontSize: DEFAULT_FONT_SIZE,
|
|
currentItemLinearStrokeSharpness: "round",
|
|
currentItemOpacity: 100,
|
|
currentItemRoughness: 1,
|
|
currentItemStartArrowhead: null,
|
|
currentItemStrokeColor: oc.black,
|
|
currentItemStrokeSharpness: "sharp",
|
|
currentItemStrokeStyle: "solid",
|
|
currentItemStrokeWidth: 1,
|
|
currentItemTextAlign: DEFAULT_TEXT_ALIGN,
|
|
cursorButton: "up",
|
|
draggingElement: null,
|
|
editingElement: null,
|
|
editingGroupId: null,
|
|
editingLinearElement: null,
|
|
activeTool: {
|
|
type: "selection",
|
|
customType: null,
|
|
locked: false,
|
|
lastActiveToolBeforeEraser: null,
|
|
},
|
|
penMode: false,
|
|
penDetected: false,
|
|
errorMessage: null,
|
|
exportBackground: true,
|
|
exportScale: defaultExportScale,
|
|
exportEmbedScene: false,
|
|
exportWithDarkMode: false,
|
|
fileHandle: null,
|
|
gridSize: null,
|
|
isBindingEnabled: true,
|
|
isSidebarDocked: false,
|
|
isLoading: false,
|
|
isResizing: false,
|
|
isRotating: false,
|
|
lastPointerDownWith: "mouse",
|
|
multiElement: null,
|
|
name: `${t("labels.untitled")}-${getDateTime()}`,
|
|
openMenu: null,
|
|
openPopup: null,
|
|
openSidebar: null,
|
|
openDialog: null,
|
|
pasteDialog: { shown: false, data: null },
|
|
previousSelectedElementIds: {},
|
|
resizingElement: null,
|
|
scrolledOutside: false,
|
|
scrollX: 0,
|
|
scrollY: 0,
|
|
selectedElementIds: {},
|
|
selectedGroupIds: {},
|
|
selectionElement: null,
|
|
shouldCacheIgnoreZoom: false,
|
|
showStats: false,
|
|
startBoundElement: null,
|
|
suggestedBindings: [],
|
|
toast: null,
|
|
viewBackgroundColor: oc.white,
|
|
zenModeEnabled: false,
|
|
zoom: {
|
|
value: 1 as NormalizedZoomValue,
|
|
},
|
|
viewModeEnabled: false,
|
|
pendingImageElementId: null,
|
|
showHyperlinkPopup: false,
|
|
selectedLinearElement: null,
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Config containing all AppState keys. Used to determine whether given state
|
|
* prop should be stripped when exporting to given storage type.
|
|
*/
|
|
const APP_STATE_STORAGE_CONF = (<
|
|
Values extends {
|
|
/** whether to keep when storing to browser storage (localStorage/IDB) */
|
|
browser: boolean;
|
|
/** whether to keep when exporting to file/database */
|
|
export: boolean;
|
|
/** server (shareLink/collab/...) */
|
|
server: boolean;
|
|
},
|
|
T extends Record<keyof AppState, Values>,
|
|
>(config: { [K in keyof T]: K extends keyof AppState ? T[K] : never }) =>
|
|
config)({
|
|
showWelcomeScreen: { browser: true, export: false, server: false },
|
|
theme: { browser: true, export: false, server: false },
|
|
collaborators: { browser: false, export: false, server: false },
|
|
currentChartType: { browser: true, export: false, server: false },
|
|
currentItemBackgroundColor: { browser: true, export: false, server: false },
|
|
currentItemEndArrowhead: { browser: true, export: false, server: false },
|
|
currentItemFillStyle: { browser: true, export: false, server: false },
|
|
currentItemFontFamily: { browser: true, export: false, server: false },
|
|
currentItemFontSize: { browser: true, export: false, server: false },
|
|
currentItemLinearStrokeSharpness: {
|
|
browser: true,
|
|
export: false,
|
|
server: false,
|
|
},
|
|
currentItemOpacity: { browser: true, export: false, server: false },
|
|
currentItemRoughness: { browser: true, export: false, server: false },
|
|
currentItemStartArrowhead: { browser: true, export: false, server: false },
|
|
currentItemStrokeColor: { browser: true, export: false, server: false },
|
|
currentItemStrokeSharpness: { browser: true, export: false, server: false },
|
|
currentItemStrokeStyle: { browser: true, export: false, server: false },
|
|
currentItemStrokeWidth: { browser: true, export: false, server: false },
|
|
currentItemTextAlign: { browser: true, export: false, server: false },
|
|
cursorButton: { browser: true, export: false, server: false },
|
|
draggingElement: { browser: false, export: false, server: false },
|
|
editingElement: { browser: false, export: false, server: false },
|
|
editingGroupId: { browser: true, export: false, server: false },
|
|
editingLinearElement: { browser: false, export: false, server: false },
|
|
activeTool: { browser: true, export: false, server: false },
|
|
penMode: { browser: true, export: false, server: false },
|
|
penDetected: { browser: true, export: false, server: false },
|
|
errorMessage: { browser: false, export: false, server: false },
|
|
exportBackground: { browser: true, export: false, server: false },
|
|
exportEmbedScene: { browser: true, export: false, server: false },
|
|
exportScale: { browser: true, export: false, server: false },
|
|
exportWithDarkMode: { browser: true, export: false, server: false },
|
|
fileHandle: { browser: false, export: false, server: false },
|
|
gridSize: { browser: true, export: true, server: true },
|
|
height: { browser: false, export: false, server: false },
|
|
isBindingEnabled: { browser: false, export: false, server: false },
|
|
isSidebarDocked: { browser: true, export: false, server: false },
|
|
isLoading: { browser: false, export: false, server: false },
|
|
isResizing: { browser: false, export: false, server: false },
|
|
isRotating: { browser: false, export: false, server: false },
|
|
lastPointerDownWith: { browser: true, export: false, server: false },
|
|
multiElement: { browser: false, export: false, server: false },
|
|
name: { browser: true, export: false, server: false },
|
|
offsetLeft: { browser: false, export: false, server: false },
|
|
offsetTop: { browser: false, export: false, server: false },
|
|
openMenu: { browser: true, export: false, server: false },
|
|
openPopup: { browser: false, export: false, server: false },
|
|
openSidebar: { browser: true, export: false, server: false },
|
|
openDialog: { browser: false, export: false, server: false },
|
|
pasteDialog: { browser: false, export: false, server: false },
|
|
previousSelectedElementIds: { browser: true, export: false, server: false },
|
|
resizingElement: { browser: false, export: false, server: false },
|
|
scrolledOutside: { browser: true, export: false, server: false },
|
|
scrollX: { browser: true, export: false, server: false },
|
|
scrollY: { browser: true, export: false, server: false },
|
|
selectedElementIds: { browser: true, export: false, server: false },
|
|
selectedGroupIds: { browser: true, export: false, server: false },
|
|
selectionElement: { browser: false, export: false, server: false },
|
|
shouldCacheIgnoreZoom: { browser: true, export: false, server: false },
|
|
showStats: { browser: true, export: false, server: false },
|
|
startBoundElement: { browser: false, export: false, server: false },
|
|
suggestedBindings: { browser: false, export: false, server: false },
|
|
toast: { browser: false, export: false, server: false },
|
|
viewBackgroundColor: { browser: true, export: true, server: true },
|
|
width: { browser: false, export: false, server: false },
|
|
zenModeEnabled: { browser: true, export: false, server: false },
|
|
zoom: { browser: true, export: false, server: false },
|
|
viewModeEnabled: { browser: false, export: false, server: false },
|
|
pendingImageElementId: { browser: false, export: false, server: false },
|
|
showHyperlinkPopup: { browser: false, export: false, server: false },
|
|
selectedLinearElement: { browser: true, export: false, server: false },
|
|
});
|
|
|
|
const _clearAppStateForStorage = <
|
|
ExportType extends "export" | "browser" | "server",
|
|
>(
|
|
appState: Partial<AppState>,
|
|
exportType: ExportType,
|
|
) => {
|
|
type ExportableKeys = {
|
|
[K in keyof typeof APP_STATE_STORAGE_CONF]: typeof APP_STATE_STORAGE_CONF[K][ExportType] extends true
|
|
? K
|
|
: never;
|
|
}[keyof typeof APP_STATE_STORAGE_CONF];
|
|
const stateForExport = {} as { [K in ExportableKeys]?: typeof appState[K] };
|
|
for (const key of Object.keys(appState) as (keyof typeof appState)[]) {
|
|
const propConfig = APP_STATE_STORAGE_CONF[key];
|
|
if (propConfig?.[exportType]) {
|
|
const nextValue = appState[key];
|
|
|
|
// https://github.com/microsoft/TypeScript/issues/31445
|
|
(stateForExport as any)[key] = nextValue;
|
|
}
|
|
}
|
|
return stateForExport;
|
|
};
|
|
|
|
export const clearAppStateForLocalStorage = (appState: Partial<AppState>) => {
|
|
return _clearAppStateForStorage(appState, "browser");
|
|
};
|
|
|
|
export const cleanAppStateForExport = (appState: Partial<AppState>) => {
|
|
return _clearAppStateForStorage(appState, "export");
|
|
};
|
|
|
|
export const clearAppStateForDatabase = (appState: Partial<AppState>) => {
|
|
return _clearAppStateForStorage(appState, "server");
|
|
};
|
|
|
|
export const isEraserActive = ({
|
|
activeTool,
|
|
}: {
|
|
activeTool: AppState["activeTool"];
|
|
}) => activeTool.type === "eraser";
|