import React from "react"; import { AppState } 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, setCursorForShape } from "../utils"; import Stack from "./Stack"; import useIsMobile from "../is-mobile"; import { getNonDeletedElements } from "../element"; 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(); return (
{renderAction("changeStrokeColor")} {(hasBackground(elementType) || targetElements.some((element) => hasBackground(element.type))) && ( <> {renderAction("changeBackgroundColor")} {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")}
{t("labels.layers")}
{renderAction("sendToBack")} {renderAction("sendBackward")} {renderAction("bringToFront")} {renderAction("bringForward")}
{!isMobile && !isEditing && targetElements.length > 0 && (
{t("labels.actions")}
{renderAction("duplicateSelection")} {renderAction("deleteSelectedElements")} {renderAction("group")} {renderAction("ungroup")}
)}
); }; const LIBRARY_ICON = ( // fa-th-large ); export const ShapesSwitcher = ({ elementType, setAppState, isLibraryOpen, }: { elementType: ExcalidrawElement["type"]; setAppState: React.Component["setState"]; isLibraryOpen: boolean; }) => ( <> {SHAPES.map(({ value, icon, key }, index) => { const label = t(`toolBar.${value}`); const letter = typeof key === "string" ? key : key[0]; const letterShortcut = /[a-z]/.test(letter) ? letter : `Shift+${letter}`; const shortcut = `${capitalizeString(letterShortcut)} ${t( "shortcutsDialog.or", )} ${index + 1}`; return ( { setAppState({ elementType: value, multiElement: null, selectedElementIds: {}, }); setCursorForShape(value); setAppState({}); }} /> ); })} { setAppState({ isLibraryOpen: !isLibraryOpen }); }} /> ); export const ZoomActions = ({ renderAction, zoom, }: { renderAction: ActionManager["renderAction"]; zoom: number; }) => ( {renderAction("zoomIn")} {renderAction("zoomOut")} {renderAction("resetZoom")}
{(zoom * 100).toFixed(0)}%
);