enable version bumping for collaboration
This commit is contained in:
parent
30903fbe04
commit
1419f17175
@ -7,6 +7,7 @@ import { done } from "../components/icons";
|
|||||||
import { t } from "../i18n";
|
import { t } from "../i18n";
|
||||||
import { register } from "./register";
|
import { register } from "./register";
|
||||||
import { invalidateShapeForElement } from "../renderer/renderElement";
|
import { invalidateShapeForElement } from "../renderer/renderElement";
|
||||||
|
import { mutateElement } from "../element/mutateElement";
|
||||||
|
|
||||||
export const actionFinalize = register({
|
export const actionFinalize = register({
|
||||||
name: "finalize",
|
name: "finalize",
|
||||||
@ -18,10 +19,12 @@ export const actionFinalize = register({
|
|||||||
if (appState.multiElement) {
|
if (appState.multiElement) {
|
||||||
// pen and mouse have hover
|
// pen and mouse have hover
|
||||||
if (appState.lastPointerDownWith !== "touch") {
|
if (appState.lastPointerDownWith !== "touch") {
|
||||||
appState.multiElement.points = appState.multiElement.points.slice(
|
mutateElement(appState.multiElement, multiElement => {
|
||||||
|
multiElement.points = multiElement.points.slice(
|
||||||
0,
|
0,
|
||||||
appState.multiElement.points.length - 1,
|
multiElement.points.length - 1,
|
||||||
);
|
);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
if (isInvisiblySmallElement(appState.multiElement)) {
|
if (isInvisiblySmallElement(appState.multiElement)) {
|
||||||
newElements = newElements.slice(0, -1);
|
newElements = newElements.slice(0, -1);
|
||||||
|
@ -6,6 +6,7 @@ import {
|
|||||||
import { KEYS } from "../keys";
|
import { KEYS } from "../keys";
|
||||||
import { DEFAULT_FONT } from "../appState";
|
import { DEFAULT_FONT } from "../appState";
|
||||||
import { register } from "./register";
|
import { register } from "./register";
|
||||||
|
import { mutateTextElement } from "../element/mutateElement";
|
||||||
|
|
||||||
let copiedStyles: string = "{}";
|
let copiedStyles: string = "{}";
|
||||||
|
|
||||||
@ -44,8 +45,10 @@ export const actionPasteStyles = register({
|
|||||||
roughness: pastedElement?.roughness,
|
roughness: pastedElement?.roughness,
|
||||||
};
|
};
|
||||||
if (isTextElement(newElement)) {
|
if (isTextElement(newElement)) {
|
||||||
|
mutateTextElement(newElement, newElement => {
|
||||||
newElement.font = pastedElement?.font || DEFAULT_FONT;
|
newElement.font = pastedElement?.font || DEFAULT_FONT;
|
||||||
redrawTextBoundingBox(newElement);
|
redrawTextBoundingBox(newElement);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
return newElement;
|
return newElement;
|
||||||
}
|
}
|
||||||
|
@ -88,6 +88,7 @@ import { LayerUI } from "./LayerUI";
|
|||||||
import { ScrollBars } from "../scene/types";
|
import { ScrollBars } from "../scene/types";
|
||||||
import { invalidateShapeForElement } from "../renderer/renderElement";
|
import { invalidateShapeForElement } from "../renderer/renderElement";
|
||||||
import { generateCollaborationLink, getCollaborationLinkData } from "../data";
|
import { generateCollaborationLink, getCollaborationLinkData } from "../data";
|
||||||
|
import { mutateElement } from "../element/mutateElement";
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
// TEST HOOKS
|
// TEST HOOKS
|
||||||
@ -262,7 +263,29 @@ export class App extends React.Component<any, AppState> {
|
|||||||
sceneAppState || getDefaultAppState(),
|
sceneAppState || getDefaultAppState(),
|
||||||
{ scrollToContent: true },
|
{ scrollToContent: true },
|
||||||
);
|
);
|
||||||
|
if (elements == null || elements.length === 0) {
|
||||||
elements = restoredState.elements;
|
elements = restoredState.elements;
|
||||||
|
} else {
|
||||||
|
const elementMap = elements.reduce(
|
||||||
|
(
|
||||||
|
acc: { [key: string]: ExcalidrawElement },
|
||||||
|
element: ExcalidrawElement,
|
||||||
|
) => {
|
||||||
|
acc[element.id] = element;
|
||||||
|
return acc;
|
||||||
|
},
|
||||||
|
{},
|
||||||
|
);
|
||||||
|
elements = restoredState.elements.map(element => {
|
||||||
|
if (
|
||||||
|
elementMap.hasOwnProperty(element.id) &&
|
||||||
|
elementMap[element.id].version > element.version
|
||||||
|
) {
|
||||||
|
return elementMap[element.id];
|
||||||
|
}
|
||||||
|
return element;
|
||||||
|
});
|
||||||
|
}
|
||||||
this.setState({});
|
this.setState({});
|
||||||
if (this.socketInitialized === false) {
|
if (this.socketInitialized === false) {
|
||||||
this.socketInitialized = true;
|
this.socketInitialized = true;
|
||||||
@ -774,8 +797,10 @@ export class App extends React.Component<any, AppState> {
|
|||||||
textY = centerElementYInViewport;
|
textY = centerElementYInViewport;
|
||||||
|
|
||||||
// x and y will change after calling newTextElement function
|
// x and y will change after calling newTextElement function
|
||||||
|
mutateElement(element, element => {
|
||||||
element.x = centerElementX;
|
element.x = centerElementX;
|
||||||
element.y = centerElementY;
|
element.y = centerElementY;
|
||||||
|
});
|
||||||
} else if (!event.altKey) {
|
} else if (!event.altKey) {
|
||||||
const snappedToCenterPosition = this.getTextWysiwygSnappedToCenterPosition(
|
const snappedToCenterPosition = this.getTextWysiwygSnappedToCenterPosition(
|
||||||
x,
|
x,
|
||||||
@ -783,8 +808,10 @@ export class App extends React.Component<any, AppState> {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (snappedToCenterPosition) {
|
if (snappedToCenterPosition) {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.x = snappedToCenterPosition.elementCenterX;
|
element.x = snappedToCenterPosition.elementCenterX;
|
||||||
element.y = snappedToCenterPosition.elementCenterY;
|
element.y = snappedToCenterPosition.elementCenterY;
|
||||||
|
});
|
||||||
textX = snappedToCenterPosition.wysiwygX;
|
textX = snappedToCenterPosition.wysiwygX;
|
||||||
textY = snappedToCenterPosition.wysiwygY;
|
textY = snappedToCenterPosition.wysiwygY;
|
||||||
}
|
}
|
||||||
@ -1336,13 +1363,17 @@ export class App extends React.Component<any, AppState> {
|
|||||||
|
|
||||||
const dx = element.x + width + p1[0];
|
const dx = element.x + width + p1[0];
|
||||||
const dy = element.y + height + p1[1];
|
const dy = element.y + height + p1[1];
|
||||||
|
mutateElement(element, element => {
|
||||||
element.x = dx;
|
element.x = dx;
|
||||||
element.y = dy;
|
element.y = dy;
|
||||||
|
});
|
||||||
p1[0] = absPx - element.x;
|
p1[0] = absPx - element.x;
|
||||||
p1[1] = absPy - element.y;
|
p1[1] = absPy - element.y;
|
||||||
} else {
|
} else {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.x += deltaX;
|
element.x += deltaX;
|
||||||
element.y += deltaY;
|
element.y += deltaY;
|
||||||
|
});
|
||||||
p1[0] -= deltaX;
|
p1[0] -= deltaX;
|
||||||
p1[1] -= deltaY;
|
p1[1] -= deltaY;
|
||||||
}
|
}
|
||||||
@ -1452,9 +1483,9 @@ export class App extends React.Component<any, AppState> {
|
|||||||
event.shiftKey,
|
event.shiftKey,
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.width -= deltaX;
|
element.width -= deltaX;
|
||||||
element.x += deltaX;
|
element.x += deltaX;
|
||||||
|
|
||||||
if (event.shiftKey) {
|
if (event.shiftKey) {
|
||||||
element.y += element.height - element.width;
|
element.y += element.height - element.width;
|
||||||
element.height = element.width;
|
element.height = element.width;
|
||||||
@ -1462,6 +1493,7 @@ export class App extends React.Component<any, AppState> {
|
|||||||
element.height -= deltaY;
|
element.height -= deltaY;
|
||||||
element.y += deltaY;
|
element.y += deltaY;
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "ne":
|
case "ne":
|
||||||
@ -1484,6 +1516,7 @@ export class App extends React.Component<any, AppState> {
|
|||||||
event.shiftKey,
|
event.shiftKey,
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.width += deltaX;
|
element.width += deltaX;
|
||||||
if (event.shiftKey) {
|
if (event.shiftKey) {
|
||||||
element.y += element.height - element.width;
|
element.y += element.height - element.width;
|
||||||
@ -1492,6 +1525,7 @@ export class App extends React.Component<any, AppState> {
|
|||||||
element.height -= deltaY;
|
element.height -= deltaY;
|
||||||
element.y += deltaY;
|
element.y += deltaY;
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "sw":
|
case "sw":
|
||||||
@ -1514,6 +1548,7 @@ export class App extends React.Component<any, AppState> {
|
|||||||
event.shiftKey,
|
event.shiftKey,
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.width -= deltaX;
|
element.width -= deltaX;
|
||||||
element.x += deltaX;
|
element.x += deltaX;
|
||||||
if (event.shiftKey) {
|
if (event.shiftKey) {
|
||||||
@ -1521,6 +1556,7 @@ export class App extends React.Component<any, AppState> {
|
|||||||
} else {
|
} else {
|
||||||
element.height += deltaY;
|
element.height += deltaY;
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "se":
|
case "se":
|
||||||
@ -1543,6 +1579,7 @@ export class App extends React.Component<any, AppState> {
|
|||||||
event.shiftKey,
|
event.shiftKey,
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
mutateElement(element, element => {
|
||||||
if (event.shiftKey) {
|
if (event.shiftKey) {
|
||||||
element.width += deltaX;
|
element.width += deltaX;
|
||||||
element.height = element.width;
|
element.height = element.width;
|
||||||
@ -1550,11 +1587,14 @@ export class App extends React.Component<any, AppState> {
|
|||||||
element.width += deltaX;
|
element.width += deltaX;
|
||||||
element.height += deltaY;
|
element.height += deltaY;
|
||||||
}
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "n": {
|
case "n": {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.height -= deltaY;
|
element.height -= deltaY;
|
||||||
element.y += deltaY;
|
element.y += deltaY;
|
||||||
|
});
|
||||||
|
|
||||||
if (element.points.length > 0) {
|
if (element.points.length > 0) {
|
||||||
const len = element.points.length;
|
const len = element.points.length;
|
||||||
@ -1569,8 +1609,10 @@ export class App extends React.Component<any, AppState> {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "w": {
|
case "w": {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.width -= deltaX;
|
element.width -= deltaX;
|
||||||
element.x += deltaX;
|
element.x += deltaX;
|
||||||
|
});
|
||||||
|
|
||||||
if (element.points.length > 0) {
|
if (element.points.length > 0) {
|
||||||
const len = element.points.length;
|
const len = element.points.length;
|
||||||
@ -1584,29 +1626,37 @@ export class App extends React.Component<any, AppState> {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "s": {
|
case "s": {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.height += deltaY;
|
element.height += deltaY;
|
||||||
if (element.points.length > 0) {
|
if (element.points.length > 0) {
|
||||||
const len = element.points.length;
|
const len = element.points.length;
|
||||||
const points = [...element.points].sort((a, b) => a[1] - b[1]);
|
const points = [...element.points].sort(
|
||||||
|
(a, b) => a[1] - b[1],
|
||||||
|
);
|
||||||
|
|
||||||
for (let i = 1; i < points.length; ++i) {
|
for (let i = 1; i < points.length; ++i) {
|
||||||
const pnt = points[i];
|
const pnt = points[i];
|
||||||
pnt[1] += deltaY / (len - i);
|
pnt[1] += deltaY / (len - i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
});
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case "e": {
|
case "e": {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.width += deltaX;
|
element.width += deltaX;
|
||||||
if (element.points.length > 0) {
|
if (element.points.length > 0) {
|
||||||
const len = element.points.length;
|
const len = element.points.length;
|
||||||
const points = [...element.points].sort((a, b) => a[0] - b[0]);
|
const points = [...element.points].sort(
|
||||||
|
(a, b) => a[0] - b[0],
|
||||||
|
);
|
||||||
|
|
||||||
for (let i = 1; i < points.length; ++i) {
|
for (let i = 1; i < points.length; ++i) {
|
||||||
const pnt = points[i];
|
const pnt = points[i];
|
||||||
pnt[0] += deltaX / (len - i);
|
pnt[0] += deltaX / (len - i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
});
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1620,8 +1670,10 @@ export class App extends React.Component<any, AppState> {
|
|||||||
element,
|
element,
|
||||||
resizeHandle,
|
resizeHandle,
|
||||||
});
|
});
|
||||||
|
mutateElement(el, el => {
|
||||||
el.x = element.x;
|
el.x = element.x;
|
||||||
el.y = element.y;
|
el.y = element.y;
|
||||||
|
});
|
||||||
invalidateShapeForElement(el);
|
invalidateShapeForElement(el);
|
||||||
|
|
||||||
lastX = x;
|
lastX = x;
|
||||||
@ -1644,9 +1696,11 @@ export class App extends React.Component<any, AppState> {
|
|||||||
);
|
);
|
||||||
|
|
||||||
selectedElements.forEach(element => {
|
selectedElements.forEach(element => {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.x += x - lastX;
|
element.x += x - lastX;
|
||||||
element.y += y - lastY;
|
element.y += y - lastY;
|
||||||
});
|
});
|
||||||
|
});
|
||||||
lastX = x;
|
lastX = x;
|
||||||
lastY = y;
|
lastY = y;
|
||||||
this.setState({});
|
this.setState({});
|
||||||
@ -1707,11 +1761,13 @@ export class App extends React.Component<any, AppState> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mutateElement(draggingElement, draggingElement => {
|
||||||
draggingElement.x = x < originX ? originX - width : originX;
|
draggingElement.x = x < originX ? originX - width : originX;
|
||||||
draggingElement.y = y < originY ? originY - height : originY;
|
draggingElement.y = y < originY ? originY - height : originY;
|
||||||
|
|
||||||
draggingElement.width = width;
|
draggingElement.width = width;
|
||||||
draggingElement.height = height;
|
draggingElement.height = height;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
invalidateShapeForElement(draggingElement);
|
invalidateShapeForElement(draggingElement);
|
||||||
|
20
src/element/mutateElement.ts
Normal file
20
src/element/mutateElement.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import {
|
||||||
|
MutableExcalidrawElement,
|
||||||
|
MutableExcalidrawTextElement,
|
||||||
|
} from "./types";
|
||||||
|
|
||||||
|
export function mutateElement(
|
||||||
|
element: MutableExcalidrawElement,
|
||||||
|
callback: (mutatableElement: MutableExcalidrawElement) => void,
|
||||||
|
): void {
|
||||||
|
element.version++;
|
||||||
|
callback(element);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function mutateTextElement(
|
||||||
|
element: MutableExcalidrawTextElement,
|
||||||
|
callback: (mutatableElement: MutableExcalidrawTextElement) => void,
|
||||||
|
): void {
|
||||||
|
element.version++;
|
||||||
|
callback(element);
|
||||||
|
}
|
@ -33,6 +33,7 @@ export function newElement(
|
|||||||
opacity,
|
opacity,
|
||||||
seed: randomSeed(),
|
seed: randomSeed(),
|
||||||
points: [] as Point[],
|
points: [] as Point[],
|
||||||
|
version: 1,
|
||||||
};
|
};
|
||||||
return element;
|
return element;
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import { ExcalidrawElement } from "./types";
|
import { ExcalidrawElement, MutableExcalidrawElement } from "./types";
|
||||||
import { invalidateShapeForElement } from "../renderer/renderElement";
|
import { invalidateShapeForElement } from "../renderer/renderElement";
|
||||||
|
import { mutateElement } from "./mutateElement";
|
||||||
|
|
||||||
export function isInvisiblySmallElement(element: ExcalidrawElement): boolean {
|
export function isInvisiblySmallElement(element: ExcalidrawElement): boolean {
|
||||||
if (element.type === "arrow" || element.type === "line") {
|
if (element.type === "arrow" || element.type === "line") {
|
||||||
@ -35,7 +36,7 @@ export function getPerfectElementSize(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function resizePerfectLineForNWHandler(
|
export function resizePerfectLineForNWHandler(
|
||||||
element: ExcalidrawElement,
|
element: MutableExcalidrawElement,
|
||||||
x: number,
|
x: number,
|
||||||
y: number,
|
y: number,
|
||||||
) {
|
) {
|
||||||
@ -78,13 +79,17 @@ export function normalizeDimensions(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (element.width < 0) {
|
if (element.width < 0) {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.width = Math.abs(element.width);
|
element.width = Math.abs(element.width);
|
||||||
element.x -= element.width;
|
element.x -= element.width;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (element.height < 0) {
|
if (element.height < 0) {
|
||||||
|
mutateElement(element, element => {
|
||||||
element.height = Math.abs(element.height);
|
element.height = Math.abs(element.height);
|
||||||
element.y -= element.height;
|
element.y -= element.height;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
invalidateShapeForElement(element);
|
invalidateShapeForElement(element);
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
import { measureText } from "../utils";
|
import { measureText } from "../utils";
|
||||||
import { ExcalidrawTextElement } from "./types";
|
import { MutableExcalidrawTextElement } from "./types";
|
||||||
|
|
||||||
export const redrawTextBoundingBox = (element: ExcalidrawTextElement) => {
|
export const redrawTextBoundingBox = (
|
||||||
|
element: MutableExcalidrawTextElement,
|
||||||
|
) => {
|
||||||
const metrics = measureText(element.text, element.font);
|
const metrics = measureText(element.text, element.font);
|
||||||
element.width = metrics.width;
|
element.width = metrics.width;
|
||||||
element.height = metrics.height;
|
element.height = metrics.height;
|
||||||
|
@ -5,8 +5,10 @@ import { newElement } from "./newElement";
|
|||||||
* no computed data. The list of all ExcalidrawElements should be shareable
|
* no computed data. The list of all ExcalidrawElements should be shareable
|
||||||
* between peers and contain no state local to the peer.
|
* between peers and contain no state local to the peer.
|
||||||
*/
|
*/
|
||||||
export type ExcalidrawElement = ReturnType<typeof newElement>;
|
export type ExcalidrawElement = Readonly<ReturnType<typeof newElement>>;
|
||||||
export type ExcalidrawTextElement = ExcalidrawElement & {
|
export type MutableExcalidrawElement = ReturnType<typeof newElement>;
|
||||||
|
|
||||||
|
export type MutableExcalidrawTextElement = MutableExcalidrawElement & {
|
||||||
type: "text";
|
type: "text";
|
||||||
font: string;
|
font: string;
|
||||||
text: string;
|
text: string;
|
||||||
@ -15,4 +17,6 @@ export type ExcalidrawTextElement = ExcalidrawElement & {
|
|||||||
baseline: number;
|
baseline: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type ExcalidrawTextElement = Readonly<MutableExcalidrawTextElement>;
|
||||||
|
|
||||||
export type PointerType = "mouse" | "pen" | "touch";
|
export type PointerType = "mouse" | "pen" | "touch";
|
||||||
|
Loading…
x
Reference in New Issue
Block a user