123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225 |
- import React from "react";
- import { AppState, Zoom } from "../types";
- import { ExcalidrawElement } from "../element/types";
- import { ActionManager } from "../actions/manager";
- import {
- hasBackground,
- hasStroke,
- canChangeSharpness,
- hasText,
- getTargetElement,
- } from "../scene";
- import { t } from "../i18n";
- import { SHAPES } from "../shapes";
- import { ToolButton } from "./ToolButton";
- import { capitalizeString, isTransparent, setCursorForShape } from "../utils";
- import Stack from "./Stack";
- import useIsMobile from "../is-mobile";
- import { getNonDeletedElements } from "../element";
- import { trackEvent, EVENT_SHAPE, EVENT_DIALOG } from "../analytics";
- export const SelectedShapeActions = ({
- appState,
- elements,
- renderAction,
- elementType,
- }: {
- appState: AppState;
- elements: readonly ExcalidrawElement[];
- renderAction: ActionManager["renderAction"];
- elementType: ExcalidrawElement["type"];
- }) => {
- const targetElements = getTargetElement(
- getNonDeletedElements(elements),
- appState,
- );
- const isEditing = Boolean(appState.editingElement);
- const isMobile = useIsMobile();
- const isRTL = document.documentElement.getAttribute("dir") === "rtl";
- const showFillIcons =
- hasBackground(elementType) ||
- targetElements.some(
- (element) =>
- hasBackground(element.type) && !isTransparent(element.backgroundColor),
- );
- const showChangeBackgroundIcons =
- hasBackground(elementType) ||
- targetElements.some((element) => hasBackground(element.type));
- return (
- <div className="panelColumn">
- {renderAction("changeStrokeColor")}
- {showChangeBackgroundIcons && renderAction("changeBackgroundColor")}
- {showFillIcons && renderAction("changeFillStyle")}
- {(hasStroke(elementType) ||
- targetElements.some((element) => hasStroke(element.type))) && (
- <>
- {renderAction("changeStrokeWidth")}
- {renderAction("changeStrokeStyle")}
- {renderAction("changeSloppiness")}
- </>
- )}
- {(canChangeSharpness(elementType) ||
- targetElements.some((element) => canChangeSharpness(element.type))) && (
- <>{renderAction("changeSharpness")}</>
- )}
- {(hasText(elementType) ||
- targetElements.some((element) => hasText(element.type))) && (
- <>
- {renderAction("changeFontSize")}
- {renderAction("changeFontFamily")}
- {renderAction("changeTextAlign")}
- </>
- )}
- {renderAction("changeOpacity")}
- <fieldset>
- <legend>{t("labels.layers")}</legend>
- <div className="buttonList">
- {renderAction("sendToBack")}
- {renderAction("sendBackward")}
- {renderAction("bringToFront")}
- {renderAction("bringForward")}
- </div>
- </fieldset>
- {targetElements.length > 1 && (
- <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")}
- <div className="iconRow">
- {renderAction("alignTop")}
- {renderAction("alignVerticallyCentered")}
- {renderAction("alignBottom")}
- {targetElements.length > 2 &&
- renderAction("distributeVertically")}
- </div>
- </div>
- </fieldset>
- )}
- {!isMobile && !isEditing && targetElements.length > 0 && (
- <fieldset>
- <legend>{t("labels.actions")}</legend>
- <div className="buttonList">
- {renderAction("duplicateSelection")}
- {renderAction("deleteSelectedElements")}
- {renderAction("group")}
- {renderAction("ungroup")}
- </div>
- </fieldset>
- )}
- </div>
- );
- };
- const LIBRARY_ICON = (
- // fa-th-large
- <svg viewBox="0 0 512 512">
- <path d="M296 32h192c13.255 0 24 10.745 24 24v160c0 13.255-10.745 24-24 24H296c-13.255 0-24-10.745-24-24V56c0-13.255 10.745-24 24-24zm-80 0H24C10.745 32 0 42.745 0 56v160c0 13.255 10.745 24 24 24h192c13.255 0 24-10.745 24-24V56c0-13.255-10.745-24-24-24zM0 296v160c0 13.255 10.745 24 24 24h192c13.255 0 24-10.745 24-24V296c0-13.255-10.745-24-24-24H24c-13.255 0-24 10.745-24 24zm296 184h192c13.255 0 24-10.745 24-24V296c0-13.255-10.745-24-24-24H296c-13.255 0-24 10.745-24 24v160c0 13.255 10.745 24 24 24z" />
- </svg>
- );
- export const ShapesSwitcher = ({
- elementType,
- setAppState,
- isLibraryOpen,
- }: {
- elementType: ExcalidrawElement["type"];
- setAppState: React.Component<any, AppState>["setState"];
- isLibraryOpen: boolean;
- }) => (
- <>
- {SHAPES.map(({ value, icon, key }, index) => {
- const label = t(`toolBar.${value}`);
- const letter = typeof key === "string" ? key : key[0];
- const shortcut = `${capitalizeString(letter)} ${t(
- "shortcutsDialog.or",
- )} ${index + 1}`;
- return (
- <ToolButton
- className="Shape"
- key={value}
- type="radio"
- icon={icon}
- checked={elementType === value}
- name="editor-current-shape"
- title={`${capitalizeString(label)} — ${shortcut}`}
- keyBindingLabel={`${index + 1}`}
- aria-label={capitalizeString(label)}
- aria-keyshortcuts={shortcut}
- data-testid={value}
- onChange={() => {
- trackEvent(EVENT_SHAPE, value, "toolbar");
- setAppState({
- elementType: value,
- multiElement: null,
- selectedElementIds: {},
- });
- setCursorForShape(value);
- setAppState({});
- }}
- />
- );
- })}
- <ToolButton
- className="Shape ToolIcon_type_button__library"
- type="button"
- icon={LIBRARY_ICON}
- name="editor-library"
- keyBindingLabel="9"
- aria-keyshortcuts="9"
- title={`${capitalizeString(t("toolBar.library"))} — 9`}
- aria-label={capitalizeString(t("toolBar.library"))}
- onClick={() => {
- if (!isLibraryOpen) {
- trackEvent(EVENT_DIALOG, "library");
- }
- setAppState({ isLibraryOpen: !isLibraryOpen });
- }}
- />
- </>
- );
- export const ZoomActions = ({
- renderAction,
- zoom,
- }: {
- renderAction: ActionManager["renderAction"];
- zoom: Zoom;
- }) => (
- <Stack.Col gap={1}>
- <Stack.Row gap={1} align="center">
- {renderAction("zoomIn")}
- {renderAction("zoomOut")}
- {renderAction("resetZoom")}
- <div style={{ marginInlineStart: 4 }}>
- {(zoom.value * 100).toFixed(0)}%
- </div>
- </Stack.Row>
- </Stack.Col>
- );
|