f47ddb988f
* feat: Support hypelinks * dont show edit when link not present * auto submit on blur * Add link button in sidebar and do it react way * add key to hyperlink to remount when element selection changes * autofocus input * remove click handler and use pointerup/down to show /hide popup * add keydown and support enter/escape to submit * show extrrnal link icon when element has link * use icons and open link in new tab * dnt submit unless link updated * renamed ffiles * remove unnecessary changes * update snap * hide link popup once user starts interacting with element and show again only if clicked outside and clicked on element again * render link icon outside the element * fix hit testing * rewrite implementation to render hyperlinks outside elements and hide when element selected * remove * remove * tweak icon position and size * rotate link icon when element rotated, handle zooming and render exactly where ne resize handle is rendered * no need to create a new reference anymore for element when link added/updated * rotate the link image as well when rotating element * calculate hitbox of link icon and show pointer when hovering over link icon * open link when clicked on link icon * show tooltip when hovering over link icon * show link action only when single element selected * support other protocols * add shortcut cmd/ctrl+k to edit/update link * don't hide popup after submit * renderes decreased woo * Add context mneu label to add/edit link * fix tests * remove tick and show trash when in edit mode * show edit view when element contains link * fix snap * horizontally center the hyperlink container with respect to elemnt * fix padding * remove checkcircle * show popup on hover of selected element and dismiss when outside hitbox * check if element has link before setting popup state * move logic of auto hide to hyperlink and dnt hide when editing * hide popover when drag/resize/rotate * unmount during autohide * autohide after 500ms * fix regression * prevent cmd/ctrl+k when inside link editor * submit when input not updated * allow custom urls * fix centering of popup when zoomed * fix hitbox during zoom * fix * tweak link normalization * touch hyperlink tooltip DOM only if needed * consider 0 if no offsetY * reduce hitbox of link icon and make sure link icon doesn't show on top of higher z-index elements * show link tooltip only if element has higher z-index * dnt show hyperlink popup when selection changes from element with link to element with no link and also hide popover when element type changes from selection to something else * lint: EOL * fix link icon tooltip positioning * open the link only when last pointer down and last pointer up hit the link hitbox * render tooltip after 300ms delay * ensure link popup and editor input have same height * wip: cache the link icon canvas * fix the image quality after caching using device pixel ratio yay * some cleanup * remove unused selectedElementIds from renderConfig * Update src/renderer/renderElement.ts * fix `opener` vulnerability * tweak styling * decrease padding * open local links in the same tab * fix caching * code style refactor * remove unnecessary save & restore * show link shortcut in help dialog * submit on cmd/ctrl+k * merge state props * Add title for link * update editview if prop changes * tweak link action logic * make `Hyperlink` compo editor state fully controlled * dont show popup when context menu open * show in contextMenu only for single selection & change pos * set button `selected` state * set contextMenuOpen on pointerdown * set contextMenyOpen to false when action triggered * don't render link icons on export * fix tests * fix buttons wrap * move focus states to input top-level rule * fix elements sharing `Hyperlink` state * fix hitbox for link icon in case of rect * Early return if hitting link icon Co-authored-by: dwelle <luzar.david@gmail.com>
216 lines
8.7 KiB
TypeScript
216 lines
8.7 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 {
|
|
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,
|
|
elementLocked: false,
|
|
elementType: "selection",
|
|
penMode: false,
|
|
penDetected: false,
|
|
errorMessage: null,
|
|
exportBackground: true,
|
|
exportScale: defaultExportScale,
|
|
exportEmbedScene: false,
|
|
exportWithDarkMode: false,
|
|
fileHandle: null,
|
|
gridSize: null,
|
|
isBindingEnabled: true,
|
|
isLibraryOpen: false,
|
|
isLoading: false,
|
|
isResizing: false,
|
|
isRotating: false,
|
|
lastPointerDownWith: "mouse",
|
|
multiElement: null,
|
|
name: `${t("labels.untitled")}-${getDateTime()}`,
|
|
openMenu: null,
|
|
openPopup: null,
|
|
pasteDialog: { shown: false, data: null },
|
|
previousSelectedElementIds: {},
|
|
resizingElement: null,
|
|
scrolledOutside: false,
|
|
scrollX: 0,
|
|
scrollY: 0,
|
|
selectedElementIds: {},
|
|
selectedGroupIds: {},
|
|
selectionElement: null,
|
|
shouldCacheIgnoreZoom: false,
|
|
showHelpDialog: false,
|
|
showStats: false,
|
|
startBoundElement: null,
|
|
suggestedBindings: [],
|
|
toastMessage: null,
|
|
viewBackgroundColor: oc.white,
|
|
zenModeEnabled: false,
|
|
zoom: {
|
|
value: 1 as NormalizedZoomValue,
|
|
},
|
|
viewModeEnabled: false,
|
|
pendingImageElement: null,
|
|
showHyperlinkPopup: false,
|
|
};
|
|
};
|
|
|
|
/**
|
|
* 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)({
|
|
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 },
|
|
elementLocked: { browser: true, export: false, server: false },
|
|
elementType: { browser: true, export: false, server: false },
|
|
penMode: { browser: false, export: false, server: false },
|
|
penDetected: { browser: false, 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 },
|
|
isLibraryOpen: { browser: false, 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 },
|
|
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 },
|
|
showHelpDialog: { browser: false, 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 },
|
|
toastMessage: { 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 },
|
|
pendingImageElement: { browser: false, export: false, server: false },
|
|
showHyperlinkPopup: { browser: false, 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");
|
|
};
|