mirror of
https://github.com/excalidraw/excalidraw.git
synced 2025-05-03 10:00:07 -04:00
feat: Support custom elements in @excalidraw/excalidraw
This commit is contained in:
parent
2209e2c1e8
commit
39d0084a5e
13 changed files with 202 additions and 13 deletions
|
@ -119,7 +119,11 @@ import {
|
|||
} from "../element/binding";
|
||||
import { LinearElementEditor } from "../element/linearElementEditor";
|
||||
import { mutateElement, newElementWith } from "../element/mutateElement";
|
||||
import { deepCopyElement, newFreeDrawElement } from "../element/newElement";
|
||||
import {
|
||||
deepCopyElement,
|
||||
newCustomElement,
|
||||
newFreeDrawElement,
|
||||
} from "../element/newElement";
|
||||
import {
|
||||
hasBoundTextElement,
|
||||
isBindingElement,
|
||||
|
@ -327,6 +331,7 @@ class App extends React.Component<AppProps, AppState> {
|
|||
lastPointerUp: React.PointerEvent<HTMLElement> | PointerEvent | null = null;
|
||||
contextMenuOpen: boolean = false;
|
||||
lastScenePointer: { x: number; y: number } | null = null;
|
||||
customElementName: string | null = null;
|
||||
|
||||
constructor(props: AppProps) {
|
||||
super(props);
|
||||
|
@ -378,6 +383,7 @@ class App extends React.Component<AppProps, AppState> {
|
|||
importLibrary: this.importLibraryFromUrl,
|
||||
setToastMessage: this.setToastMessage,
|
||||
id: this.id,
|
||||
setCustomType: this.setCustomType,
|
||||
} as const;
|
||||
if (typeof excalidrawRef === "function") {
|
||||
excalidrawRef(api);
|
||||
|
@ -407,6 +413,48 @@ class App extends React.Component<AppProps, AppState> {
|
|||
this.actionManager.registerAction(createRedoAction(this.history));
|
||||
}
|
||||
|
||||
setCustomType = (name: string) => {
|
||||
this.setState({ elementType: "custom" });
|
||||
this.customElementName = name;
|
||||
};
|
||||
|
||||
renderCustomElement = (
|
||||
coords: { x: number; y: number },
|
||||
name: string = "",
|
||||
) => {
|
||||
const config = this.props.customElementsConfig!.find(
|
||||
(config) => config.name === name,
|
||||
)!;
|
||||
|
||||
const [gridX, gridY] = getGridPoint(
|
||||
coords.x,
|
||||
coords.y,
|
||||
this.state.gridSize,
|
||||
);
|
||||
|
||||
const width = config.width || 40;
|
||||
const height = config.height || 40;
|
||||
const customElement = newCustomElement(name, {
|
||||
type: "custom",
|
||||
x: gridX - width / 2,
|
||||
y: gridY - height / 2,
|
||||
strokeColor: this.state.currentItemStrokeColor,
|
||||
backgroundColor: this.state.currentItemBackgroundColor,
|
||||
fillStyle: this.state.currentItemFillStyle,
|
||||
strokeWidth: this.state.currentItemStrokeWidth,
|
||||
strokeStyle: this.state.currentItemStrokeStyle,
|
||||
roughness: this.state.currentItemRoughness,
|
||||
opacity: this.state.currentItemOpacity,
|
||||
strokeSharpness: this.state.currentItemLinearStrokeSharpness,
|
||||
width,
|
||||
height,
|
||||
});
|
||||
this.scene.replaceAllElements([
|
||||
...this.scene.getElementsIncludingDeleted(),
|
||||
customElement,
|
||||
]);
|
||||
};
|
||||
|
||||
private renderCanvas() {
|
||||
const canvasScale = window.devicePixelRatio;
|
||||
const {
|
||||
|
@ -530,6 +578,7 @@ class App extends React.Component<AppProps, AppState> {
|
|||
library={this.library}
|
||||
id={this.id}
|
||||
onImageAction={this.onImageAction}
|
||||
renderCustomElementWidget={this.props.renderCustomElementWidget}
|
||||
/>
|
||||
<div className="excalidraw-textEditorContainer" />
|
||||
<div className="excalidraw-contextMenuContainer" />
|
||||
|
@ -1224,6 +1273,7 @@ class App extends React.Component<AppProps, AppState> {
|
|||
imageCache: this.imageCache,
|
||||
isExporting: false,
|
||||
renderScrollbars: !this.deviceType.isMobile,
|
||||
customElementsConfig: this.props.customElementsConfig,
|
||||
},
|
||||
);
|
||||
|
||||
|
@ -2986,6 +3036,17 @@ class App extends React.Component<AppProps, AppState> {
|
|||
x,
|
||||
y,
|
||||
});
|
||||
} else if (this.state.elementType === "custom") {
|
||||
if (this.customElementName) {
|
||||
setCursor(this.canvas, CURSOR_TYPE.CROSSHAIR);
|
||||
this.renderCustomElement(
|
||||
{
|
||||
x: pointerDownState.origin.x,
|
||||
y: pointerDownState.origin.y,
|
||||
},
|
||||
this.customElementName,
|
||||
);
|
||||
}
|
||||
} else if (this.state.elementType === "freedraw") {
|
||||
this.handleFreeDrawElementOnPointerDown(
|
||||
event,
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue