import React from "react";
import { Action } from "./types";
import { ExcalidrawElement, ExcalidrawTextElement } from "../element/types";
import { getSelectedAttribute } from "../scene";
import { ButtonSelect } from "../components/ButtonSelect";
import { isTextElement, redrawTextBoundingBox } from "../element";
import { ColorPicker } from "../components/ColorPicker";
const changeProperty = (
elements: readonly ExcalidrawElement[],
callback: (element: ExcalidrawElement) => ExcalidrawElement
) => {
return elements.map(element => {
if (element.isSelected) {
return callback(element);
}
return element;
});
};
export const actionChangeStrokeColor: Action = {
name: "changeStrokeColor",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
strokeColor: value
})),
appState: { ...appState, currentItemStrokeColor: value }
};
},
PanelComponent: ({ elements, appState, updateData, t }) => {
return (
<>
{t("labels.stroke")}
element.strokeColor) ||
appState.currentItemStrokeColor
}
onChange={updateData}
/>
>
);
}
};
export const actionChangeBackgroundColor: Action = {
name: "changeBackgroundColor",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
backgroundColor: value
})),
appState: { ...appState, currentItemBackgroundColor: value }
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
{t("labels.background")}
element.backgroundColor) ||
appState.currentItemBackgroundColor
}
onChange={updateData}
/>
>
)
};
export const actionChangeFillStyle: Action = {
name: "changeFillStyle",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
fillStyle: value
}))
};
},
PanelComponent: ({ elements, updateData, t }) => (
<>
{t("labels.fill")}
element.fillStyle)}
onChange={value => {
updateData(value);
}}
/>
>
)
};
export const actionChangeStrokeWidth: Action = {
name: "changeStrokeWidth",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
strokeWidth: value
}))
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
{t("labels.strokeWidth")}
element.strokeWidth)}
onChange={value => updateData(value)}
/>
>
)
};
export const actionChangeSloppiness: Action = {
name: "changeSloppiness",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
roughness: value
}))
};
},
PanelComponent: ({ elements, appState, updateData, t }) => (
<>
{t("labels.sloppiness")}
element.roughness)}
onChange={value => updateData(value)}
/>
>
)
};
export const actionChangeOpacity: Action = {
name: "changeOpacity",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => ({
...el,
shape: null,
opacity: value
}))
};
},
PanelComponent: ({ elements, updateData, t }) => (
<>
{t("labels.oppacity")}
updateData(+e.target.value)}
value={
getSelectedAttribute(elements, element => element.opacity) ||
0 /* Put the opacity at 0 if there are two conflicting ones */
}
/>
>
)
};
export const actionChangeFontSize: Action = {
name: "changeFontSize",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => {
if (isTextElement(el)) {
const element: ExcalidrawTextElement = {
...el,
shape: null,
font: `${value}px ${el.font.split("px ")[1]}`
};
redrawTextBoundingBox(element);
return element;
}
return el;
})
};
},
PanelComponent: ({ elements, updateData, t }) => (
<>
{t("labels.fontSize")}
isTextElement(element) && +element.font.split("px ")[0]
)}
onChange={value => updateData(value)}
/>
>
)
};
export const actionChangeFontFamily: Action = {
name: "changeFontFamily",
perform: (elements, appState, value) => {
return {
elements: changeProperty(elements, el => {
if (isTextElement(el)) {
const element: ExcalidrawTextElement = {
...el,
shape: null,
font: `${el.font.split("px ")[0]}px ${value}`
};
redrawTextBoundingBox(element);
return element;
}
return el;
})
};
},
PanelComponent: ({ elements, updateData, t }) => (
<>
{t("labels.fontFamily")}
isTextElement(element) && element.font.split("px ")[1]
)}
onChange={value => updateData(value)}
/>
>
)
};