excalidraw/src/actions/actionProperties.tsx

299 lines
7.7 KiB
TypeScript
Raw Normal View History

import React from "react";
import { Action } from "./types";
import { ExcalidrawElement, ExcalidrawTextElement } from "../element/types";
import { getCommonAttributeOfSelectedElements } from "../scene";
import { ButtonSelect } from "../components/ButtonSelect";
import { isTextElement, redrawTextBoundingBox } from "../element";
import { ColorPicker } from "../components/ColorPicker";
import { AppState } from "../../src/types";
const changeProperty = (
elements: readonly ExcalidrawElement[],
2020-01-24 12:04:54 +02:00
callback: (element: ExcalidrawElement) => ExcalidrawElement,
) => {
return elements.map(element => {
if (element.isSelected) {
return callback(element);
}
return element;
});
};
const getFormValue = function<T>(
editingElement: AppState["editingElement"],
elements: readonly ExcalidrawElement[],
getAttribute: (element: ExcalidrawElement) => T,
2020-01-24 12:04:54 +02:00
defaultValue?: T,
): T | null {
return (
(editingElement && getAttribute(editingElement)) ??
getCommonAttributeOfSelectedElements(elements, getAttribute) ??
defaultValue ??
null
);
};
export const actionChangeStrokeColor: Action = {
name: "changeStrokeColor",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
2020-01-24 12:04:54 +02:00
strokeColor: value,
})),
2020-01-24 12:04:54 +02:00
appState: { ...appState, currentItemStrokeColor: value },
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
<h5>{t("labels.stroke")}</h5>
<ColorPicker
type="elementStroke"
color={getFormValue(
appState.editingElement,
elements,
element => element.strokeColor,
2020-01-24 12:04:54 +02:00
appState.currentItemStrokeColor,
)}
onChange={updateData}
/>
</>
2020-01-24 12:04:54 +02:00
),
};
export const actionChangeBackgroundColor: Action = {
name: "changeBackgroundColor",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
2020-01-24 12:04:54 +02:00
backgroundColor: value,
})),
2020-01-24 12:04:54 +02:00
appState: { ...appState, currentItemBackgroundColor: value },
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
<h5>{t("labels.background")}</h5>
<ColorPicker
type="elementBackground"
color={getFormValue(
appState.editingElement,
elements,
element => element.backgroundColor,
2020-01-24 12:04:54 +02:00
appState.currentItemBackgroundColor,
)}
onChange={updateData}
/>
</>
2020-01-24 12:04:54 +02:00
),
};
export const actionChangeFillStyle: Action = {
name: "changeFillStyle",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
2020-01-24 12:04:54 +02:00
fillStyle: value,
})),
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
<h5>{t("labels.fill")}</h5>
<ButtonSelect
options={[
{ value: "solid", text: t("labels.solid") },
{ value: "hachure", text: t("labels.hachure") },
2020-01-24 12:04:54 +02:00
{ value: "cross-hatch", text: t("labels.crossHatch") },
]}
value={getFormValue(
appState.editingElement,
elements,
2020-01-24 12:04:54 +02:00
element => element.fillStyle,
)}
onChange={value => {
updateData(value);
}}
/>
</>
2020-01-24 12:04:54 +02:00
),
};
export const actionChangeStrokeWidth: Action = {
name: "changeStrokeWidth",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
2020-01-24 12:04:54 +02:00
strokeWidth: value,
})),
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
<h5>{t("labels.strokeWidth")}</h5>
<ButtonSelect
options={[
{ value: 1, text: t("labels.thin") },
{ value: 2, text: t("labels.bold") },
2020-01-24 12:04:54 +02:00
{ value: 4, text: t("labels.extraBold") },
]}
value={getFormValue(
appState.editingElement,
elements,
2020-01-24 12:04:54 +02:00
element => element.strokeWidth,
)}
onChange={value => updateData(value)}
/>
</>
2020-01-24 12:04:54 +02:00
),
};
export const actionChangeSloppiness: Action = {
name: "changeSloppiness",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
2020-01-24 12:04:54 +02:00
roughness: value,
})),
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
<h5>{t("labels.sloppiness")}</h5>
<ButtonSelect
options={[
{ value: 0, text: t("labels.architect") },
{ value: 1, text: t("labels.artist") },
2020-01-24 12:04:54 +02:00
{ value: 3, text: t("labels.cartoonist") },
]}
value={getFormValue(
appState.editingElement,
elements,
2020-01-24 12:04:54 +02:00
element => element.roughness,
)}
onChange={value => updateData(value)}
/>
</>
2020-01-24 12:04:54 +02:00
),
};
export const actionChangeOpacity: Action = {
name: "changeOpacity",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
2020-01-24 12:04:54 +02:00
opacity: value,
})),
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
2020-01-21 17:39:39 +01:00
<h5>{t("labels.opacity")}</h5>
<input
type="range"
min="0"
max="100"
onChange={e => updateData(+e.target.value)}
value={
getFormValue(
appState.editingElement,
elements,
element => element.opacity,
2020-01-24 12:04:54 +02:00
100 /* default opacity */,
) ?? undefined
}
/>
</>
2020-01-24 12:04:54 +02:00
),
};
export const actionChangeFontSize: Action = {
name: "changeFontSize",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => {
if (isTextElement(el)) {
const element: ExcalidrawTextElement = {
...el,
shape: null,
2020-01-24 12:04:54 +02:00
font: `${value}px ${el.font.split("px ")[1]}`,
};
redrawTextBoundingBox(element);
return element;
}
return el;
2020-01-24 12:04:54 +02:00
}),
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
<h5>{t("labels.fontSize")}</h5>
<ButtonSelect
options={[
{ value: 16, text: t("labels.small") },
{ value: 20, text: t("labels.medium") },
{ value: 28, text: t("labels.large") },
2020-01-24 12:04:54 +02:00
{ value: 36, text: t("labels.veryLarge") },
]}
value={getFormValue(
appState.editingElement,
elements,
2020-01-24 12:04:54 +02:00
element => isTextElement(element) && +element.font.split("px ")[0],
)}
onChange={value => updateData(value)}
/>
</>
2020-01-24 12:04:54 +02:00
),
};
export const actionChangeFontFamily: Action = {
name: "changeFontFamily",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => {
if (isTextElement(el)) {
const element: ExcalidrawTextElement = {
...el,
shape: null,
2020-01-24 12:04:54 +02:00
font: `${el.font.split("px ")[0]}px ${value}`,
};
redrawTextBoundingBox(element);
return element;
}
return el;
2020-01-24 12:04:54 +02:00
}),
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
<h5>{t("labels.fontFamily")}</h5>
<ButtonSelect
options={[
{ value: "Virgil", text: t("labels.handDrawn") },
{ value: "Helvetica", text: t("labels.normal") },
2020-01-24 12:04:54 +02:00
{ value: "Cascadia", text: t("labels.code") },
]}
value={getFormValue(
appState.editingElement,
elements,
2020-01-24 12:04:54 +02:00
element => isTextElement(element) && element.font.split("px ")[1],
)}
onChange={value => updateData(value)}
/>
</>
2020-01-24 12:04:54 +02:00
),
};