mirror of
https://github.com/excalidraw/excalidraw.git
synced 2025-04-14 16:40:58 -04:00
* 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>
416 lines
14 KiB
TypeScript
416 lines
14 KiB
TypeScript
import React from "react";
|
|
import { t } from "../i18n";
|
|
import { isDarwin, isWindows } from "../keys";
|
|
import { Dialog } from "./Dialog";
|
|
import { getShortcutKey } from "../utils";
|
|
import "./HelpDialog.scss";
|
|
|
|
const Header = () => (
|
|
<div className="HelpDialog--header">
|
|
<a
|
|
className="HelpDialog--btn"
|
|
href="https://github.com/excalidraw/excalidraw#documentation"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
>
|
|
{t("helpDialog.documentation")}
|
|
</a>
|
|
<a
|
|
className="HelpDialog--btn"
|
|
href="https://blog.excalidraw.com"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
>
|
|
{t("helpDialog.blog")}
|
|
</a>
|
|
<a
|
|
className="HelpDialog--btn"
|
|
href="https://github.com/excalidraw/excalidraw/issues"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
>
|
|
{t("helpDialog.github")}
|
|
</a>
|
|
</div>
|
|
);
|
|
|
|
const Section = (props: { title: string; children: React.ReactNode }) => (
|
|
<>
|
|
<h3>{props.title}</h3>
|
|
{props.children}
|
|
</>
|
|
);
|
|
|
|
const Columns = (props: { children: React.ReactNode }) => (
|
|
<div
|
|
style={{
|
|
display: "flex",
|
|
flexDirection: "row",
|
|
flexWrap: "wrap",
|
|
justifyContent: "space-between",
|
|
}}
|
|
>
|
|
{props.children}
|
|
</div>
|
|
);
|
|
|
|
const Column = (props: { children: React.ReactNode }) => (
|
|
<div style={{ width: "49%" }}>{props.children}</div>
|
|
);
|
|
|
|
const ShortcutIsland = (props: {
|
|
caption: string;
|
|
children: React.ReactNode;
|
|
}) => (
|
|
<div className="HelpDialog--island">
|
|
<h3 className="HelpDialog--island-title">{props.caption}</h3>
|
|
{props.children}
|
|
</div>
|
|
);
|
|
|
|
const Shortcut = (props: {
|
|
label: string;
|
|
shortcuts: string[];
|
|
isOr: boolean;
|
|
}) => {
|
|
return (
|
|
<div className="HelpDialog--shortcut">
|
|
<div
|
|
style={{
|
|
display: "flex",
|
|
margin: "0",
|
|
padding: "4px 8px",
|
|
alignItems: "center",
|
|
}}
|
|
>
|
|
<div
|
|
style={{
|
|
lineHeight: 1.4,
|
|
}}
|
|
>
|
|
{props.label}
|
|
</div>
|
|
<div
|
|
style={{
|
|
display: "flex",
|
|
flex: "0 0 auto",
|
|
justifyContent: "flex-end",
|
|
marginInlineStart: "auto",
|
|
minWidth: "30%",
|
|
}}
|
|
>
|
|
{props.shortcuts.map((shortcut, index) => (
|
|
<React.Fragment key={index}>
|
|
<ShortcutKey>{shortcut}</ShortcutKey>
|
|
{props.isOr &&
|
|
index !== props.shortcuts.length - 1 &&
|
|
t("helpDialog.or")}
|
|
</React.Fragment>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
Shortcut.defaultProps = {
|
|
isOr: true,
|
|
};
|
|
|
|
const ShortcutKey = (props: { children: React.ReactNode }) => (
|
|
<kbd className="HelpDialog--key" {...props} />
|
|
);
|
|
|
|
export const HelpDialog = ({ onClose }: { onClose?: () => void }) => {
|
|
const handleClose = React.useCallback(() => {
|
|
if (onClose) {
|
|
onClose();
|
|
}
|
|
}, [onClose]);
|
|
|
|
return (
|
|
<>
|
|
<Dialog
|
|
onCloseRequest={handleClose}
|
|
title={t("helpDialog.title")}
|
|
className={"HelpDialog"}
|
|
>
|
|
<Header />
|
|
<Section title={t("helpDialog.shortcuts")}>
|
|
<Columns>
|
|
<Column>
|
|
<ShortcutIsland caption={t("helpDialog.shapes")}>
|
|
<Shortcut
|
|
label={t("toolBar.selection")}
|
|
shortcuts={["V", "1"]}
|
|
/>
|
|
<Shortcut
|
|
label={t("toolBar.rectangle")}
|
|
shortcuts={["R", "2"]}
|
|
/>
|
|
<Shortcut label={t("toolBar.diamond")} shortcuts={["D", "3"]} />
|
|
<Shortcut label={t("toolBar.ellipse")} shortcuts={["E", "4"]} />
|
|
<Shortcut label={t("toolBar.arrow")} shortcuts={["A", "5"]} />
|
|
<Shortcut label={t("toolBar.line")} shortcuts={["P", "6"]} />
|
|
<Shortcut
|
|
label={t("toolBar.freedraw")}
|
|
shortcuts={["Shift + P", "X", "7"]}
|
|
/>
|
|
<Shortcut label={t("toolBar.text")} shortcuts={["T", "8"]} />
|
|
<Shortcut label={t("toolBar.image")} shortcuts={["9"]} />
|
|
<Shortcut label={t("toolBar.library")} shortcuts={["0"]} />
|
|
<Shortcut
|
|
label={t("helpDialog.editSelectedShape")}
|
|
shortcuts={[
|
|
getShortcutKey("Enter"),
|
|
t("helpDialog.doubleClick"),
|
|
]}
|
|
/>
|
|
<Shortcut
|
|
label={t("helpDialog.textNewLine")}
|
|
shortcuts={[
|
|
getShortcutKey("Enter"),
|
|
getShortcutKey("Shift+Enter"),
|
|
]}
|
|
/>
|
|
<Shortcut
|
|
label={t("helpDialog.textFinish")}
|
|
shortcuts={[
|
|
getShortcutKey("Esc"),
|
|
getShortcutKey("CtrlOrCmd+Enter"),
|
|
]}
|
|
/>
|
|
<Shortcut
|
|
label={t("helpDialog.curvedArrow")}
|
|
shortcuts={[
|
|
"A",
|
|
t("helpDialog.click"),
|
|
t("helpDialog.click"),
|
|
t("helpDialog.click"),
|
|
]}
|
|
isOr={false}
|
|
/>
|
|
<Shortcut
|
|
label={t("helpDialog.curvedLine")}
|
|
shortcuts={[
|
|
"L",
|
|
t("helpDialog.click"),
|
|
t("helpDialog.click"),
|
|
t("helpDialog.click"),
|
|
]}
|
|
isOr={false}
|
|
/>
|
|
<Shortcut label={t("toolBar.lock")} shortcuts={["Q"]} />
|
|
<Shortcut
|
|
label={t("helpDialog.preventBinding")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("toolBar.link")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+K")]}
|
|
/>
|
|
</ShortcutIsland>
|
|
<ShortcutIsland caption={t("helpDialog.view")}>
|
|
<Shortcut
|
|
label={t("buttons.zoomIn")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd++")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("buttons.zoomOut")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+-")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("buttons.resetZoom")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+0")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("helpDialog.zoomToFit")}
|
|
shortcuts={["Shift+1"]}
|
|
/>
|
|
<Shortcut
|
|
label={t("helpDialog.zoomToSelection")}
|
|
shortcuts={["Shift+2"]}
|
|
/>
|
|
<Shortcut label={t("buttons.fullScreen")} shortcuts={["F"]} />
|
|
<Shortcut
|
|
label={t("buttons.zenMode")}
|
|
shortcuts={[getShortcutKey("Alt+Z")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.showGrid")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+'")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.viewMode")}
|
|
shortcuts={[getShortcutKey("Alt+R")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.toggleTheme")}
|
|
shortcuts={[getShortcutKey("Alt+Shift+D")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("stats.title")}
|
|
shortcuts={[getShortcutKey("Alt+/")]}
|
|
/>
|
|
</ShortcutIsland>
|
|
</Column>
|
|
<Column>
|
|
<ShortcutIsland caption={t("helpDialog.editor")}>
|
|
<Shortcut
|
|
label={t("labels.selectAll")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+A")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.multiSelect")}
|
|
shortcuts={[getShortcutKey(`Shift+${t("helpDialog.click")}`)]}
|
|
/>
|
|
<Shortcut
|
|
label={t("helpDialog.deepSelect")}
|
|
shortcuts={[
|
|
getShortcutKey(`CtrlOrCmd+${t("helpDialog.click")}`),
|
|
]}
|
|
/>
|
|
<Shortcut
|
|
label={t("helpDialog.deepBoxSelect")}
|
|
shortcuts={[
|
|
getShortcutKey(`CtrlOrCmd+${t("helpDialog.drag")}`),
|
|
]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.moveCanvas")}
|
|
shortcuts={[
|
|
getShortcutKey(`Space+${t("helpDialog.drag")}`),
|
|
getShortcutKey(`Wheel+${t("helpDialog.drag")}`),
|
|
]}
|
|
isOr={true}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.cut")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+X")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.copy")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+C")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.paste")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+V")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.copyAsPng")}
|
|
shortcuts={[getShortcutKey("Shift+Alt+C")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.copyStyles")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Alt+C")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.pasteStyles")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Alt+V")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.delete")}
|
|
shortcuts={[getShortcutKey("Del")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.sendToBack")}
|
|
shortcuts={[
|
|
isDarwin
|
|
? getShortcutKey("CtrlOrCmd+Alt+[")
|
|
: getShortcutKey("CtrlOrCmd+Shift+["),
|
|
]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.bringToFront")}
|
|
shortcuts={[
|
|
isDarwin
|
|
? getShortcutKey("CtrlOrCmd+Alt+]")
|
|
: getShortcutKey("CtrlOrCmd+Shift+]"),
|
|
]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.sendBackward")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+[")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.bringForward")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+]")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.alignTop")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+Up")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.alignBottom")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+Down")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.alignLeft")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+Left")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.alignRight")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+Right")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.duplicateSelection")}
|
|
shortcuts={[
|
|
getShortcutKey("CtrlOrCmd+D"),
|
|
getShortcutKey(`Alt+${t("helpDialog.drag")}`),
|
|
]}
|
|
/>
|
|
<Shortcut
|
|
label={t("buttons.undo")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Z")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("buttons.redo")}
|
|
shortcuts={
|
|
isWindows
|
|
? [
|
|
getShortcutKey("CtrlOrCmd+Y"),
|
|
getShortcutKey("CtrlOrCmd+Shift+Z"),
|
|
]
|
|
: [getShortcutKey("CtrlOrCmd+Shift+Z")]
|
|
}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.group")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+G")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.ungroup")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+G")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.flipHorizontal")}
|
|
shortcuts={[getShortcutKey("Shift+H")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.flipVertical")}
|
|
shortcuts={[getShortcutKey("Shift+V")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.showStroke")}
|
|
shortcuts={[getShortcutKey("S")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.showBackground")}
|
|
shortcuts={[getShortcutKey("G")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.decreaseFontSize")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+<")]}
|
|
/>
|
|
<Shortcut
|
|
label={t("labels.increaseFontSize")}
|
|
shortcuts={[getShortcutKey("CtrlOrCmd+Shift+>")]}
|
|
/>
|
|
</ShortcutIsland>
|
|
</Column>
|
|
</Columns>
|
|
</Section>
|
|
</Dialog>
|
|
</>
|
|
);
|
|
};
|