123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335 |
- import React from "react";
- import { ActionManager } from "../actions/manager";
- import { getNonDeletedElements } from "../element";
- import { ExcalidrawElement, PointerType } from "../element/types";
- import { t } from "../i18n";
- import { useDevice } from "../components/App";
- import {
- canChangeSharpness,
- canHaveArrowheads,
- getTargetElements,
- hasBackground,
- hasStrokeStyle,
- hasStrokeWidth,
- hasText,
- } from "../scene";
- import { SHAPES } from "../shapes";
- import { AppState, Zoom } from "../types";
- import {
- capitalizeString,
- isTransparent,
- updateActiveTool,
- setCursorForShape,
- } from "../utils";
- import Stack from "./Stack";
- import { ToolButton } from "./ToolButton";
- import { hasStrokeColor } from "../scene/comparisons";
- import { trackEvent } from "../analytics";
- import { hasBoundTextElement, isBoundToContainer } from "../element/typeChecks";
- import clsx from "clsx";
- import { actionToggleZenMode } from "../actions";
- import "./Actions.scss";
- import { Tooltip } from "./Tooltip";
- export const SelectedShapeActions = ({
- appState,
- elements,
- renderAction,
- }: {
- appState: AppState;
- elements: readonly ExcalidrawElement[];
- renderAction: ActionManager["renderAction"];
- }) => {
- const targetElements = getTargetElements(
- getNonDeletedElements(elements),
- appState,
- );
- let isSingleElementBoundContainer = false;
- if (
- targetElements.length === 2 &&
- (hasBoundTextElement(targetElements[0]) ||
- hasBoundTextElement(targetElements[1]))
- ) {
- isSingleElementBoundContainer = true;
- }
- const isEditing = Boolean(appState.editingElement);
- const device = useDevice();
- const isRTL = document.documentElement.getAttribute("dir") === "rtl";
- const showFillIcons =
- hasBackground(appState.activeTool.type) ||
- targetElements.some(
- (element) =>
- hasBackground(element.type) && !isTransparent(element.backgroundColor),
- );
- const showChangeBackgroundIcons =
- hasBackground(appState.activeTool.type) ||
- targetElements.some((element) => hasBackground(element.type));
- const showLinkIcon =
- targetElements.length === 1 || isSingleElementBoundContainer;
- let commonSelectedType: string | null = targetElements[0]?.type || null;
- for (const element of targetElements) {
- if (element.type !== commonSelectedType) {
- commonSelectedType = null;
- break;
- }
- }
- return (
- <div className="panelColumn">
- <div>
- {((hasStrokeColor(appState.activeTool.type) &&
- appState.activeTool.type !== "image" &&
- commonSelectedType !== "image") ||
- targetElements.some((element) => hasStrokeColor(element.type))) &&
- renderAction("changeStrokeColor")}
- </div>
- {showChangeBackgroundIcons && (
- <div>{renderAction("changeBackgroundColor")}</div>
- )}
- {showFillIcons && renderAction("changeFillStyle")}
- {(hasStrokeWidth(appState.activeTool.type) ||
- targetElements.some((element) => hasStrokeWidth(element.type))) &&
- renderAction("changeStrokeWidth")}
- {(appState.activeTool.type === "freedraw" ||
- targetElements.some((element) => element.type === "freedraw")) &&
- renderAction("changeStrokeShape")}
- {(hasStrokeStyle(appState.activeTool.type) ||
- targetElements.some((element) => hasStrokeStyle(element.type))) && (
- <>
- {renderAction("changeStrokeStyle")}
- {renderAction("changeSloppiness")}
- </>
- )}
- {(canChangeSharpness(appState.activeTool.type) ||
- targetElements.some((element) => canChangeSharpness(element.type))) && (
- <>{renderAction("changeSharpness")}</>
- )}
- {(hasText(appState.activeTool.type) ||
- targetElements.some((element) => hasText(element.type))) && (
- <>
- {renderAction("changeFontSize")}
- {renderAction("changeFontFamily")}
- {renderAction("changeTextAlign")}
- </>
- )}
- {targetElements.some(
- (element) =>
- hasBoundTextElement(element) || isBoundToContainer(element),
- ) && renderAction("changeVerticalAlign")}
- {(canHaveArrowheads(appState.activeTool.type) ||
- targetElements.some((element) => canHaveArrowheads(element.type))) && (
- <>{renderAction("changeArrowhead")}</>
- )}
- {renderAction("changeOpacity")}
- <fieldset>
- <legend>{t("labels.layers")}</legend>
- <div className="buttonList">
- {renderAction("sendToBack")}
- {renderAction("sendBackward")}
- {renderAction("bringToFront")}
- {renderAction("bringForward")}
- </div>
- </fieldset>
- {targetElements.length > 1 && !isSingleElementBoundContainer && (
- <fieldset>
- <legend>{t("labels.align")}</legend>
- <div className="buttonList">
- {
- // swap this order for RTL so the button positions always match their action
- // (i.e. the leftmost button aligns left)
- }
- {isRTL ? (
- <>
- {renderAction("alignRight")}
- {renderAction("alignHorizontallyCentered")}
- {renderAction("alignLeft")}
- </>
- ) : (
- <>
- {renderAction("alignLeft")}
- {renderAction("alignHorizontallyCentered")}
- {renderAction("alignRight")}
- </>
- )}
- {targetElements.length > 2 &&
- renderAction("distributeHorizontally")}
- {/* breaks the row ˇˇ */}
- <div style={{ flexBasis: "100%", height: 0 }} />
- <div
- style={{
- display: "flex",
- flexWrap: "wrap",
- gap: ".5rem",
- marginTop: "-0.5rem",
- }}
- >
- {renderAction("alignTop")}
- {renderAction("alignVerticallyCentered")}
- {renderAction("alignBottom")}
- {targetElements.length > 2 &&
- renderAction("distributeVertically")}
- </div>
- </div>
- </fieldset>
- )}
- {!isEditing && targetElements.length > 0 && (
- <fieldset>
- <legend>{t("labels.actions")}</legend>
- <div className="buttonList">
- {!device.isMobile && renderAction("duplicateSelection")}
- {!device.isMobile && renderAction("deleteSelectedElements")}
- {renderAction("group")}
- {renderAction("ungroup")}
- {showLinkIcon && renderAction("hyperlink")}
- </div>
- </fieldset>
- )}
- </div>
- );
- };
- export const ShapesSwitcher = ({
- canvas,
- activeTool,
- setAppState,
- onImageAction,
- appState,
- }: {
- canvas: HTMLCanvasElement | null;
- activeTool: AppState["activeTool"];
- setAppState: React.Component<any, AppState>["setState"];
- onImageAction: (data: { pointerType: PointerType | null }) => void;
- appState: AppState;
- }) => (
- <>
- {SHAPES.map(({ value, icon, key, fillable }, index) => {
- const numberKey = value === "eraser" ? 0 : index + 1;
- const label = t(`toolBar.${value}`);
- const letter = key && (typeof key === "string" ? key : key[0]);
- const shortcut = letter
- ? `${capitalizeString(letter)} ${t("helpDialog.or")} ${numberKey}`
- : `${numberKey}`;
- return (
- <ToolButton
- className={clsx("Shape", { fillable })}
- key={value}
- type="radio"
- icon={icon}
- checked={activeTool.type === value}
- name="editor-current-shape"
- title={`${capitalizeString(label)} — ${shortcut}`}
- keyBindingLabel={`${numberKey}`}
- aria-label={capitalizeString(label)}
- aria-keyshortcuts={shortcut}
- data-testid={value}
- onPointerDown={({ pointerType }) => {
- if (!appState.penDetected && pointerType === "pen") {
- setAppState({
- penDetected: true,
- penMode: true,
- });
- }
- }}
- onChange={({ pointerType }) => {
- if (appState.activeTool.type !== value) {
- trackEvent("toolbar", value, "ui");
- }
- const nextActiveTool = updateActiveTool(appState, {
- type: value,
- });
- setAppState({
- activeTool: nextActiveTool,
- multiElement: null,
- selectedElementIds: {},
- });
- setCursorForShape(canvas, {
- ...appState,
- activeTool: nextActiveTool,
- });
- if (value === "image") {
- onImageAction({ pointerType });
- }
- }}
- />
- );
- })}
- </>
- );
- export const ZoomActions = ({
- renderAction,
- zoom,
- }: {
- renderAction: ActionManager["renderAction"];
- zoom: Zoom;
- }) => (
- <Stack.Col gap={1} className="zoom-actions">
- <Stack.Row align="center">
- {renderAction("zoomOut")}
- {renderAction("resetZoom")}
- {renderAction("zoomIn")}
- </Stack.Row>
- </Stack.Col>
- );
- export const UndoRedoActions = ({
- renderAction,
- className,
- }: {
- renderAction: ActionManager["renderAction"];
- className?: string;
- }) => (
- <div className={`undo-redo-buttons ${className}`}>
- <div className="undo-button-container">
- <Tooltip label={t("buttons.undo")}>{renderAction("undo")}</Tooltip>
- </div>
- <div className="redo-button-container">
- <Tooltip label={t("buttons.redo")}> {renderAction("redo")}</Tooltip>
- </div>
- </div>
- );
- export const ExitZenModeAction = ({
- actionManager,
- showExitZenModeBtn,
- }: {
- actionManager: ActionManager;
- showExitZenModeBtn: boolean;
- }) => (
- <button
- className={clsx("disable-zen-mode", {
- "disable-zen-mode--visible": showExitZenModeBtn,
- })}
- onClick={() => actionManager.executeAction(actionToggleZenMode)}
- >
- {t("buttons.exitZenMode")}
- </button>
- );
- export const FinalizeAction = ({
- renderAction,
- className,
- }: {
- renderAction: ActionManager["renderAction"];
- className?: string;
- }) => (
- <div className={`finalize-button ${className}`}>
- {renderAction("finalize", { size: "small" })}
- </div>
- );
|