import React from "react"; import { ExcalidrawElement, ExcalidrawTextElement, TextAlign, FontFamily, } from "../element/types"; import { getCommonAttributeOfSelectedElements, isSomeElementSelected, getTargetElement, canChangeSharpness, } from "../scene"; import { ButtonSelect } from "../components/ButtonSelect"; import { isTextElement, redrawTextBoundingBox, getNonDeletedElements, } from "../element"; import { isLinearElement, isLinearElementType } from "../element/typeChecks"; import { ColorPicker } from "../components/ColorPicker"; import { AppState } from "../../src/types"; import { t } from "../i18n"; import { register } from "./register"; import { newElementWith } from "../element/mutateElement"; import { DEFAULT_FONT_SIZE, DEFAULT_FONT_FAMILY } from "../constants"; import { randomInteger } from "../random"; const changeProperty = ( elements: readonly ExcalidrawElement[], appState: AppState, callback: (element: ExcalidrawElement) => ExcalidrawElement, ) => { return elements.map((element) => { if ( appState.selectedElementIds[element.id] || element.id === appState.editingElement?.id ) { return callback(element); } return element; }); }; const getFormValue = function ( elements: readonly ExcalidrawElement[], appState: AppState, getAttribute: (element: ExcalidrawElement) => T, defaultValue?: T, ): T | null { const editingElement = appState.editingElement; const nonDeletedElements = getNonDeletedElements(elements); return ( (editingElement && getAttribute(editingElement)) ?? (isSomeElementSelected(nonDeletedElements, appState) ? getCommonAttributeOfSelectedElements( nonDeletedElements, appState, getAttribute, ) : defaultValue) ?? null ); }; export const actionChangeStrokeColor = register({ name: "changeStrokeColor", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => newElementWith(el, { strokeColor: value, }), ), appState: { ...appState, currentItemStrokeColor: value }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => ( <> element.strokeColor, appState.currentItemStrokeColor, )} onChange={updateData} /> ), }); export const actionChangeBackgroundColor = register({ name: "changeBackgroundColor", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => newElementWith(el, { backgroundColor: value, }), ), appState: { ...appState, currentItemBackgroundColor: value }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => ( <> element.backgroundColor, appState.currentItemBackgroundColor, )} onChange={updateData} /> ), }); export const actionChangeFillStyle = register({ name: "changeFillStyle", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => newElementWith(el, { fillStyle: value, }), ), appState: { ...appState, currentItemFillStyle: value }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => (
{t("labels.fill")} element.fillStyle, appState.currentItemFillStyle, )} onChange={(value) => { updateData(value); }} />
), }); export const actionChangeStrokeWidth = register({ name: "changeStrokeWidth", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => newElementWith(el, { strokeWidth: value, }), ), appState: { ...appState, currentItemStrokeWidth: value }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => (
{t("labels.strokeWidth")} element.strokeWidth, appState.currentItemStrokeWidth, )} onChange={(value) => updateData(value)} />
), }); export const actionChangeSloppiness = register({ name: "changeSloppiness", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => newElementWith(el, { seed: randomInteger(), roughness: value, }), ), appState: { ...appState, currentItemRoughness: value }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => (
{t("labels.sloppiness")} element.roughness, appState.currentItemRoughness, )} onChange={(value) => updateData(value)} />
), }); export const actionChangeStrokeStyle = register({ name: "changeStrokeStyle", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => newElementWith(el, { strokeStyle: value, }), ), appState: { ...appState, currentItemStrokeStyle: value }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => (
{t("labels.strokeStyle")} element.strokeStyle, appState.currentItemStrokeStyle, )} onChange={(value) => updateData(value)} />
), }); export const actionChangeOpacity = register({ name: "changeOpacity", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => newElementWith(el, { opacity: value, }), ), appState: { ...appState, currentItemOpacity: value }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => ( ), }); export const actionChangeFontSize = register({ name: "changeFontSize", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => { if (isTextElement(el)) { const element: ExcalidrawTextElement = newElementWith(el, { fontSize: value, }); redrawTextBoundingBox(element); return element; } return el; }), appState: { ...appState, currentItemFontSize: value, }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => (
{t("labels.fontSize")} isTextElement(element) && element.fontSize, appState.currentItemFontSize || DEFAULT_FONT_SIZE, )} onChange={(value) => updateData(value)} />
), }); export const actionChangeFontFamily = register({ name: "changeFontFamily", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => { if (isTextElement(el)) { const element: ExcalidrawTextElement = newElementWith(el, { fontFamily: value, }); redrawTextBoundingBox(element); return element; } return el; }), appState: { ...appState, currentItemFontFamily: value, }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => { const options: { value: FontFamily; text: string }[] = [ { value: 1, text: t("labels.handDrawn") }, { value: 2, text: t("labels.normal") }, { value: 3, text: t("labels.code") }, ]; return (
{t("labels.fontFamily")} group="font-family" options={options} value={getFormValue( elements, appState, (element) => isTextElement(element) && element.fontFamily, appState.currentItemFontFamily || DEFAULT_FONT_FAMILY, )} onChange={(value) => updateData(value)} />
); }, }); export const actionChangeTextAlign = register({ name: "changeTextAlign", perform: (elements, appState, value) => { return { elements: changeProperty(elements, appState, (el) => { if (isTextElement(el)) { const element: ExcalidrawTextElement = newElementWith(el, { textAlign: value, }); redrawTextBoundingBox(element); return element; } return el; }), appState: { ...appState, currentItemTextAlign: value, }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => (
{t("labels.textAlign")} group="text-align" options={[ { value: "left", text: t("labels.left") }, { value: "center", text: t("labels.center") }, { value: "right", text: t("labels.right") }, ]} value={getFormValue( elements, appState, (element) => isTextElement(element) && element.textAlign, appState.currentItemTextAlign, )} onChange={(value) => updateData(value)} />
), }); export const actionChangeSharpness = register({ name: "changeSharpness", perform: (elements, appState, value) => { const targetElements = getTargetElement( getNonDeletedElements(elements), appState, ); const shouldUpdateForNonLinearElements = targetElements.length ? targetElements.every((e) => !isLinearElement(e)) : !isLinearElementType(appState.elementType); const shouldUpdateForLinearElements = targetElements.length ? targetElements.every(isLinearElement) : isLinearElementType(appState.elementType); return { elements: changeProperty(elements, appState, (el) => newElementWith(el, { strokeSharpness: value, }), ), appState: { ...appState, currentItemStrokeSharpness: shouldUpdateForNonLinearElements ? value : appState.currentItemStrokeSharpness, currentItemLinearStrokeSharpness: shouldUpdateForLinearElements ? value : appState.currentItemLinearStrokeSharpness, }, commitToHistory: true, }; }, PanelComponent: ({ elements, appState, updateData }) => (
{t("labels.edges")} element.strokeSharpness, (canChangeSharpness(appState.elementType) && (isLinearElementType(appState.elementType) ? appState.currentItemLinearStrokeSharpness : appState.currentItemStrokeSharpness)) || null, )} onChange={(value) => updateData(value)} />
), });