Actions.tsx 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. import React from "react";
  2. import { AppState, Zoom } from "../types";
  3. import { ExcalidrawElement } from "../element/types";
  4. import { ActionManager } from "../actions/manager";
  5. import {
  6. hasBackground,
  7. hasStroke,
  8. canChangeSharpness,
  9. hasText,
  10. getTargetElement,
  11. } from "../scene";
  12. import { t } from "../i18n";
  13. import { SHAPES } from "../shapes";
  14. import { ToolButton } from "./ToolButton";
  15. import { capitalizeString, isTransparent, setCursorForShape } from "../utils";
  16. import Stack from "./Stack";
  17. import useIsMobile from "../is-mobile";
  18. import { getNonDeletedElements } from "../element";
  19. import { trackEvent, EVENT_SHAPE, EVENT_DIALOG } from "../analytics";
  20. export const SelectedShapeActions = ({
  21. appState,
  22. elements,
  23. renderAction,
  24. elementType,
  25. }: {
  26. appState: AppState;
  27. elements: readonly ExcalidrawElement[];
  28. renderAction: ActionManager["renderAction"];
  29. elementType: ExcalidrawElement["type"];
  30. }) => {
  31. const targetElements = getTargetElement(
  32. getNonDeletedElements(elements),
  33. appState,
  34. );
  35. const isEditing = Boolean(appState.editingElement);
  36. const isMobile = useIsMobile();
  37. const isRTL = document.documentElement.getAttribute("dir") === "rtl";
  38. const showFillIcons =
  39. hasBackground(elementType) ||
  40. targetElements.some(
  41. (element) =>
  42. hasBackground(element.type) && !isTransparent(element.backgroundColor),
  43. );
  44. const showChangeBackgroundIcons =
  45. hasBackground(elementType) ||
  46. targetElements.some((element) => hasBackground(element.type));
  47. return (
  48. <div className="panelColumn">
  49. {renderAction("changeStrokeColor")}
  50. {showChangeBackgroundIcons && renderAction("changeBackgroundColor")}
  51. {showFillIcons && renderAction("changeFillStyle")}
  52. {(hasStroke(elementType) ||
  53. targetElements.some((element) => hasStroke(element.type))) && (
  54. <>
  55. {renderAction("changeStrokeWidth")}
  56. {renderAction("changeStrokeStyle")}
  57. {renderAction("changeSloppiness")}
  58. </>
  59. )}
  60. {(canChangeSharpness(elementType) ||
  61. targetElements.some((element) => canChangeSharpness(element.type))) && (
  62. <>{renderAction("changeSharpness")}</>
  63. )}
  64. {(hasText(elementType) ||
  65. targetElements.some((element) => hasText(element.type))) && (
  66. <>
  67. {renderAction("changeFontSize")}
  68. {renderAction("changeFontFamily")}
  69. {renderAction("changeTextAlign")}
  70. </>
  71. )}
  72. {renderAction("changeOpacity")}
  73. <fieldset>
  74. <legend>{t("labels.layers")}</legend>
  75. <div className="buttonList">
  76. {renderAction("sendToBack")}
  77. {renderAction("sendBackward")}
  78. {renderAction("bringToFront")}
  79. {renderAction("bringForward")}
  80. </div>
  81. </fieldset>
  82. {targetElements.length > 1 && (
  83. <fieldset>
  84. <legend>{t("labels.align")}</legend>
  85. <div className="buttonList">
  86. {
  87. // swap this order for RTL so the button positions always match their action
  88. // (i.e. the leftmost button aligns left)
  89. }
  90. {isRTL ? (
  91. <>
  92. {renderAction("alignRight")}
  93. {renderAction("alignHorizontallyCentered")}
  94. {renderAction("alignLeft")}
  95. </>
  96. ) : (
  97. <>
  98. {renderAction("alignLeft")}
  99. {renderAction("alignHorizontallyCentered")}
  100. {renderAction("alignRight")}
  101. </>
  102. )}
  103. {targetElements.length > 2 &&
  104. renderAction("distributeHorizontally")}
  105. <div className="iconRow">
  106. {renderAction("alignTop")}
  107. {renderAction("alignVerticallyCentered")}
  108. {renderAction("alignBottom")}
  109. {targetElements.length > 2 &&
  110. renderAction("distributeVertically")}
  111. </div>
  112. </div>
  113. </fieldset>
  114. )}
  115. {!isMobile && !isEditing && targetElements.length > 0 && (
  116. <fieldset>
  117. <legend>{t("labels.actions")}</legend>
  118. <div className="buttonList">
  119. {renderAction("duplicateSelection")}
  120. {renderAction("deleteSelectedElements")}
  121. {renderAction("group")}
  122. {renderAction("ungroup")}
  123. </div>
  124. </fieldset>
  125. )}
  126. </div>
  127. );
  128. };
  129. const LIBRARY_ICON = (
  130. // fa-th-large
  131. <svg viewBox="0 0 512 512">
  132. <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" />
  133. </svg>
  134. );
  135. export const ShapesSwitcher = ({
  136. elementType,
  137. setAppState,
  138. isLibraryOpen,
  139. }: {
  140. elementType: ExcalidrawElement["type"];
  141. setAppState: React.Component<any, AppState>["setState"];
  142. isLibraryOpen: boolean;
  143. }) => (
  144. <>
  145. {SHAPES.map(({ value, icon, key }, index) => {
  146. const label = t(`toolBar.${value}`);
  147. const letter = typeof key === "string" ? key : key[0];
  148. const shortcut = `${capitalizeString(letter)} ${t(
  149. "shortcutsDialog.or",
  150. )} ${index + 1}`;
  151. return (
  152. <ToolButton
  153. className="Shape"
  154. key={value}
  155. type="radio"
  156. icon={icon}
  157. checked={elementType === value}
  158. name="editor-current-shape"
  159. title={`${capitalizeString(label)} — ${shortcut}`}
  160. keyBindingLabel={`${index + 1}`}
  161. aria-label={capitalizeString(label)}
  162. aria-keyshortcuts={shortcut}
  163. data-testid={value}
  164. onChange={() => {
  165. trackEvent(EVENT_SHAPE, value, "toolbar");
  166. setAppState({
  167. elementType: value,
  168. multiElement: null,
  169. selectedElementIds: {},
  170. });
  171. setCursorForShape(value);
  172. setAppState({});
  173. }}
  174. />
  175. );
  176. })}
  177. <ToolButton
  178. className="Shape ToolIcon_type_button__library"
  179. type="button"
  180. icon={LIBRARY_ICON}
  181. name="editor-library"
  182. keyBindingLabel="9"
  183. aria-keyshortcuts="9"
  184. title={`${capitalizeString(t("toolBar.library"))} — 9`}
  185. aria-label={capitalizeString(t("toolBar.library"))}
  186. onClick={() => {
  187. if (!isLibraryOpen) {
  188. trackEvent(EVENT_DIALOG, "library");
  189. }
  190. setAppState({ isLibraryOpen: !isLibraryOpen });
  191. }}
  192. />
  193. </>
  194. );
  195. export const ZoomActions = ({
  196. renderAction,
  197. zoom,
  198. }: {
  199. renderAction: ActionManager["renderAction"];
  200. zoom: Zoom;
  201. }) => (
  202. <Stack.Col gap={1}>
  203. <Stack.Row gap={1} align="center">
  204. {renderAction("zoomIn")}
  205. {renderAction("zoomOut")}
  206. {renderAction("resetZoom")}
  207. <div style={{ marginInlineStart: 4 }}>
  208. {(zoom.value * 100).toFixed(0)}%
  209. </div>
  210. </Stack.Row>
  211. </Stack.Col>
  212. );