Merge remote-tracking branch 'origin/master' into aakansha-custom-elements
This commit is contained in:
commit
645f9a5dc0
@ -29,7 +29,7 @@
|
|||||||
"@types/react": "17.0.39",
|
"@types/react": "17.0.39",
|
||||||
"@types/react-dom": "17.0.11",
|
"@types/react-dom": "17.0.11",
|
||||||
"@types/socket.io-client": "1.4.36",
|
"@types/socket.io-client": "1.4.36",
|
||||||
"browser-fs-access": "0.24.1",
|
"browser-fs-access": "0.29.1",
|
||||||
"clsx": "1.1.1",
|
"clsx": "1.1.1",
|
||||||
"fake-indexeddb": "3.1.7",
|
"fake-indexeddb": "3.1.7",
|
||||||
"firebase": "8.3.3",
|
"firebase": "8.3.3",
|
||||||
@ -38,7 +38,7 @@
|
|||||||
"image-blob-reduce": "3.0.1",
|
"image-blob-reduce": "3.0.1",
|
||||||
"jotai": "1.6.4",
|
"jotai": "1.6.4",
|
||||||
"lodash.throttle": "4.1.1",
|
"lodash.throttle": "4.1.1",
|
||||||
"nanoid": "3.1.32",
|
"nanoid": "3.3.3",
|
||||||
"open-color": "1.9.1",
|
"open-color": "1.9.1",
|
||||||
"pako": "1.0.11",
|
"pako": "1.0.11",
|
||||||
"perfect-freehand": "1.0.16",
|
"perfect-freehand": "1.0.16",
|
||||||
|
@ -25,9 +25,9 @@ export const actionAddToLibrary = register({
|
|||||||
}
|
}
|
||||||
|
|
||||||
return app.library
|
return app.library
|
||||||
.loadLibrary()
|
.getLatestLibrary()
|
||||||
.then((items) => {
|
.then((items) => {
|
||||||
return app.library.saveLibrary([
|
return app.library.setLibrary([
|
||||||
{
|
{
|
||||||
id: randomId(),
|
id: randomId(),
|
||||||
status: "unpublished",
|
status: "unpublished",
|
||||||
|
@ -15,7 +15,9 @@ export const actionCopy = register({
|
|||||||
name: "copy",
|
name: "copy",
|
||||||
trackEvent: { category: "element" },
|
trackEvent: { category: "element" },
|
||||||
perform: (elements, appState, _, app) => {
|
perform: (elements, appState, _, app) => {
|
||||||
copyToClipboard(getNonDeletedElements(elements), appState, app.files);
|
const selectedElements = getSelectedElements(elements, appState, true);
|
||||||
|
|
||||||
|
copyToClipboard(selectedElements, appState, app.files);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
commitToHistory: false,
|
commitToHistory: false,
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { getClientColors, getClientInitials } from "../clients";
|
import { getClientColors } from "../clients";
|
||||||
import { Avatar } from "../components/Avatar";
|
import { Avatar } from "../components/Avatar";
|
||||||
import { centerScrollOn } from "../scene/scroll";
|
import { centerScrollOn } from "../scene/scroll";
|
||||||
import { Collaborator } from "../types";
|
import { Collaborator } from "../types";
|
||||||
@ -43,16 +43,15 @@ export const actionGoToCollaborator = register({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const { background, stroke } = getClientColors(clientId, appState);
|
const { background, stroke } = getClientColors(clientId, appState);
|
||||||
const shortName = getClientInitials(collaborator.username);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Avatar
|
<Avatar
|
||||||
color={background}
|
color={background}
|
||||||
border={stroke}
|
border={stroke}
|
||||||
onClick={() => updateData(collaborator.pointer)}
|
onClick={() => updateData(collaborator.pointer)}
|
||||||
>
|
name={collaborator.username || ""}
|
||||||
{shortName}
|
src={collaborator.src}
|
||||||
</Avatar>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@ -503,20 +503,6 @@ export const actionChangeOpacity = register({
|
|||||||
max="100"
|
max="100"
|
||||||
step="10"
|
step="10"
|
||||||
onChange={(event) => updateData(+event.target.value)}
|
onChange={(event) => updateData(+event.target.value)}
|
||||||
onWheel={(event) => {
|
|
||||||
event.stopPropagation();
|
|
||||||
const target = event.target as HTMLInputElement;
|
|
||||||
const STEP = 10;
|
|
||||||
const MAX = 100;
|
|
||||||
const MIN = 0;
|
|
||||||
const value = +target.value;
|
|
||||||
|
|
||||||
if (event.deltaY < 0 && value < MAX) {
|
|
||||||
updateData(value + STEP);
|
|
||||||
} else if (event.deltaY > 0 && value > MIN) {
|
|
||||||
updateData(value - STEP);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
value={
|
value={
|
||||||
getFormValue(
|
getFormValue(
|
||||||
elements,
|
elements,
|
||||||
|
@ -117,6 +117,7 @@ export class ActionManager {
|
|||||||
trackAction(action, "keyboard", appState, elements, this.app, null);
|
trackAction(action, "keyboard", appState, elements, this.app, null);
|
||||||
|
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
this.updater(data[0].perform(elements, appState, value, this.app));
|
this.updater(data[0].perform(elements, appState, value, this.app));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,6 @@ import {
|
|||||||
ExcalidrawElement,
|
ExcalidrawElement,
|
||||||
NonDeletedExcalidrawElement,
|
NonDeletedExcalidrawElement,
|
||||||
} from "./element/types";
|
} from "./element/types";
|
||||||
import { getSelectedElements } from "./scene";
|
|
||||||
import { AppState, BinaryFiles } from "./types";
|
import { AppState, BinaryFiles } from "./types";
|
||||||
import { SVG_EXPORT_TAG } from "./scene/export";
|
import { SVG_EXPORT_TAG } from "./scene/export";
|
||||||
import { tryParseSpreadsheet, Spreadsheet, VALID_SPREADSHEET } from "./charts";
|
import { tryParseSpreadsheet, Spreadsheet, VALID_SPREADSHEET } from "./charts";
|
||||||
@ -12,7 +11,7 @@ import { isPromiseLike } from "./utils";
|
|||||||
|
|
||||||
type ElementsClipboard = {
|
type ElementsClipboard = {
|
||||||
type: typeof EXPORT_DATA_TYPES.excalidrawClipboard;
|
type: typeof EXPORT_DATA_TYPES.excalidrawClipboard;
|
||||||
elements: ExcalidrawElement[];
|
elements: readonly NonDeletedExcalidrawElement[];
|
||||||
files: BinaryFiles | undefined;
|
files: BinaryFiles | undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -57,19 +56,20 @@ const clipboardContainsElements = (
|
|||||||
export const copyToClipboard = async (
|
export const copyToClipboard = async (
|
||||||
elements: readonly NonDeletedExcalidrawElement[],
|
elements: readonly NonDeletedExcalidrawElement[],
|
||||||
appState: AppState,
|
appState: AppState,
|
||||||
files: BinaryFiles,
|
files: BinaryFiles | null,
|
||||||
) => {
|
) => {
|
||||||
// select binded text elements when copying
|
// select binded text elements when copying
|
||||||
const selectedElements = getSelectedElements(elements, appState, true);
|
|
||||||
const contents: ElementsClipboard = {
|
const contents: ElementsClipboard = {
|
||||||
type: EXPORT_DATA_TYPES.excalidrawClipboard,
|
type: EXPORT_DATA_TYPES.excalidrawClipboard,
|
||||||
elements: selectedElements,
|
elements,
|
||||||
files: selectedElements.reduce((acc, element) => {
|
files: files
|
||||||
if (isInitializedImageElement(element) && files[element.fileId]) {
|
? elements.reduce((acc, element) => {
|
||||||
acc[element.fileId] = files[element.fileId];
|
if (isInitializedImageElement(element) && files[element.fileId]) {
|
||||||
}
|
acc[element.fileId] = files[element.fileId];
|
||||||
return acc;
|
}
|
||||||
}, {} as BinaryFiles),
|
return acc;
|
||||||
|
}, {} as BinaryFiles)
|
||||||
|
: undefined,
|
||||||
};
|
};
|
||||||
const json = JSON.stringify(contents);
|
const json = JSON.stringify(contents);
|
||||||
CLIPBOARD = json;
|
CLIPBOARD = json;
|
||||||
|
@ -263,6 +263,7 @@ import {
|
|||||||
isPointHittingLinkIcon,
|
isPointHittingLinkIcon,
|
||||||
isLocalLink,
|
isLocalLink,
|
||||||
} from "../element/Hyperlink";
|
} from "../element/Hyperlink";
|
||||||
|
import { AbortError } from "../errors";
|
||||||
|
|
||||||
const defaultDeviceTypeContext: DeviceType = {
|
const defaultDeviceTypeContext: DeviceType = {
|
||||||
isMobile: false,
|
isMobile: false,
|
||||||
@ -769,21 +770,35 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
window.history.replaceState({}, APP_NAME, `?${query.toString()}`);
|
window.history.replaceState({}, APP_NAME, `?${query.toString()}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const defaultStatus = "published";
|
||||||
|
|
||||||
|
this.setState({ isLibraryOpen: true });
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const request = await fetch(decodeURIComponent(url));
|
await this.library.importLibrary(
|
||||||
const blob = await request.blob();
|
new Promise<LibraryItems>(async (resolve, reject) => {
|
||||||
const defaultStatus = "published";
|
try {
|
||||||
const libraryItems = await loadLibraryFromBlob(blob, defaultStatus);
|
const request = await fetch(decodeURIComponent(url));
|
||||||
if (
|
const blob = await request.blob();
|
||||||
token === this.id ||
|
const libraryItems = await loadLibraryFromBlob(blob, defaultStatus);
|
||||||
window.confirm(
|
|
||||||
t("alerts.confirmAddLibrary", {
|
if (
|
||||||
numShapes: libraryItems.length,
|
token === this.id ||
|
||||||
}),
|
window.confirm(
|
||||||
)
|
t("alerts.confirmAddLibrary", {
|
||||||
) {
|
numShapes: libraryItems.length,
|
||||||
await this.library.importLibrary(libraryItems, defaultStatus);
|
}),
|
||||||
}
|
)
|
||||||
|
) {
|
||||||
|
resolve(libraryItems);
|
||||||
|
} else {
|
||||||
|
reject(new AbortError());
|
||||||
|
}
|
||||||
|
} catch (error: any) {
|
||||||
|
reject(error);
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
);
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
this.setState({ errorMessage: t("errors.importLibraryError") });
|
this.setState({ errorMessage: t("errors.importLibraryError") });
|
||||||
@ -1339,6 +1354,7 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
}
|
}
|
||||||
this.cutAll();
|
this.cutAll();
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
});
|
});
|
||||||
|
|
||||||
private onCopy = withBatchedUpdates((event: ClipboardEvent) => {
|
private onCopy = withBatchedUpdates((event: ClipboardEvent) => {
|
||||||
@ -1350,6 +1366,7 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
}
|
}
|
||||||
this.copyAll();
|
this.copyAll();
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
});
|
});
|
||||||
|
|
||||||
private cutAll = () => {
|
private cutAll = () => {
|
||||||
@ -1743,6 +1760,11 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
collaborators?: SceneData["collaborators"];
|
collaborators?: SceneData["collaborators"];
|
||||||
commitToHistory?: SceneData["commitToHistory"];
|
commitToHistory?: SceneData["commitToHistory"];
|
||||||
libraryItems?:
|
libraryItems?:
|
||||||
|
| ((
|
||||||
|
currentLibraryItems: LibraryItems,
|
||||||
|
) =>
|
||||||
|
| Required<SceneData>["libraryItems"]
|
||||||
|
| Promise<Required<SceneData>["libraryItems"]>)
|
||||||
| Required<SceneData>["libraryItems"]
|
| Required<SceneData>["libraryItems"]
|
||||||
| Promise<Required<SceneData>["libraryItems"]>;
|
| Promise<Required<SceneData>["libraryItems"]>;
|
||||||
}) => {
|
}) => {
|
||||||
@ -1763,20 +1785,20 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (sceneData.libraryItems) {
|
if (sceneData.libraryItems) {
|
||||||
this.library.saveLibrary(
|
this.library.setLibrary((currentLibraryItems) => {
|
||||||
new Promise<LibraryItems>(async (resolve, reject) => {
|
const nextItems =
|
||||||
|
typeof sceneData.libraryItems === "function"
|
||||||
|
? sceneData.libraryItems(currentLibraryItems)
|
||||||
|
: sceneData.libraryItems;
|
||||||
|
|
||||||
|
return new Promise<LibraryItems>(async (resolve, reject) => {
|
||||||
try {
|
try {
|
||||||
resolve(
|
resolve(restoreLibraryItems(await nextItems, "unpublished"));
|
||||||
restoreLibraryItems(
|
} catch (error: any) {
|
||||||
await sceneData.libraryItems,
|
reject(error);
|
||||||
"unpublished",
|
|
||||||
),
|
|
||||||
);
|
|
||||||
} catch {
|
|
||||||
reject(new Error(t("errors.importLibraryError")));
|
|
||||||
}
|
}
|
||||||
}),
|
});
|
||||||
);
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@ -1945,8 +1967,10 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
this.setActiveTool({ ...this.state.activeTool, type: shape });
|
this.setActiveTool({ ...this.state.activeTool, type: shape });
|
||||||
|
event.stopPropagation();
|
||||||
} else if (event.key === KEYS.Q) {
|
} else if (event.key === KEYS.Q) {
|
||||||
this.toggleLock("keyboard");
|
this.toggleLock("keyboard");
|
||||||
|
event.stopPropagation();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (event.key === KEYS.SPACE && gesture.pointers.size === 0) {
|
if (event.key === KEYS.SPACE && gesture.pointers.size === 0) {
|
||||||
@ -1955,7 +1979,11 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (event.key === KEYS.G || event.key === KEYS.S) {
|
if (
|
||||||
|
(event.key === KEYS.G || event.key === KEYS.S) &&
|
||||||
|
!event.altKey &&
|
||||||
|
!event[KEYS.CTRL_OR_CMD]
|
||||||
|
) {
|
||||||
const selectedElements = getSelectedElements(
|
const selectedElements = getSelectedElements(
|
||||||
this.scene.getElements(),
|
this.scene.getElements(),
|
||||||
this.state,
|
this.state,
|
||||||
@ -1973,9 +2001,11 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
selectedElements.some((element) => hasBackground(element.type)))
|
selectedElements.some((element) => hasBackground(element.type)))
|
||||||
) {
|
) {
|
||||||
this.setState({ openPopup: "backgroundColorPicker" });
|
this.setState({ openPopup: "backgroundColorPicker" });
|
||||||
|
event.stopPropagation();
|
||||||
}
|
}
|
||||||
if (event.key === KEYS.S) {
|
if (event.key === KEYS.S) {
|
||||||
this.setState({ openPopup: "strokeColorPicker" });
|
this.setState({ openPopup: "strokeColorPicker" });
|
||||||
|
event.stopPropagation();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -2300,8 +2330,7 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
if (isTextElement(selectedElements[0])) {
|
if (isTextElement(selectedElements[0])) {
|
||||||
existingTextElement = selectedElements[0];
|
existingTextElement = selectedElements[0];
|
||||||
} else if (isTextBindableContainer(selectedElements[0], false)) {
|
} else if (isTextBindableContainer(selectedElements[0], false)) {
|
||||||
container = selectedElements[0];
|
existingTextElement = getBoundTextElement(selectedElements[0]);
|
||||||
existingTextElement = getBoundTextElement(container);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5374,11 +5403,14 @@ class App extends React.Component<AppProps, AppState> {
|
|||||||
file?.type === MIME_TYPES.excalidrawlib ||
|
file?.type === MIME_TYPES.excalidrawlib ||
|
||||||
file?.name?.endsWith(".excalidrawlib")
|
file?.name?.endsWith(".excalidrawlib")
|
||||||
) {
|
) {
|
||||||
this.library
|
this.setState({ isLibraryOpen: true });
|
||||||
.importLibrary(file)
|
this.library.importLibrary(file).catch((error) => {
|
||||||
.catch((error) =>
|
console.error(error);
|
||||||
this.setState({ isLoading: false, errorMessage: error.message }),
|
this.setState({
|
||||||
);
|
isLoading: false,
|
||||||
|
errorMessage: t("errors.importLibraryError"),
|
||||||
|
});
|
||||||
|
});
|
||||||
// default: assume an Excalidraw file regardless of extension/MimeType
|
// default: assume an Excalidraw file regardless of extension/MimeType
|
||||||
} else if (file) {
|
} else if (file) {
|
||||||
this.setState({ isLoading: true });
|
this.setState({ isLoading: true });
|
||||||
|
@ -12,5 +12,11 @@
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
|
|
||||||
|
&-img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border-radius: 100%;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,20 +1,28 @@
|
|||||||
import "./Avatar.scss";
|
import "./Avatar.scss";
|
||||||
|
|
||||||
import React from "react";
|
import React from "react";
|
||||||
|
import { getClientInitials } from "../clients";
|
||||||
|
|
||||||
type AvatarProps = {
|
type AvatarProps = {
|
||||||
children: string;
|
|
||||||
onClick: (e: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
onClick: (e: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
||||||
color: string;
|
color: string;
|
||||||
border: string;
|
border: string;
|
||||||
|
name: string;
|
||||||
|
src?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Avatar = ({ children, color, border, onClick }: AvatarProps) => (
|
export const Avatar = ({ color, border, onClick, name, src }: AvatarProps) => {
|
||||||
<div
|
const shortName = getClientInitials(name);
|
||||||
className="Avatar"
|
const style = src
|
||||||
style={{ background: color, border: `1px solid ${border}` }}
|
? undefined
|
||||||
onClick={onClick}
|
: { background: color, border: `1px solid ${border}` };
|
||||||
>
|
return (
|
||||||
{children}
|
<div className="Avatar" style={style} onClick={onClick}>
|
||||||
</div>
|
{src ? (
|
||||||
);
|
<img className="Avatar-img" src={src} alt={shortName} />
|
||||||
|
) : (
|
||||||
|
shortName
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
@ -13,6 +13,10 @@
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
margin: 2px 0;
|
margin: 2px 0;
|
||||||
|
|
||||||
|
.Spinner {
|
||||||
|
margin-right: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
button {
|
button {
|
||||||
// 2px from the left to account for focus border of left-most button
|
// 2px from the left to account for focus border of left-most button
|
||||||
margin: 0 2px;
|
margin: 0 2px;
|
||||||
|
@ -139,7 +139,7 @@ export const LibraryMenu = ({
|
|||||||
const nextItems = libraryItems.filter(
|
const nextItems = libraryItems.filter(
|
||||||
(item) => !selectedItems.includes(item.id),
|
(item) => !selectedItems.includes(item.id),
|
||||||
);
|
);
|
||||||
library.saveLibrary(nextItems).catch(() => {
|
library.setLibrary(nextItems).catch(() => {
|
||||||
setAppState({ errorMessage: t("alerts.errorRemovingFromLibrary") });
|
setAppState({ errorMessage: t("alerts.errorRemovingFromLibrary") });
|
||||||
});
|
});
|
||||||
setSelectedItems([]);
|
setSelectedItems([]);
|
||||||
@ -170,7 +170,7 @@ export const LibraryMenu = ({
|
|||||||
...libraryItems,
|
...libraryItems,
|
||||||
];
|
];
|
||||||
onAddToLibrary();
|
onAddToLibrary();
|
||||||
library.saveLibrary(nextItems).catch(() => {
|
library.setLibrary(nextItems).catch(() => {
|
||||||
setAppState({ errorMessage: t("alerts.errorAddingToLibrary") });
|
setAppState({ errorMessage: t("alerts.errorAddingToLibrary") });
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
@ -220,7 +220,7 @@ export const LibraryMenu = ({
|
|||||||
libItem.status = "published";
|
libItem.status = "published";
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
library.saveLibrary(nextLibItems);
|
library.setLibrary(nextLibItems);
|
||||||
},
|
},
|
||||||
[setShowPublishLibraryDialog, setPublishLibSuccess, selectedItems, library],
|
[setShowPublishLibraryDialog, setPublishLibSuccess, selectedItems, library],
|
||||||
);
|
);
|
||||||
@ -229,7 +229,10 @@ export const LibraryMenu = ({
|
|||||||
LibraryItem["id"] | null
|
LibraryItem["id"] | null
|
||||||
>(null);
|
>(null);
|
||||||
|
|
||||||
if (libraryItemsData.status === "loading") {
|
if (
|
||||||
|
libraryItemsData.status === "loading" &&
|
||||||
|
!libraryItemsData.isInitialized
|
||||||
|
) {
|
||||||
return (
|
return (
|
||||||
<LibraryMenuWrapper ref={ref}>
|
<LibraryMenuWrapper ref={ref}>
|
||||||
<div className="layer-ui__library-message">
|
<div className="layer-ui__library-message">
|
||||||
@ -255,7 +258,7 @@ export const LibraryMenu = ({
|
|||||||
}
|
}
|
||||||
onError={(error) => window.alert(error)}
|
onError={(error) => window.alert(error)}
|
||||||
updateItemsInStorage={() =>
|
updateItemsInStorage={() =>
|
||||||
library.saveLibrary(libraryItemsData.libraryItems)
|
library.setLibrary(libraryItemsData.libraryItems)
|
||||||
}
|
}
|
||||||
onRemove={(id: string) =>
|
onRemove={(id: string) =>
|
||||||
setSelectedItems(selectedItems.filter((_id) => _id !== id))
|
setSelectedItems(selectedItems.filter((_id) => _id !== id))
|
||||||
@ -264,6 +267,7 @@ export const LibraryMenu = ({
|
|||||||
)}
|
)}
|
||||||
{publishLibSuccess && renderPublishSuccess()}
|
{publishLibSuccess && renderPublishSuccess()}
|
||||||
<LibraryMenuItems
|
<LibraryMenuItems
|
||||||
|
isLoading={libraryItemsData.status === "loading"}
|
||||||
libraryItems={libraryItemsData.libraryItems}
|
libraryItems={libraryItemsData.libraryItems}
|
||||||
onRemoveFromLibrary={() =>
|
onRemoveFromLibrary={() =>
|
||||||
removeFromLibrary(libraryItemsData.libraryItems)
|
removeFromLibrary(libraryItemsData.libraryItems)
|
||||||
|
@ -22,8 +22,10 @@ import { Tooltip } from "./Tooltip";
|
|||||||
|
|
||||||
import "./LibraryMenuItems.scss";
|
import "./LibraryMenuItems.scss";
|
||||||
import { VERSIONS } from "../constants";
|
import { VERSIONS } from "../constants";
|
||||||
|
import Spinner from "./Spinner";
|
||||||
|
|
||||||
const LibraryMenuItems = ({
|
const LibraryMenuItems = ({
|
||||||
|
isLoading,
|
||||||
libraryItems,
|
libraryItems,
|
||||||
onRemoveFromLibrary,
|
onRemoveFromLibrary,
|
||||||
onAddToLibrary,
|
onAddToLibrary,
|
||||||
@ -40,6 +42,7 @@ const LibraryMenuItems = ({
|
|||||||
onPublish,
|
onPublish,
|
||||||
resetLibrary,
|
resetLibrary,
|
||||||
}: {
|
}: {
|
||||||
|
isLoading: boolean;
|
||||||
libraryItems: LibraryItems;
|
libraryItems: LibraryItems;
|
||||||
pendingElements: LibraryItem["elements"];
|
pendingElements: LibraryItem["elements"];
|
||||||
onRemoveFromLibrary: () => void;
|
onRemoveFromLibrary: () => void;
|
||||||
@ -108,7 +111,8 @@ const LibraryMenuItems = ({
|
|||||||
importLibraryFromJSON(library)
|
importLibraryFromJSON(library)
|
||||||
.catch(muteFSAbortError)
|
.catch(muteFSAbortError)
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
setAppState({ errorMessage: error.message });
|
console.error(error);
|
||||||
|
setAppState({ errorMessage: t("errors.importLibraryError") });
|
||||||
});
|
});
|
||||||
}}
|
}}
|
||||||
className="library-actions--load"
|
className="library-actions--load"
|
||||||
@ -125,7 +129,7 @@ const LibraryMenuItems = ({
|
|||||||
onClick={async () => {
|
onClick={async () => {
|
||||||
const libraryItems = itemsSelected
|
const libraryItems = itemsSelected
|
||||||
? items
|
? items
|
||||||
: await library.loadLibrary();
|
: await library.getLatestLibrary();
|
||||||
saveLibraryAsJSON(libraryItems)
|
saveLibraryAsJSON(libraryItems)
|
||||||
.catch(muteFSAbortError)
|
.catch(muteFSAbortError)
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
@ -284,16 +288,20 @@ const LibraryMenuItems = ({
|
|||||||
{showRemoveLibAlert && renderRemoveLibAlert()}
|
{showRemoveLibAlert && renderRemoveLibAlert()}
|
||||||
<div className="layer-ui__library-header" key="library-header">
|
<div className="layer-ui__library-header" key="library-header">
|
||||||
{renderLibraryActions()}
|
{renderLibraryActions()}
|
||||||
<a
|
{isLoading ? (
|
||||||
href={`${process.env.REACT_APP_LIBRARY_URL}?target=${
|
<Spinner />
|
||||||
window.name || "_blank"
|
) : (
|
||||||
}&referrer=${referrer}&useHash=true&token=${id}&theme=${theme}&version=${
|
<a
|
||||||
VERSIONS.excalidrawLibrary
|
href={`${process.env.REACT_APP_LIBRARY_URL}?target=${
|
||||||
}`}
|
window.name || "_blank"
|
||||||
target="_excalidraw_libraries"
|
}&referrer=${referrer}&useHash=true&token=${id}&theme=${theme}&version=${
|
||||||
>
|
VERSIONS.excalidrawLibrary
|
||||||
{t("labels.libraries")}
|
}`}
|
||||||
</a>
|
target="_excalidraw_libraries"
|
||||||
|
>
|
||||||
|
{t("labels.libraries")}
|
||||||
|
</a>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
<Stack.Col
|
<Stack.Col
|
||||||
className="library-menu-items-container__items"
|
className="library-menu-items-container__items"
|
||||||
|
@ -155,7 +155,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
width: 2rem;
|
width: 2rem;
|
||||||
height: 2em;
|
height: 2rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
// container in body where the actual tooltip is appended to
|
// container in body where the actual tooltip is appended to
|
||||||
.excalidraw-tooltip {
|
.excalidraw-tooltip {
|
||||||
position: absolute;
|
position: fixed;
|
||||||
z-index: 1000;
|
z-index: 1000;
|
||||||
|
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
|
@ -94,7 +94,9 @@ export const MIME_TYPES = {
|
|||||||
excalidrawlib: "application/vnd.excalidrawlib+json",
|
excalidrawlib: "application/vnd.excalidrawlib+json",
|
||||||
json: "application/json",
|
json: "application/json",
|
||||||
svg: "image/svg+xml",
|
svg: "image/svg+xml",
|
||||||
|
"excalidraw.svg": "image/svg+xml",
|
||||||
png: "image/png",
|
png: "image/png",
|
||||||
|
"excalidraw.png": "image/png",
|
||||||
jpg: "image/jpeg",
|
jpg: "image/jpeg",
|
||||||
gif: "image/gif",
|
gif: "image/gif",
|
||||||
binary: "application/octet-stream",
|
binary: "application/octet-stream",
|
||||||
@ -106,7 +108,8 @@ export const EXPORT_DATA_TYPES = {
|
|||||||
excalidrawLibrary: "excalidrawlib",
|
excalidrawLibrary: "excalidrawlib",
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
export const EXPORT_SOURCE = window.location.origin;
|
export const EXPORT_SOURCE =
|
||||||
|
window.EXCALIDRAW_EXPORT_SOURCE || window.location.origin;
|
||||||
|
|
||||||
// time in milliseconds
|
// time in milliseconds
|
||||||
export const IMAGE_RENDER_TIMEOUT = 500;
|
export const IMAGE_RENDER_TIMEOUT = 500;
|
||||||
|
@ -13,7 +13,9 @@ type FILE_EXTENSION =
|
|||||||
| "gif"
|
| "gif"
|
||||||
| "jpg"
|
| "jpg"
|
||||||
| "png"
|
| "png"
|
||||||
|
| "excalidraw.png"
|
||||||
| "svg"
|
| "svg"
|
||||||
|
| "excalidraw.svg"
|
||||||
| "json"
|
| "json"
|
||||||
| "excalidraw"
|
| "excalidraw"
|
||||||
| "excalidrawlib";
|
| "excalidrawlib";
|
||||||
|
@ -56,7 +56,7 @@ export const exportCanvas = async (
|
|||||||
{
|
{
|
||||||
description: "Export to SVG",
|
description: "Export to SVG",
|
||||||
name,
|
name,
|
||||||
extension: "svg",
|
extension: appState.exportEmbedScene ? "excalidraw.svg" : "svg",
|
||||||
fileHandle,
|
fileHandle,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@ -89,7 +89,7 @@ export const exportCanvas = async (
|
|||||||
return await fileSave(blob, {
|
return await fileSave(blob, {
|
||||||
description: "Export to PNG",
|
description: "Export to PNG",
|
||||||
name,
|
name,
|
||||||
extension: "png",
|
extension: appState.exportEmbedScene ? "excalidraw.png" : "png",
|
||||||
fileHandle,
|
fileHandle,
|
||||||
});
|
});
|
||||||
} else if (type === "clipboard") {
|
} else if (type === "clipboard") {
|
||||||
|
@ -5,13 +5,12 @@ import type App from "../components/App";
|
|||||||
import { ImportedDataState } from "./types";
|
import { ImportedDataState } from "./types";
|
||||||
import { atom } from "jotai";
|
import { atom } from "jotai";
|
||||||
import { jotaiStore } from "../jotai";
|
import { jotaiStore } from "../jotai";
|
||||||
import { isPromiseLike } from "../utils";
|
|
||||||
import { t } from "../i18n";
|
|
||||||
|
|
||||||
export const libraryItemsAtom = atom<
|
export const libraryItemsAtom = atom<{
|
||||||
| { status: "loading"; libraryItems: null; promise: Promise<LibraryItems> }
|
status: "loading" | "loaded";
|
||||||
| { status: "loaded"; libraryItems: LibraryItems }
|
isInitialized: boolean;
|
||||||
>({ status: "loaded", libraryItems: [] });
|
libraryItems: LibraryItems;
|
||||||
|
}>({ status: "loaded", isInitialized: true, libraryItems: [] });
|
||||||
|
|
||||||
const cloneLibraryItems = (libraryItems: LibraryItems): LibraryItems =>
|
const cloneLibraryItems = (libraryItems: LibraryItems): LibraryItems =>
|
||||||
JSON.parse(JSON.stringify(libraryItems));
|
JSON.parse(JSON.stringify(libraryItems));
|
||||||
@ -40,12 +39,28 @@ const isUniqueItem = (
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/** Merges otherItems into localItems. Unique items in otherItems array are
|
||||||
|
sorted first. */
|
||||||
|
export const mergeLibraryItems = (
|
||||||
|
localItems: LibraryItems,
|
||||||
|
otherItems: LibraryItems,
|
||||||
|
): LibraryItems => {
|
||||||
|
const newItems = [];
|
||||||
|
for (const item of otherItems) {
|
||||||
|
if (isUniqueItem(localItems, item)) {
|
||||||
|
newItems.push(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return [...newItems, ...localItems];
|
||||||
|
};
|
||||||
|
|
||||||
class Library {
|
class Library {
|
||||||
/** cache for currently active promise when initializing/updating libaries
|
/** latest libraryItems */
|
||||||
asynchronously */
|
|
||||||
private libraryItemsPromise: Promise<LibraryItems> | null = null;
|
|
||||||
/** last resolved libraryItems */
|
|
||||||
private lastLibraryItems: LibraryItems = [];
|
private lastLibraryItems: LibraryItems = [];
|
||||||
|
/** indicates whether library is initialized with library items (has gone
|
||||||
|
* though at least one update) */
|
||||||
|
private isInitialized = false;
|
||||||
|
|
||||||
private app: App;
|
private app: App;
|
||||||
|
|
||||||
@ -53,95 +68,138 @@ class Library {
|
|||||||
this.app = app;
|
this.app = app;
|
||||||
}
|
}
|
||||||
|
|
||||||
resetLibrary = async () => {
|
private updateQueue: Promise<LibraryItems>[] = [];
|
||||||
this.saveLibrary([]);
|
|
||||||
|
private getLastUpdateTask = (): Promise<LibraryItems> | undefined => {
|
||||||
|
return this.updateQueue[this.updateQueue.length - 1];
|
||||||
};
|
};
|
||||||
|
|
||||||
/** imports library (currently merges, removing duplicates) */
|
private notifyListeners = () => {
|
||||||
async importLibrary(
|
if (this.updateQueue.length > 0) {
|
||||||
|
jotaiStore.set(libraryItemsAtom, {
|
||||||
|
status: "loading",
|
||||||
|
libraryItems: this.lastLibraryItems,
|
||||||
|
isInitialized: this.isInitialized,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
this.isInitialized = true;
|
||||||
|
jotaiStore.set(libraryItemsAtom, {
|
||||||
|
status: "loaded",
|
||||||
|
libraryItems: this.lastLibraryItems,
|
||||||
|
isInitialized: this.isInitialized,
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
this.app.props.onLibraryChange?.(
|
||||||
|
cloneLibraryItems(this.lastLibraryItems),
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
resetLibrary = () => {
|
||||||
|
return this.setLibrary([]);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* imports library (from blob or libraryItems), merging with current library
|
||||||
|
* (attempting to remove duplicates)
|
||||||
|
*/
|
||||||
|
importLibrary(
|
||||||
library:
|
library:
|
||||||
| Blob
|
| Blob
|
||||||
| Required<ImportedDataState>["libraryItems"]
|
| Required<ImportedDataState>["libraryItems"]
|
||||||
| Promise<Required<ImportedDataState>["libraryItems"]>,
|
| Promise<Required<ImportedDataState>["libraryItems"]>,
|
||||||
defaultStatus: LibraryItem["status"] = "unpublished",
|
defaultStatus: LibraryItem["status"] = "unpublished",
|
||||||
) {
|
): Promise<LibraryItems> {
|
||||||
return this.saveLibrary(
|
return this.setLibrary(
|
||||||
new Promise<LibraryItems>(async (resolve, reject) => {
|
() =>
|
||||||
try {
|
new Promise<LibraryItems>(async (resolve, reject) => {
|
||||||
let libraryItems: LibraryItems;
|
try {
|
||||||
if (library instanceof Blob) {
|
let libraryItems: LibraryItems;
|
||||||
libraryItems = await loadLibraryFromBlob(library, defaultStatus);
|
if (library instanceof Blob) {
|
||||||
} else {
|
libraryItems = await loadLibraryFromBlob(library, defaultStatus);
|
||||||
libraryItems = restoreLibraryItems(await library, defaultStatus);
|
} else {
|
||||||
}
|
libraryItems = restoreLibraryItems(await library, defaultStatus);
|
||||||
|
|
||||||
const existingLibraryItems = this.lastLibraryItems;
|
|
||||||
|
|
||||||
const filteredItems = [];
|
|
||||||
for (const item of libraryItems) {
|
|
||||||
if (isUniqueItem(existingLibraryItems, item)) {
|
|
||||||
filteredItems.push(item);
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
resolve([...filteredItems, ...existingLibraryItems]);
|
resolve(mergeLibraryItems(this.lastLibraryItems, libraryItems));
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
reject(new Error(t("errors.importLibraryError")));
|
reject(error);
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
loadLibrary = (): Promise<LibraryItems> => {
|
/**
|
||||||
|
* @returns latest cloned libraryItems. Awaits all in-progress updates first.
|
||||||
|
*/
|
||||||
|
getLatestLibrary = (): Promise<LibraryItems> => {
|
||||||
return new Promise(async (resolve) => {
|
return new Promise(async (resolve) => {
|
||||||
try {
|
try {
|
||||||
resolve(
|
const libraryItems = await (this.getLastUpdateTask() ||
|
||||||
cloneLibraryItems(
|
this.lastLibraryItems);
|
||||||
await (this.libraryItemsPromise || this.lastLibraryItems),
|
if (this.updateQueue.length > 0) {
|
||||||
),
|
resolve(this.getLatestLibrary());
|
||||||
);
|
} else {
|
||||||
|
resolve(cloneLibraryItems(libraryItems));
|
||||||
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
return resolve(this.lastLibraryItems);
|
return resolve(this.lastLibraryItems);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
saveLibrary = async (items: LibraryItems | Promise<LibraryItems>) => {
|
setLibrary = (
|
||||||
const prevLibraryItems = this.lastLibraryItems;
|
/**
|
||||||
try {
|
* LibraryItems that will replace current items. Can be a function which
|
||||||
let nextLibraryItems;
|
* will be invoked after all previous tasks are resolved
|
||||||
if (isPromiseLike(items)) {
|
* (this is the prefered way to update the library to avoid race conditions,
|
||||||
const promise = items.then((items) => cloneLibraryItems(items));
|
* but you'll want to manually merge the library items in the callback
|
||||||
this.libraryItemsPromise = promise;
|
* - which is what we're doing in Library.importLibrary()).
|
||||||
jotaiStore.set(libraryItemsAtom, {
|
*
|
||||||
status: "loading",
|
* If supplied promise is rejected with AbortError, we swallow it and
|
||||||
promise,
|
* do not update the library.
|
||||||
libraryItems: null,
|
*/
|
||||||
});
|
libraryItems:
|
||||||
nextLibraryItems = await promise;
|
| LibraryItems
|
||||||
} else {
|
| Promise<LibraryItems>
|
||||||
nextLibraryItems = cloneLibraryItems(items);
|
| ((
|
||||||
|
latestLibraryItems: LibraryItems,
|
||||||
|
) => LibraryItems | Promise<LibraryItems>),
|
||||||
|
): Promise<LibraryItems> => {
|
||||||
|
const task = new Promise<LibraryItems>(async (resolve, reject) => {
|
||||||
|
try {
|
||||||
|
await this.getLastUpdateTask();
|
||||||
|
|
||||||
|
if (typeof libraryItems === "function") {
|
||||||
|
libraryItems = libraryItems(this.lastLibraryItems);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.lastLibraryItems = cloneLibraryItems(await libraryItems);
|
||||||
|
|
||||||
|
resolve(this.lastLibraryItems);
|
||||||
|
} catch (error: any) {
|
||||||
|
reject(error);
|
||||||
}
|
}
|
||||||
|
})
|
||||||
this.lastLibraryItems = nextLibraryItems;
|
.catch((error) => {
|
||||||
this.libraryItemsPromise = null;
|
if (error.name === "AbortError") {
|
||||||
|
console.warn("Library update aborted by user");
|
||||||
jotaiStore.set(libraryItemsAtom, {
|
return this.lastLibraryItems;
|
||||||
status: "loaded",
|
}
|
||||||
libraryItems: nextLibraryItems,
|
throw error;
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
this.updateQueue = this.updateQueue.filter((_task) => _task !== task);
|
||||||
|
this.notifyListeners();
|
||||||
});
|
});
|
||||||
await this.app.props.onLibraryChange?.(
|
|
||||||
cloneLibraryItems(nextLibraryItems),
|
this.updateQueue.push(task);
|
||||||
);
|
this.notifyListeners();
|
||||||
} catch (error: any) {
|
|
||||||
this.lastLibraryItems = prevLibraryItems;
|
return task;
|
||||||
this.libraryItemsPromise = null;
|
|
||||||
jotaiStore.set(libraryItemsAtom, {
|
|
||||||
status: "loaded",
|
|
||||||
libraryItems: prevLibraryItems,
|
|
||||||
});
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -544,6 +544,29 @@ describe("textWysiwyg", () => {
|
|||||||
expect((h.elements[1] as ExcalidrawTextElement).containerId).toBe(null);
|
expect((h.elements[1] as ExcalidrawTextElement).containerId).toBe(null);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("should'nt bind text to container when not double clicked on center", async () => {
|
||||||
|
expect(h.elements.length).toBe(1);
|
||||||
|
expect(h.elements[0].id).toBe(rectangle.id);
|
||||||
|
|
||||||
|
// clicking somewhere on top left
|
||||||
|
mouse.doubleClickAt(rectangle.x + 20, rectangle.y + 20);
|
||||||
|
expect(h.elements.length).toBe(2);
|
||||||
|
|
||||||
|
const text = h.elements[1] as ExcalidrawTextElementWithContainer;
|
||||||
|
expect(text.type).toBe("text");
|
||||||
|
expect(text.containerId).toBe(null);
|
||||||
|
mouse.down();
|
||||||
|
const editor = document.querySelector(
|
||||||
|
".excalidraw-textEditorContainer > textarea",
|
||||||
|
) as HTMLTextAreaElement;
|
||||||
|
|
||||||
|
fireEvent.change(editor, { target: { value: "Hello World!" } });
|
||||||
|
|
||||||
|
await new Promise((r) => setTimeout(r, 0));
|
||||||
|
editor.blur();
|
||||||
|
expect(rectangle.boundElements).toBe(null);
|
||||||
|
});
|
||||||
|
|
||||||
it("should update font family correctly on undo/redo by selecting bounded text when font family was updated", async () => {
|
it("should update font family correctly on undo/redo by selecting bounded text when font family was updated", async () => {
|
||||||
expect(h.elements.length).toBe(1);
|
expect(h.elements.length).toBe(1);
|
||||||
|
|
||||||
|
@ -283,7 +283,14 @@ export const textWysiwyg = ({
|
|||||||
// using scrollHeight here since we need to calculate
|
// using scrollHeight here since we need to calculate
|
||||||
// number of lines so cannot use editable.style.height
|
// number of lines so cannot use editable.style.height
|
||||||
// as that gets updated below
|
// as that gets updated below
|
||||||
const lines = editable.scrollHeight / getApproxLineHeight(font);
|
// Rounding here so that the lines calculated is more accurate in all browsers.
|
||||||
|
// The scrollHeight and approxLineHeight differs in diff browsers
|
||||||
|
// eg it gives 1.05 in firefox for handewritten small font due to which
|
||||||
|
// height gets updated as lines > 1 and leads to jumping text for first line in bound container
|
||||||
|
// hence rounding here to avoid that
|
||||||
|
const lines = Math.round(
|
||||||
|
editable.scrollHeight / getApproxLineHeight(font),
|
||||||
|
);
|
||||||
// auto increase height only when lines > 1 so its
|
// auto increase height only when lines > 1 so its
|
||||||
// measured correctly and vertically aligns for
|
// measured correctly and vertically aligns for
|
||||||
// first line as well as setting height to "auto"
|
// first line as well as setting height to "auto"
|
||||||
@ -298,7 +305,6 @@ export const textWysiwyg = ({
|
|||||||
font,
|
font,
|
||||||
container!.width,
|
container!.width,
|
||||||
).split("\n").length;
|
).split("\n").length;
|
||||||
|
|
||||||
// This is browser behaviour when setting height to "auto"
|
// This is browser behaviour when setting height to "auto"
|
||||||
// It sets the height needed for 2 lines even if actual
|
// It sets the height needed for 2 lines even if actual
|
||||||
// line count is 1 as mentioned above as well
|
// line count is 1 as mentioned above as well
|
||||||
@ -316,8 +322,6 @@ export const textWysiwyg = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
editable.onkeydown = (event) => {
|
editable.onkeydown = (event) => {
|
||||||
event.stopPropagation();
|
|
||||||
|
|
||||||
if (!event.shiftKey && actionZoomIn.keyTest(event)) {
|
if (!event.shiftKey && actionZoomIn.keyTest(event)) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
app.actionManager.executeAction(actionZoomIn);
|
app.actionManager.executeAction(actionZoomIn);
|
||||||
|
@ -19,7 +19,8 @@ import {
|
|||||||
} from "../element/types";
|
} from "../element/types";
|
||||||
import { useCallbackRefState } from "../hooks/useCallbackRefState";
|
import { useCallbackRefState } from "../hooks/useCallbackRefState";
|
||||||
import { Language, t } from "../i18n";
|
import { Language, t } from "../i18n";
|
||||||
import Excalidraw, {
|
import {
|
||||||
|
Excalidraw,
|
||||||
defaultLang,
|
defaultLang,
|
||||||
languages,
|
languages,
|
||||||
} from "../packages/excalidraw/index";
|
} from "../packages/excalidraw/index";
|
||||||
|
1
src/global.d.ts
vendored
1
src/global.d.ts
vendored
@ -13,6 +13,7 @@ interface Window {
|
|||||||
ClipboardItem: any;
|
ClipboardItem: any;
|
||||||
__EXCALIDRAW_SHA__: string | undefined;
|
__EXCALIDRAW_SHA__: string | undefined;
|
||||||
EXCALIDRAW_ASSET_PATH: string | undefined;
|
EXCALIDRAW_ASSET_PATH: string | undefined;
|
||||||
|
EXCALIDRAW_EXPORT_SOURCE: string;
|
||||||
gtag: Function;
|
gtag: Function;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -17,11 +17,18 @@ Please add the latest change on the top under the correct section.
|
|||||||
|
|
||||||
#### Features
|
#### Features
|
||||||
|
|
||||||
|
- Export [`MIME_TYPES`](https://github.com/excalidraw/excalidraw/blob/master/src/constants.ts#L92) supported by Excalidraw [#5135](https://github.com/excalidraw/excalidraw/pull/5135).
|
||||||
|
- Support [`src`](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L50) for collaborators. Now onwards host can pass `src` to render the customized avatar for collaborators [#5114](https://github.com/excalidraw/excalidraw/pull/5114).
|
||||||
|
- Support `libraryItems` argument in `initialData.libraryItems` and `updateScene({ libraryItems })` to be a Promise resolving to `LibraryItems`, and support functional update of `libraryItems` in [`updateScene({ libraryItems })`](https://github.com/excalidraw/excalidraw/blob/master/src/packages/excalidraw/README.md#updateScene). [#5101](https://github.com/excalidraw/excalidraw/pull/5101).
|
||||||
|
- Expose util [`mergeLibraryItems`](https://github.com/excalidraw/excalidraw/blob/master/src/packages/excalidraw/README.md#mergeLibraryItems) [#5101](https://github.com/excalidraw/excalidraw/pull/5101).
|
||||||
|
- Expose util [`exportToClipboard`](https://github.com/excalidraw/excalidraw/blob/master/src/packages/excalidraw/README.md#exportToClipboard) which allows to copy the scene contents to clipboard as `svg`, `png` or `json` [#5103](https://github.com/excalidraw/excalidraw/pull/5103).
|
||||||
|
- Expose `window.EXCALIDRAW_EXPORT_SOURCE` which you can use to overwrite the `source` field in exported data [#5095](https://github.com/excalidraw/excalidraw/pull/5095).
|
||||||
- The `exportToBlob` utility now supports the `exportEmbedScene` option when generating a png image [#5047](https://github.com/excalidraw/excalidraw/pull/5047).
|
- The `exportToBlob` utility now supports the `exportEmbedScene` option when generating a png image [#5047](https://github.com/excalidraw/excalidraw/pull/5047).
|
||||||
- Exported [`restoreLibraryItems`](https://github.com/excalidraw/excalidraw/blob/master/src/packages/excalidraw/README.md#restoreLibraryItems) API [#4995](https://github.com/excalidraw/excalidraw/pull/4995).
|
- Exported [`restoreLibraryItems`](https://github.com/excalidraw/excalidraw/blob/master/src/packages/excalidraw/README.md#restoreLibraryItems) API [#4995](https://github.com/excalidraw/excalidraw/pull/4995).
|
||||||
|
|
||||||
#### Fixes
|
#### Fixes
|
||||||
|
|
||||||
|
- Use `window.EXCALIDRAW_ASSET_PATH` for fonts when exporting to svg [#5065](https://github.com/excalidraw/excalidraw/pull/5065).
|
||||||
- Library menu now properly rerenders if open when library is updated using `updateScene({ libraryItems })` [#4995](https://github.com/excalidraw/excalidraw/pull/4995).
|
- Library menu now properly rerenders if open when library is updated using `updateScene({ libraryItems })` [#4995](https://github.com/excalidraw/excalidraw/pull/4995).
|
||||||
|
|
||||||
#### Refactor
|
#### Refactor
|
||||||
|
@ -436,7 +436,7 @@ This helps to load Excalidraw with `initialData`. It must be an object or a [pro
|
|||||||
| `elements` | [ExcalidrawElement[]](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L78) | The elements with which Excalidraw should be mounted. |
|
| `elements` | [ExcalidrawElement[]](https://github.com/excalidraw/excalidraw/blob/master/src/element/types.ts#L78) | The elements with which Excalidraw should be mounted. |
|
||||||
| `appState` | [AppState](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L42) | The App state with which Excalidraw should be mounted. |
|
| `appState` | [AppState](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L42) | The App state with which Excalidraw should be mounted. |
|
||||||
| `scrollToContent` | boolean | This attribute implies whether to scroll to the nearest element to center once Excalidraw is mounted. By default, it will not scroll the nearest element to the center. Make sure you pass `initialData.appState.scrollX` and `initialData.appState.scrollY` when `scrollToContent` is false so that scroll positions are retained |
|
| `scrollToContent` | boolean | This attribute implies whether to scroll to the nearest element to center once Excalidraw is mounted. By default, it will not scroll the nearest element to the center. Make sure you pass `initialData.appState.scrollX` and `initialData.appState.scrollY` when `scrollToContent` is false so that scroll positions are retained |
|
||||||
| `libraryItems` | [LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L151) | This library items with which Excalidraw should be mounted. |
|
| `libraryItems` | [LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200) | Promise<[LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200)> | This library items with which Excalidraw should be mounted. |
|
||||||
| `files` | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L64) | The files added to the scene. |
|
| `files` | [BinaryFiles](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L64) | The files added to the scene. |
|
||||||
|
|
||||||
```json
|
```json
|
||||||
@ -512,9 +512,9 @@ You can use this function to update the scene with the sceneData. It accepts the
|
|||||||
| --- | --- | --- |
|
| --- | --- | --- |
|
||||||
| `elements` | [`ImportedDataState["elements"]`](https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L17) | The `elements` to be updated in the scene |
|
| `elements` | [`ImportedDataState["elements"]`](https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L17) | The `elements` to be updated in the scene |
|
||||||
| `appState` | [`ImportedDataState["appState"]`](https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L18) | The `appState` to be updated in the scene. |
|
| `appState` | [`ImportedDataState["appState"]`](https://github.com/excalidraw/excalidraw/blob/master/src/data/types.ts#L18) | The `appState` to be updated in the scene. |
|
||||||
| `collaborators` | <pre>Map<string, <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L29">Collaborator></a></pre> | The list of collaborators to be updated in the scene. |
|
| `collaborators` | <pre>Map<string, <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L35">Collaborator></a></pre> | The list of collaborators to be updated in the scene. |
|
||||||
| `commitToHistory` | `boolean` | Implies if the `history (undo/redo)` should be recorded. Defaults to `false`. |
|
| `commitToHistory` | `boolean` | Implies if the `history (undo/redo)` should be recorded. Defaults to `false`. |
|
||||||
| `libraryItems` | [LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L258) | The `libraryItems` to be update in the scene. |
|
| `libraryItems` | [LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200) | Promise<[LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200)> | ((currentItems: [LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200)>) => [LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200) | Promise<[LibraryItems](https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200)>) | The `libraryItems` to be update in the scene. |
|
||||||
|
|
||||||
### `addFiles`
|
### `addFiles`
|
||||||
|
|
||||||
@ -857,7 +857,7 @@ This function returns the canvas with the exported elements, appState and dimens
|
|||||||
|
|
||||||
<pre>
|
<pre>
|
||||||
exportToBlob(
|
exportToBlob(
|
||||||
opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/packages/utils.ts#L10">ExportOpts</a> & {
|
opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/packages/utils.ts#L14">ExportOpts</a> & {
|
||||||
mimeType?: string,
|
mimeType?: string,
|
||||||
quality?: number;
|
quality?: number;
|
||||||
})
|
})
|
||||||
@ -900,6 +900,34 @@ exportToSvg({
|
|||||||
|
|
||||||
This function returns a promise which resolves to svg of the exported drawing.
|
This function returns a promise which resolves to svg of the exported drawing.
|
||||||
|
|
||||||
|
#### `exportToClipboard`
|
||||||
|
|
||||||
|
**_Signature_**
|
||||||
|
|
||||||
|
<pre>
|
||||||
|
exportToClipboard(
|
||||||
|
opts: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/packages/utils.ts#L14">ExportOpts</a> & {
|
||||||
|
mimeType?: string,
|
||||||
|
quality?: number;
|
||||||
|
type: 'png' | 'svg' |'json'
|
||||||
|
})
|
||||||
|
</pre>
|
||||||
|
|
||||||
|
| Name | Type | Default | Description |
|
||||||
|
| --- | --- | --- | --- | --- | --- |
|
||||||
|
| opts | | | This param is same as the params passed to `exportToCanvas`. You can refer to [`exportToCanvas`](#exportToCanvas). |
|
||||||
|
| mimeType | string | "image/png" | Indicates the image format, this will be used when exporting as `png`. |
|
||||||
|
| quality | number | 0.92 | A value between 0 and 1 indicating the [image quality](https://developer.mozilla.org/en-US/docs/Web/API/HTMLCanvasElement/toBlob#parameters). Applies only to `image/jpeg`/`image/webp` MIME types. This will be used when exporting as `png`. |
|
||||||
|
| type | 'png' | 'svg' | 'json' | | This determines the format to which the scene data should be exported. |
|
||||||
|
|
||||||
|
**How to use**
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { exportToClipboard } from "@excalidraw/excalidraw-next";
|
||||||
|
```
|
||||||
|
|
||||||
|
Copies the scene data in the specified format (determined by `type`) to clipboard.
|
||||||
|
|
||||||
##### Additional attributes of appState for `export\*` APIs
|
##### Additional attributes of appState for `export\*` APIs
|
||||||
|
|
||||||
| Name | Type | Default | Description |
|
| Name | Type | Default | Description |
|
||||||
@ -924,17 +952,21 @@ serializeAsJSON({
|
|||||||
|
|
||||||
Takes the scene elements and state and returns a JSON string. Deleted `elements`as well as most properties from `AppState` are removed from the resulting JSON. (see [`serializeAsJSON()`](https://github.com/excalidraw/excalidraw/blob/master/src/data/json.ts#L16) source for details).
|
Takes the scene elements and state and returns a JSON string. Deleted `elements`as well as most properties from `AppState` are removed from the resulting JSON. (see [`serializeAsJSON()`](https://github.com/excalidraw/excalidraw/blob/master/src/data/json.ts#L16) source for details).
|
||||||
|
|
||||||
|
If you want to overwrite the source field in the JSON string, you can set `window.EXCALIDRAW_EXPORT_SOURCE` to the desired value.
|
||||||
|
|
||||||
#### `serializeLibraryAsJSON`
|
#### `serializeLibraryAsJSON`
|
||||||
|
|
||||||
**_Signature_**
|
**_Signature_**
|
||||||
|
|
||||||
<pre>
|
<pre>
|
||||||
serializeLibraryAsJSON({
|
serializeLibraryAsJSON({
|
||||||
libraryItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L191">LibraryItems[]</a>,
|
libraryItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200">LibraryItems[]</a>,
|
||||||
</pre>
|
</pre>
|
||||||
|
|
||||||
Takes the library items and returns a JSON string.
|
Takes the library items and returns a JSON string.
|
||||||
|
|
||||||
|
If you want to overwrite the source field in the JSON string, you can set `window.EXCALIDRAW_EXPORT_SOURCE` to the desired value.
|
||||||
|
|
||||||
#### `getSceneVersion`
|
#### `getSceneVersion`
|
||||||
|
|
||||||
**How to use**
|
**How to use**
|
||||||
@ -1040,6 +1072,20 @@ getNonDeletedElements(elements: <a href="https://github.com/excalidraw/excalidra
|
|||||||
|
|
||||||
This function returns an array of deleted elements.
|
This function returns an array of deleted elements.
|
||||||
|
|
||||||
|
#### `mergeLibraryItems`
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { mergeLibraryItems } from "@excalidraw/excalidraw-next";
|
||||||
|
```
|
||||||
|
|
||||||
|
**_Signature_**
|
||||||
|
|
||||||
|
<pre>
|
||||||
|
mergeLibraryItems(localItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200">LibraryItems</a>, otherItems: <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200">LibraryItems</a>) => <a href="https://github.com/excalidraw/excalidraw/blob/master/src/types.ts#L200">LibraryItems</a>
|
||||||
|
</pre>
|
||||||
|
|
||||||
|
This function merges two `LibraryItems` arrays, where unique items from `otherItems` are sorted first in the returned array.
|
||||||
|
|
||||||
### Exported constants
|
### Exported constants
|
||||||
|
|
||||||
#### `FONT_FAMILY`
|
#### `FONT_FAMILY`
|
||||||
@ -1077,6 +1123,16 @@ import { THEME } from "@excalidraw/excalidraw-next";
|
|||||||
|
|
||||||
Defaults to `THEME.LIGHT` unless passed in `initialData.appState.theme`
|
Defaults to `THEME.LIGHT` unless passed in `initialData.appState.theme`
|
||||||
|
|
||||||
|
### `MIME_TYPES`
|
||||||
|
|
||||||
|
**How to use **
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { MIME_TYPES } from "@excalidraw/excalidraw-next";
|
||||||
|
```
|
||||||
|
|
||||||
|
[`MIME_TYPES`](https://github.com/excalidraw/excalidraw/blob/master/src/constants.ts#L92) contains all the mime types supported by `Excalidraw`.
|
||||||
|
|
||||||
## Need help?
|
## Need help?
|
||||||
|
|
||||||
Check out the existing [Q&A](https://github.com/excalidraw/excalidraw/discussions?discussions_q=label%3Apackage%3Aexcalidraw). If you have any queries or need help, ask us [here](https://github.com/excalidraw/excalidraw/discussions?discussions_q=label%3Apackage%3Aexcalidraw).
|
Check out the existing [Q&A](https://github.com/excalidraw/excalidraw/discussions?discussions_q=label%3Apackage%3Aexcalidraw). If you have any queries or need help, ask us [here](https://github.com/excalidraw/excalidraw/discussions?discussions_q=label%3Apackage%3Aexcalidraw).
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
import Excalidraw from "./index";
|
import "./publicPath";
|
||||||
|
|
||||||
import "../../../public/fonts.css";
|
import "../../../public/fonts.css";
|
||||||
|
|
||||||
export { Excalidraw };
|
|
||||||
export * from "./index";
|
export * from "./index";
|
||||||
|
@ -1,14 +1,18 @@
|
|||||||
const dotenv = require("dotenv");
|
const dotenv = require("dotenv");
|
||||||
const { readFileSync } = require("fs");
|
const { readFileSync } = require("fs");
|
||||||
|
const pkg = require("./package.json");
|
||||||
const parseEnvVariables = (filepath) => {
|
const parseEnvVariables = (filepath) => {
|
||||||
return Object.entries(dotenv.parse(readFileSync(filepath))).reduce(
|
const envVars = Object.entries(dotenv.parse(readFileSync(filepath))).reduce(
|
||||||
(env, [key, value]) => {
|
(env, [key, value]) => {
|
||||||
env[key] = JSON.stringify(value);
|
env[key] = JSON.stringify(value);
|
||||||
return env;
|
return env;
|
||||||
},
|
},
|
||||||
{},
|
{},
|
||||||
);
|
);
|
||||||
|
envVars.PKG_NAME = JSON.stringify(pkg.name);
|
||||||
|
envVars.PKG_VERSION = JSON.stringify(pkg.version);
|
||||||
|
envVars.IS_EXCALIDRAW_NPM_PACKAGE = JSON.stringify(true);
|
||||||
|
return envVars;
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = { parseEnvVariables };
|
module.exports = { parseEnvVariables };
|
||||||
|
@ -5,15 +5,16 @@ import Sidebar from "./sidebar/Sidebar";
|
|||||||
|
|
||||||
import "./App.scss";
|
import "./App.scss";
|
||||||
import initialData from "./initialData";
|
import initialData from "./initialData";
|
||||||
import { MIME_TYPES } from "../../../constants";
|
|
||||||
|
|
||||||
// This is so that we use the bundled excalidraw.development.js file instead
|
// This is so that we use the bundled excalidraw.development.js file instead
|
||||||
// of the actual source code
|
// of the actual source code
|
||||||
const {
|
const {
|
||||||
Excalidraw,
|
|
||||||
exportToCanvas,
|
exportToCanvas,
|
||||||
exportToSvg,
|
exportToSvg,
|
||||||
exportToBlob,
|
exportToBlob,
|
||||||
|
exportToClipboard,
|
||||||
|
Excalidraw,
|
||||||
|
MIME_TYPES,
|
||||||
sceneCoordsToViewportCoords,
|
sceneCoordsToViewportCoords,
|
||||||
} = window.ExcalidrawLib;
|
} = window.ExcalidrawLib;
|
||||||
|
|
||||||
@ -49,7 +50,10 @@ const resolvablePromise = () => {
|
|||||||
|
|
||||||
const renderTopRightUI = () => {
|
const renderTopRightUI = () => {
|
||||||
return (
|
return (
|
||||||
<button onClick={() => alert("This is dummy top right UI")}>
|
<button
|
||||||
|
onClick={() => alert("This is dummy top right UI")}
|
||||||
|
style={{ height: "2.5rem" }}
|
||||||
|
>
|
||||||
{" "}
|
{" "}
|
||||||
Click me{" "}
|
Click me{" "}
|
||||||
</button>
|
</button>
|
||||||
@ -76,6 +80,7 @@ export default function App() {
|
|||||||
const [exportWithDarkMode, setExportWithDarkMode] = useState(false);
|
const [exportWithDarkMode, setExportWithDarkMode] = useState(false);
|
||||||
const [exportEmbedScene, setExportEmbedScene] = useState(false);
|
const [exportEmbedScene, setExportEmbedScene] = useState(false);
|
||||||
const [theme, setTheme] = useState("light");
|
const [theme, setTheme] = useState("light");
|
||||||
|
const [isCollaborating, setIsCollaborating] = useState(false);
|
||||||
|
|
||||||
const initialStatePromiseRef = useRef({ promise: null });
|
const initialStatePromiseRef = useRef({ promise: null });
|
||||||
if (!initialStatePromiseRef.current.promise) {
|
if (!initialStatePromiseRef.current.promise) {
|
||||||
@ -293,6 +298,16 @@ export default function App() {
|
|||||||
addTextArea(element);
|
addTextArea(element);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
const onCopy = async (type) => {
|
||||||
|
await exportToClipboard({
|
||||||
|
elements: excalidrawRef.current.getSceneElements(),
|
||||||
|
appState: excalidrawRef.current.getAppState(),
|
||||||
|
files: excalidrawRef.current.getFiles(),
|
||||||
|
type,
|
||||||
|
});
|
||||||
|
window.alert(`Copied to clipboard as ${type} sucessfully`);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="App">
|
<div className="App">
|
||||||
<h1> Excalidraw Example</h1>
|
<h1> Excalidraw Example</h1>
|
||||||
@ -327,6 +342,7 @@ export default function App() {
|
|||||||
>
|
>
|
||||||
Update Library
|
Update Library
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<label>
|
<label>
|
||||||
<input
|
<input
|
||||||
type="checkbox"
|
type="checkbox"
|
||||||
@ -365,6 +381,47 @@ export default function App() {
|
|||||||
/>
|
/>
|
||||||
Switch to Dark Theme
|
Switch to Dark Theme
|
||||||
</label>
|
</label>
|
||||||
|
<label>
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={isCollaborating}
|
||||||
|
onChange={() => {
|
||||||
|
if (!isCollaborating) {
|
||||||
|
const collaborators = new Map();
|
||||||
|
collaborators.set("id1", {
|
||||||
|
username: "Doremon",
|
||||||
|
src: "doremon.png",
|
||||||
|
});
|
||||||
|
collaborators.set("id2", {
|
||||||
|
username: "Excalibot",
|
||||||
|
src: "https://avatars.githubusercontent.com/excalibot",
|
||||||
|
});
|
||||||
|
collaborators.set("id3", {
|
||||||
|
username: "Pika",
|
||||||
|
src: "pika.jpeg",
|
||||||
|
});
|
||||||
|
excalidrawRef.current.updateScene({ collaborators });
|
||||||
|
} else {
|
||||||
|
excalidrawRef.current.updateScene({
|
||||||
|
collaborators: new Map(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
setIsCollaborating(!isCollaborating);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
Show collaborators
|
||||||
|
</label>
|
||||||
|
<div>
|
||||||
|
<button onClick={onCopy.bind(null, "png")}>
|
||||||
|
Copy to Clipboard as PNG
|
||||||
|
</button>
|
||||||
|
<button onClick={onCopy.bind(null, "svg")}>
|
||||||
|
Copy to Clipboard as SVG
|
||||||
|
</button>
|
||||||
|
<button onClick={onCopy.bind(null, "json")}>
|
||||||
|
Copy to Clipboard as JSON
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="excalidraw-wrapper" ref={excalidrawWrapperRef}>
|
<div className="excalidraw-wrapper" ref={excalidrawWrapperRef}>
|
||||||
<Excalidraw
|
<Excalidraw
|
||||||
|
BIN
src/packages/excalidraw/example/public/doremon.png
Normal file
BIN
src/packages/excalidraw/example/public/doremon.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 197 KiB |
BIN
src/packages/excalidraw/example/public/pika.jpeg
Normal file
BIN
src/packages/excalidraw/example/public/pika.jpeg
Normal file
Binary file not shown.
After Width: | Height: | Size: 6.1 KiB |
@ -1,5 +1,4 @@
|
|||||||
import React, { useEffect, forwardRef } from "react";
|
import React, { useEffect, forwardRef } from "react";
|
||||||
import "./publicPath";
|
|
||||||
|
|
||||||
import { InitializeApp } from "../../components/InitializeApp";
|
import { InitializeApp } from "../../components/InitializeApp";
|
||||||
import App from "../../components/App";
|
import App from "../../components/App";
|
||||||
@ -16,7 +15,7 @@ import {
|
|||||||
import { Provider } from "jotai";
|
import { Provider } from "jotai";
|
||||||
import { jotaiScope, jotaiStore } from "../../jotai";
|
import { jotaiScope, jotaiStore } from "../../jotai";
|
||||||
|
|
||||||
const Excalidraw = (props: ExcalidrawProps) => {
|
const ExcalidrawBase = (props: ExcalidrawProps) => {
|
||||||
const {
|
const {
|
||||||
onChange,
|
onChange,
|
||||||
initialData,
|
initialData,
|
||||||
@ -186,8 +185,10 @@ const areEqual = (
|
|||||||
const forwardedRefComp = forwardRef<
|
const forwardedRefComp = forwardRef<
|
||||||
ExcalidrawAPIRefValue,
|
ExcalidrawAPIRefValue,
|
||||||
PublicExcalidrawProps
|
PublicExcalidrawProps
|
||||||
>((props, ref) => <Excalidraw {...props} excalidrawRef={ref} />);
|
>((props, ref) => <ExcalidrawBase {...props} excalidrawRef={ref} />);
|
||||||
export default React.memo(forwardedRefComp, areEqual);
|
|
||||||
|
export const Excalidraw = React.memo(forwardedRefComp, areEqual);
|
||||||
|
|
||||||
export {
|
export {
|
||||||
getSceneVersion,
|
getSceneVersion,
|
||||||
isInvisiblySmallElement,
|
isInvisiblySmallElement,
|
||||||
@ -209,10 +210,12 @@ export {
|
|||||||
loadLibraryFromBlob,
|
loadLibraryFromBlob,
|
||||||
loadFromBlob,
|
loadFromBlob,
|
||||||
getFreeDrawSvgPath,
|
getFreeDrawSvgPath,
|
||||||
|
exportToClipboard,
|
||||||
|
mergeLibraryItems,
|
||||||
} from "../../packages/utils";
|
} from "../../packages/utils";
|
||||||
export { isLinearElement } from "../../element/typeChecks";
|
export { isLinearElement } from "../../element/typeChecks";
|
||||||
|
|
||||||
export { FONT_FAMILY, THEME } from "../../constants";
|
export { FONT_FAMILY, THEME, MIME_TYPES } from "../../constants";
|
||||||
|
|
||||||
export {
|
export {
|
||||||
mutateElement,
|
mutateElement,
|
||||||
|
@ -47,12 +47,12 @@
|
|||||||
"@babel/core": "7.17.0",
|
"@babel/core": "7.17.0",
|
||||||
"@babel/plugin-transform-arrow-functions": "7.16.7",
|
"@babel/plugin-transform-arrow-functions": "7.16.7",
|
||||||
"@babel/plugin-transform-async-to-generator": "7.16.0",
|
"@babel/plugin-transform-async-to-generator": "7.16.0",
|
||||||
"@babel/plugin-transform-runtime": "7.16.8",
|
"@babel/plugin-transform-runtime": "7.17.10",
|
||||||
"@babel/plugin-transform-typescript": "7.16.1",
|
"@babel/plugin-transform-typescript": "7.16.1",
|
||||||
"@babel/preset-env": "7.16.7",
|
"@babel/preset-env": "7.16.7",
|
||||||
"@babel/preset-react": "7.16.7",
|
"@babel/preset-react": "7.16.7",
|
||||||
"@babel/preset-typescript": "7.16.7",
|
"@babel/preset-typescript": "7.16.7",
|
||||||
"autoprefixer": "10.4.2",
|
"autoprefixer": "10.4.5",
|
||||||
"babel-loader": "8.2.3",
|
"babel-loader": "8.2.3",
|
||||||
"babel-plugin-transform-class-properties": "6.24.1",
|
"babel-plugin-transform-class-properties": "6.24.1",
|
||||||
"cross-env": "7.0.3",
|
"cross-env": "7.0.3",
|
||||||
@ -63,7 +63,7 @@
|
|||||||
"terser-webpack-plugin": "5.3.1",
|
"terser-webpack-plugin": "5.3.1",
|
||||||
"ts-loader": "9.2.8",
|
"ts-loader": "9.2.8",
|
||||||
"typescript": "4.5.4",
|
"typescript": "4.5.4",
|
||||||
"webpack": "5.65.0",
|
"webpack": "5.72.0",
|
||||||
"webpack-bundle-analyzer": "4.5.0",
|
"webpack-bundle-analyzer": "4.5.0",
|
||||||
"webpack-cli": "4.9.2",
|
"webpack-cli": "4.9.2",
|
||||||
"webpack-dev-server": "4.7.4",
|
"webpack-dev-server": "4.7.4",
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
import { ENV } from "../../constants";
|
import { ENV } from "../../constants";
|
||||||
import pkg from "./package.json";
|
|
||||||
if (process.env.NODE_ENV !== ENV.TEST) {
|
if (process.env.NODE_ENV !== ENV.TEST) {
|
||||||
/* eslint-disable */
|
/* eslint-disable */
|
||||||
/* global __webpack_public_path__:writable */
|
/* global __webpack_public_path__:writable */
|
||||||
__webpack_public_path__ =
|
__webpack_public_path__ =
|
||||||
window.EXCALIDRAW_ASSET_PATH ||
|
window.EXCALIDRAW_ASSET_PATH ||
|
||||||
`https://unpkg.com/${pkg.name}@${pkg.version}/dist/`;
|
`https://unpkg.com/${process.env.PKG_NAME}@${process.env.PKG_VERSION}/dist/`;
|
||||||
}
|
}
|
||||||
|
@ -862,10 +862,10 @@
|
|||||||
dependencies:
|
dependencies:
|
||||||
"@babel/helper-plugin-utils" "^7.16.7"
|
"@babel/helper-plugin-utils" "^7.16.7"
|
||||||
|
|
||||||
"@babel/plugin-transform-runtime@7.16.8":
|
"@babel/plugin-transform-runtime@7.17.10":
|
||||||
version "7.16.8"
|
version "7.17.10"
|
||||||
resolved "https://registry.yarnpkg.com/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.16.8.tgz#3339368701103edae708f0fba9e4bfb70a3e5872"
|
resolved "https://registry.yarnpkg.com/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.17.10.tgz#b89d821c55d61b5e3d3c3d1d636d8d5a81040ae1"
|
||||||
integrity sha512-6Kg2XHPFnIarNweZxmzbgYnnWsXxkx9WQUVk2sksBRL80lBC1RAQV3wQagWxdCHiYHqPN+oenwNIuttlYgIbQQ==
|
integrity sha512-6jrMilUAJhktTr56kACL8LnWC5hx3Lf27BS0R0DSyW/OoJfb/iTHeE96V3b1dgKG3FSFdd/0culnYWMkjcKCig==
|
||||||
dependencies:
|
dependencies:
|
||||||
"@babel/helper-module-imports" "^7.16.7"
|
"@babel/helper-module-imports" "^7.16.7"
|
||||||
"@babel/helper-plugin-utils" "^7.16.7"
|
"@babel/helper-plugin-utils" "^7.16.7"
|
||||||
@ -1174,10 +1174,10 @@
|
|||||||
dependencies:
|
dependencies:
|
||||||
"@types/node" "*"
|
"@types/node" "*"
|
||||||
|
|
||||||
"@types/eslint-scope@^3.7.0":
|
"@types/eslint-scope@^3.7.3":
|
||||||
version "3.7.0"
|
version "3.7.3"
|
||||||
resolved "https://registry.yarnpkg.com/@types/eslint-scope/-/eslint-scope-3.7.0.tgz#4792816e31119ebd506902a482caec4951fabd86"
|
resolved "https://registry.yarnpkg.com/@types/eslint-scope/-/eslint-scope-3.7.3.tgz#125b88504b61e3c8bc6f870882003253005c3224"
|
||||||
integrity sha512-O/ql2+rrCUe2W2rs7wMR+GqPRcgB6UiqN5RhrR5xruFlY7l9YLMn0ZkDzjoHLeiFkR8MCQZVudUuuvQ2BLC9Qw==
|
integrity sha512-PB3ldyrcnAicT35TWPs5IcwKD8S333HMaa2VVv4+wdvebJkjWuW/xESoB8IwRcog8HYVYamb1g/R31Qv5Bx03g==
|
||||||
dependencies:
|
dependencies:
|
||||||
"@types/eslint" "*"
|
"@types/eslint" "*"
|
||||||
"@types/estree" "*"
|
"@types/estree" "*"
|
||||||
@ -1190,10 +1190,10 @@
|
|||||||
"@types/estree" "*"
|
"@types/estree" "*"
|
||||||
"@types/json-schema" "*"
|
"@types/json-schema" "*"
|
||||||
|
|
||||||
"@types/estree@*", "@types/estree@^0.0.50":
|
"@types/estree@*", "@types/estree@^0.0.51":
|
||||||
version "0.0.50"
|
version "0.0.51"
|
||||||
resolved "https://registry.yarnpkg.com/@types/estree/-/estree-0.0.50.tgz#1e0caa9364d3fccd2931c3ed96fdbeaa5d4cca83"
|
resolved "https://registry.yarnpkg.com/@types/estree/-/estree-0.0.51.tgz#cfd70924a25a3fd32b218e5e420e6897e1ac4f40"
|
||||||
integrity sha512-C6N5s2ZFtuZRj54k2/zyRhNDjJwwcViAM3Nbm8zjBpbqAdZ00mr0CFxvSKeO8Y/e03WVFLpQMdHYVfUd6SB+Hw==
|
integrity sha512-CuPgU6f3eT/XgKKPqKd/gLZV1Xmvf1a2R5POBOGQa6uv82xpls89HU5zKeVoyR8XzHd1RGNOlQlvUe3CFkjWNQ==
|
||||||
|
|
||||||
"@types/express-serve-static-core@*", "@types/express-serve-static-core@^4.17.18":
|
"@types/express-serve-static-core@*", "@types/express-serve-static-core@^4.17.18":
|
||||||
version "4.17.27"
|
version "4.17.27"
|
||||||
@ -1571,20 +1571,20 @@ array-union@^2.1.0:
|
|||||||
integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==
|
integrity sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==
|
||||||
|
|
||||||
async@^2.6.2:
|
async@^2.6.2:
|
||||||
version "2.6.3"
|
version "2.6.4"
|
||||||
resolved "https://registry.yarnpkg.com/async/-/async-2.6.3.tgz#d72625e2344a3656e3a3ad4fa749fa83299d82ff"
|
resolved "https://registry.yarnpkg.com/async/-/async-2.6.4.tgz#706b7ff6084664cd7eae713f6f965433b5504221"
|
||||||
integrity sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==
|
integrity sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==
|
||||||
dependencies:
|
dependencies:
|
||||||
lodash "^4.17.14"
|
lodash "^4.17.14"
|
||||||
|
|
||||||
autoprefixer@10.4.2:
|
autoprefixer@10.4.5:
|
||||||
version "10.4.2"
|
version "10.4.5"
|
||||||
resolved "https://registry.yarnpkg.com/autoprefixer/-/autoprefixer-10.4.2.tgz#25e1df09a31a9fba5c40b578936b90d35c9d4d3b"
|
resolved "https://registry.yarnpkg.com/autoprefixer/-/autoprefixer-10.4.5.tgz#662193c744094b53d3637f39be477e07bd904998"
|
||||||
integrity sha512-9fOPpHKuDW1w/0EKfRmVnxTDt8166MAnLI3mgZ1JCnhNtYWxcJ6Ud5CO/AVOZi/AvFa8DY9RTy3h3+tFBlrrdQ==
|
integrity sha512-Fvd8yCoA7lNX/OUllvS+aS1I7WRBclGXsepbvT8ZaPgrH24rgXpZzF0/6Hh3ZEkwg+0AES/Osd196VZmYoEFtw==
|
||||||
dependencies:
|
dependencies:
|
||||||
browserslist "^4.19.1"
|
browserslist "^4.20.2"
|
||||||
caniuse-lite "^1.0.30001297"
|
caniuse-lite "^1.0.30001332"
|
||||||
fraction.js "^4.1.2"
|
fraction.js "^4.2.0"
|
||||||
normalize-range "^0.1.2"
|
normalize-range "^0.1.2"
|
||||||
picocolors "^1.0.0"
|
picocolors "^1.0.0"
|
||||||
postcss-value-parser "^4.2.0"
|
postcss-value-parser "^4.2.0"
|
||||||
@ -1800,15 +1800,15 @@ braces@^3.0.1, braces@~3.0.2:
|
|||||||
dependencies:
|
dependencies:
|
||||||
fill-range "^7.0.1"
|
fill-range "^7.0.1"
|
||||||
|
|
||||||
browserslist@^4.14.5, browserslist@^4.17.5, browserslist@^4.17.6, browserslist@^4.19.1:
|
browserslist@^4.14.5, browserslist@^4.17.5, browserslist@^4.17.6, browserslist@^4.19.1, browserslist@^4.20.2:
|
||||||
version "4.19.1"
|
version "4.20.2"
|
||||||
resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.19.1.tgz#4ac0435b35ab655896c31d53018b6dd5e9e4c9a3"
|
resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.20.2.tgz#567b41508757ecd904dab4d1c646c612cd3d4f88"
|
||||||
integrity sha512-u2tbbG5PdKRTUoctO3NBD8FQ5HdPh1ZXPHzp1rwaa5jTc+RV9/+RlWiAIKmjRPQF+xbGM9Kklj5bZQFa2s/38A==
|
integrity sha512-CQOBCqp/9pDvDbx3xfMi+86pr4KXIf2FDkTTdeuYw8OxS9t898LA1Khq57gtufFILXpfgsSx5woNgsBgvGjpsA==
|
||||||
dependencies:
|
dependencies:
|
||||||
caniuse-lite "^1.0.30001286"
|
caniuse-lite "^1.0.30001317"
|
||||||
electron-to-chromium "^1.4.17"
|
electron-to-chromium "^1.4.84"
|
||||||
escalade "^3.1.1"
|
escalade "^3.1.1"
|
||||||
node-releases "^2.0.1"
|
node-releases "^2.0.2"
|
||||||
picocolors "^1.0.0"
|
picocolors "^1.0.0"
|
||||||
|
|
||||||
buffer-from@^1.0.0:
|
buffer-from@^1.0.0:
|
||||||
@ -1844,10 +1844,10 @@ callsites@^3.0.0:
|
|||||||
resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73"
|
resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73"
|
||||||
integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==
|
integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==
|
||||||
|
|
||||||
caniuse-lite@^1.0.30001286, caniuse-lite@^1.0.30001297:
|
caniuse-lite@^1.0.30001317, caniuse-lite@^1.0.30001332:
|
||||||
version "1.0.30001298"
|
version "1.0.30001332"
|
||||||
resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001298.tgz#0e690039f62e91c3ea581673d716890512e7ec52"
|
resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001332.tgz#39476d3aa8d83ea76359c70302eafdd4a1d727dd"
|
||||||
integrity sha512-AcKqikjMLlvghZL/vfTHorlQsLDhGRalYf1+GmWCf5SCMziSGjRYQW/JEksj14NaYHIR6KIhrFAy0HV5C25UzQ==
|
integrity sha512-10T30NYOEQtN6C11YGg411yebhvpnC6Z102+B95eAsN0oB6KUs01ivE8u+G6FMIRtIrVlYXhL+LUwQ3/hXwDWw==
|
||||||
|
|
||||||
chalk@^1.1.3:
|
chalk@^1.1.3:
|
||||||
version "1.1.3"
|
version "1.1.3"
|
||||||
@ -2211,10 +2211,10 @@ ee-first@1.1.1:
|
|||||||
resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d"
|
resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d"
|
||||||
integrity sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=
|
integrity sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=
|
||||||
|
|
||||||
electron-to-chromium@^1.4.17:
|
electron-to-chromium@^1.4.84:
|
||||||
version "1.4.38"
|
version "1.4.118"
|
||||||
resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.38.tgz#10ea58d73d36b13e78d5024f3b74a352d3958d01"
|
resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.118.tgz#2d917c71712dac9652cc01af46c7d0bd51552974"
|
||||||
integrity sha512-WhHt3sZazKj0KK/UpgsbGQnUUoFeAHVishzHFExMxagpZgjiGYSC9S0ZlbhCfSH2L2i+2A1yyqOIliTctMx7KQ==
|
integrity sha512-maZIKjnYDvF7Fs35nvVcyr44UcKNwybr93Oba2n3HkKDFAtk0svERkLN/HyczJDS3Fo4wU9th9fUQd09ZLtj1w==
|
||||||
|
|
||||||
emojis-list@^3.0.0:
|
emojis-list@^3.0.0:
|
||||||
version "3.0.0"
|
version "3.0.0"
|
||||||
@ -2226,10 +2226,10 @@ encodeurl@~1.0.2:
|
|||||||
resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59"
|
resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59"
|
||||||
integrity sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=
|
integrity sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=
|
||||||
|
|
||||||
enhanced-resolve@^5.0.0, enhanced-resolve@^5.8.3:
|
enhanced-resolve@^5.0.0, enhanced-resolve@^5.9.2:
|
||||||
version "5.8.3"
|
version "5.9.2"
|
||||||
resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-5.8.3.tgz#6d552d465cce0423f5b3d718511ea53826a7b2f0"
|
resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-5.9.2.tgz#0224dcd6a43389ebfb2d55efee517e5466772dd9"
|
||||||
integrity sha512-EGAbGvH7j7Xt2nc0E7D99La1OiEs8LnyimkRgwExpUMScN6O+3x9tIWs7PLQZVNx4YD+00skHXPXi1yQHpAmZA==
|
integrity sha512-GIm3fQfwLJ8YZx2smuHpBKkXC1yOk+OBEmKckVyL0i/ea8mqDEykK3ld5dgH1QYPNyT/lIllxV2LULnxCHaHkA==
|
||||||
dependencies:
|
dependencies:
|
||||||
graceful-fs "^4.2.4"
|
graceful-fs "^4.2.4"
|
||||||
tapable "^2.2.0"
|
tapable "^2.2.0"
|
||||||
@ -2449,10 +2449,10 @@ forwarded@0.2.0:
|
|||||||
resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.2.0.tgz#2269936428aad4c15c7ebe9779a84bf0b2a81811"
|
resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.2.0.tgz#2269936428aad4c15c7ebe9779a84bf0b2a81811"
|
||||||
integrity sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==
|
integrity sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==
|
||||||
|
|
||||||
fraction.js@^4.1.2:
|
fraction.js@^4.2.0:
|
||||||
version "4.1.2"
|
version "4.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/fraction.js/-/fraction.js-4.1.2.tgz#13e420a92422b6cf244dff8690ed89401029fbe8"
|
resolved "https://registry.yarnpkg.com/fraction.js/-/fraction.js-4.2.0.tgz#448e5109a313a3527f5a3ab2119ec4cf0e0e2950"
|
||||||
integrity sha512-o2RiJQ6DZaR/5+Si0qJUIy637QMRudSi9kU/FFzx9EZazrIdnBgpU+3sEWCxAVhH2RtxW2Oz+T4p2o8uOPVcgA==
|
integrity sha512-MhLuK+2gUcnZe8ZHlaaINnQLl0xRIGRfcGk2yl8xoQAfHrSsL3rYu6FCmBdkdbhc9EPlwyGHewaRsvwRMJtAlA==
|
||||||
|
|
||||||
fresh@0.5.2:
|
fresh@0.5.2:
|
||||||
version "0.5.2"
|
version "0.5.2"
|
||||||
@ -2544,10 +2544,10 @@ globby@^11.0.1:
|
|||||||
merge2 "^1.3.0"
|
merge2 "^1.3.0"
|
||||||
slash "^3.0.0"
|
slash "^3.0.0"
|
||||||
|
|
||||||
graceful-fs@^4.1.2, graceful-fs@^4.2.4, graceful-fs@^4.2.6:
|
graceful-fs@^4.1.2, graceful-fs@^4.2.4, graceful-fs@^4.2.6, graceful-fs@^4.2.9:
|
||||||
version "4.2.8"
|
version "4.2.10"
|
||||||
resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.8.tgz#e412b8d33f5e006593cbd3cee6df9f2cebbe802a"
|
resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.10.tgz#147d3a006da4ca3ce14728c7aefc287c367d7a6c"
|
||||||
integrity sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==
|
integrity sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==
|
||||||
|
|
||||||
gzip-size@^6.0.0:
|
gzip-size@^6.0.0:
|
||||||
version "6.0.0"
|
version "6.0.0"
|
||||||
@ -3044,24 +3044,12 @@ micromatch@^4.0.0, micromatch@^4.0.2, micromatch@^4.0.4:
|
|||||||
braces "^3.0.1"
|
braces "^3.0.1"
|
||||||
picomatch "^2.2.3"
|
picomatch "^2.2.3"
|
||||||
|
|
||||||
mime-db@1.45.0:
|
|
||||||
version "1.45.0"
|
|
||||||
resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.45.0.tgz#cceeda21ccd7c3a745eba2decd55d4b73e7879ea"
|
|
||||||
integrity sha512-CkqLUxUk15hofLoLyljJSrukZi8mAtgd+yE5uO4tqRZsdsAJKv0O+rFMhVDRJgozy+yG6md5KwuXhD4ocIoP+w==
|
|
||||||
|
|
||||||
mime-db@1.51.0, "mime-db@>= 1.43.0 < 2":
|
mime-db@1.51.0, "mime-db@>= 1.43.0 < 2":
|
||||||
version "1.51.0"
|
version "1.51.0"
|
||||||
resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.51.0.tgz#d9ff62451859b18342d960850dc3cfb77e63fb0c"
|
resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.51.0.tgz#d9ff62451859b18342d960850dc3cfb77e63fb0c"
|
||||||
integrity sha512-5y8A56jg7XVQx2mbv1lu49NR4dokRnhZYTtL+KGfaa27uq4pSTXkwQkFJl4pkRMyNFz/EtYDSkiiEHx3F7UN6g==
|
integrity sha512-5y8A56jg7XVQx2mbv1lu49NR4dokRnhZYTtL+KGfaa27uq4pSTXkwQkFJl4pkRMyNFz/EtYDSkiiEHx3F7UN6g==
|
||||||
|
|
||||||
mime-types@^2.1.27:
|
mime-types@^2.1.27, mime-types@^2.1.31, mime-types@~2.1.17, mime-types@~2.1.24:
|
||||||
version "2.1.28"
|
|
||||||
resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.28.tgz#1160c4757eab2c5363888e005273ecf79d2a0ecd"
|
|
||||||
integrity sha512-0TO2yJ5YHYr7M2zzT7gDU1tbwHxEUWBCLt0lscSNpcdAfFyJOVEpRYNS7EXVcTLNj/25QO8gulHC5JtTzSE2UQ==
|
|
||||||
dependencies:
|
|
||||||
mime-db "1.45.0"
|
|
||||||
|
|
||||||
mime-types@^2.1.31, mime-types@~2.1.17, mime-types@~2.1.24:
|
|
||||||
version "2.1.34"
|
version "2.1.34"
|
||||||
resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.34.tgz#5a712f9ec1503511a945803640fafe09d3793c24"
|
resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.34.tgz#5a712f9ec1503511a945803640fafe09d3793c24"
|
||||||
integrity sha512-6cP692WwGIs9XXdOO4++N+7qjqv0rqxxVvJ3VHPh/Sc9mVZcQP+ZGhkKiTvWMQRr2tbHkJP/Yn7Y0npb3ZBs4A==
|
integrity sha512-6cP692WwGIs9XXdOO4++N+7qjqv0rqxxVvJ3VHPh/Sc9mVZcQP+ZGhkKiTvWMQRr2tbHkJP/Yn7Y0npb3ZBs4A==
|
||||||
@ -3103,9 +3091,9 @@ minimatch@^3.0.4:
|
|||||||
brace-expansion "^1.1.7"
|
brace-expansion "^1.1.7"
|
||||||
|
|
||||||
minimist@^1.2.0, minimist@^1.2.5:
|
minimist@^1.2.0, minimist@^1.2.5:
|
||||||
version "1.2.5"
|
version "1.2.6"
|
||||||
resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.5.tgz#67d66014b66a6a8aaa0c083c5fd58df4e4e97602"
|
resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.6.tgz#8637a5b759ea0d6e98702cfb3a9283323c93af44"
|
||||||
integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==
|
integrity sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q==
|
||||||
|
|
||||||
mkdirp@^0.5.5:
|
mkdirp@^0.5.5:
|
||||||
version "0.5.5"
|
version "0.5.5"
|
||||||
@ -3158,14 +3146,14 @@ neo-async@^2.6.2:
|
|||||||
integrity sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==
|
integrity sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==
|
||||||
|
|
||||||
node-forge@^1.2.0:
|
node-forge@^1.2.0:
|
||||||
version "1.2.1"
|
version "1.3.1"
|
||||||
resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-1.2.1.tgz#82794919071ef2eb5c509293325cec8afd0fd53c"
|
resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-1.3.1.tgz#be8da2af243b2417d5f646a770663a92b7e9ded3"
|
||||||
integrity sha512-Fcvtbb+zBcZXbTTVwqGA5W+MKBj56UjVRevvchv5XrcyXbmNdesfZL37nlcWOfpgHhgmxApw3tQbTr4CqNmX4w==
|
integrity sha512-dPEtOeMvF9VMcYV/1Wb8CPoVAXtp6MKMlcbAt4ddqmGqUJ6fQZFXkNZNkNlfevtNkGtaSoXf/vNNNSvgrdXwtA==
|
||||||
|
|
||||||
node-releases@^2.0.1:
|
node-releases@^2.0.2:
|
||||||
version "2.0.1"
|
version "2.0.3"
|
||||||
resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.1.tgz#3d1d395f204f1f2f29a54358b9fb678765ad2fc5"
|
resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.3.tgz#225ee7488e4a5e636da8da52854844f9d716ca96"
|
||||||
integrity sha512-CqyzN6z7Q6aMeF/ktcMVTzhAHCEpf8SOarwpzpf8pNBY2k5/oM34UHldUwp8VKI7uxct2HxSRdJjBaZeESzcxA==
|
integrity sha512-maHFz6OLqYxz+VQyCAtA3PTX4UP/53pa05fyDNc9CwjvJ0yEh6+xBwKsgCxMNhS8taUKBFYxfuiaD9U/55iFaw==
|
||||||
|
|
||||||
normalize-path@^3.0.0, normalize-path@~3.0.0:
|
normalize-path@^3.0.0, normalize-path@~3.0.0:
|
||||||
version "3.0.0"
|
version "3.0.0"
|
||||||
@ -4180,18 +4168,18 @@ webpack-merge@5.8.0, webpack-merge@^5.7.3:
|
|||||||
clone-deep "^4.0.1"
|
clone-deep "^4.0.1"
|
||||||
wildcard "^2.0.0"
|
wildcard "^2.0.0"
|
||||||
|
|
||||||
webpack-sources@^3.2.2:
|
webpack-sources@^3.2.3:
|
||||||
version "3.2.2"
|
version "3.2.3"
|
||||||
resolved "https://registry.yarnpkg.com/webpack-sources/-/webpack-sources-3.2.2.tgz#d88e3741833efec57c4c789b6010db9977545260"
|
resolved "https://registry.yarnpkg.com/webpack-sources/-/webpack-sources-3.2.3.tgz#2d4daab8451fd4b240cc27055ff6a0c2ccea0cde"
|
||||||
integrity sha512-cp5qdmHnu5T8wRg2G3vZZHoJPN14aqQ89SyQ11NpGH5zEMDCclt49rzo+MaRazk7/UeILhAI+/sEtcM+7Fr0nw==
|
integrity sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w==
|
||||||
|
|
||||||
webpack@5.65.0:
|
webpack@5.72.0:
|
||||||
version "5.65.0"
|
version "5.72.0"
|
||||||
resolved "https://registry.yarnpkg.com/webpack/-/webpack-5.65.0.tgz#ed2891d9145ba1f0d318e4ea4f89c3fa18e6f9be"
|
resolved "https://registry.yarnpkg.com/webpack/-/webpack-5.72.0.tgz#f8bc40d9c6bb489a4b7a8a685101d6022b8b6e28"
|
||||||
integrity sha512-Q5or2o6EKs7+oKmJo7LaqZaMOlDWQse9Tm5l1WAfU/ujLGN5Pb0SqGeVkN/4bpPmEqEP5RnVhiqsOtWtUVwGRw==
|
integrity sha512-qmSmbspI0Qo5ld49htys8GY9XhS9CGqFoHTsOVAnjBdg0Zn79y135R+k4IR4rKK6+eKaabMhJwiVB7xw0SJu5w==
|
||||||
dependencies:
|
dependencies:
|
||||||
"@types/eslint-scope" "^3.7.0"
|
"@types/eslint-scope" "^3.7.3"
|
||||||
"@types/estree" "^0.0.50"
|
"@types/estree" "^0.0.51"
|
||||||
"@webassemblyjs/ast" "1.11.1"
|
"@webassemblyjs/ast" "1.11.1"
|
||||||
"@webassemblyjs/wasm-edit" "1.11.1"
|
"@webassemblyjs/wasm-edit" "1.11.1"
|
||||||
"@webassemblyjs/wasm-parser" "1.11.1"
|
"@webassemblyjs/wasm-parser" "1.11.1"
|
||||||
@ -4199,12 +4187,12 @@ webpack@5.65.0:
|
|||||||
acorn-import-assertions "^1.7.6"
|
acorn-import-assertions "^1.7.6"
|
||||||
browserslist "^4.14.5"
|
browserslist "^4.14.5"
|
||||||
chrome-trace-event "^1.0.2"
|
chrome-trace-event "^1.0.2"
|
||||||
enhanced-resolve "^5.8.3"
|
enhanced-resolve "^5.9.2"
|
||||||
es-module-lexer "^0.9.0"
|
es-module-lexer "^0.9.0"
|
||||||
eslint-scope "5.1.1"
|
eslint-scope "5.1.1"
|
||||||
events "^3.2.0"
|
events "^3.2.0"
|
||||||
glob-to-regexp "^0.4.1"
|
glob-to-regexp "^0.4.1"
|
||||||
graceful-fs "^4.2.4"
|
graceful-fs "^4.2.9"
|
||||||
json-parse-better-errors "^1.0.2"
|
json-parse-better-errors "^1.0.2"
|
||||||
loader-runner "^4.2.0"
|
loader-runner "^4.2.0"
|
||||||
mime-types "^2.1.27"
|
mime-types "^2.1.27"
|
||||||
@ -4213,7 +4201,7 @@ webpack@5.65.0:
|
|||||||
tapable "^2.1.1"
|
tapable "^2.1.1"
|
||||||
terser-webpack-plugin "^5.1.3"
|
terser-webpack-plugin "^5.1.3"
|
||||||
watchpack "^2.3.1"
|
watchpack "^2.3.1"
|
||||||
webpack-sources "^3.2.2"
|
webpack-sources "^3.2.3"
|
||||||
|
|
||||||
websocket-driver@>=0.5.1, websocket-driver@^0.7.4:
|
websocket-driver@>=0.5.1, websocket-driver@^0.7.4:
|
||||||
version "0.7.4"
|
version "0.7.4"
|
||||||
|
@ -10,6 +10,11 @@ import { restore } from "../data/restore";
|
|||||||
import { MIME_TYPES } from "../constants";
|
import { MIME_TYPES } from "../constants";
|
||||||
import { encodePngMetadata } from "../data/image";
|
import { encodePngMetadata } from "../data/image";
|
||||||
import { serializeAsJSON } from "../data/json";
|
import { serializeAsJSON } from "../data/json";
|
||||||
|
import {
|
||||||
|
copyBlobToClipboardAsPng,
|
||||||
|
copyTextToSystemClipboard,
|
||||||
|
copyToClipboard,
|
||||||
|
} from "../clipboard";
|
||||||
|
|
||||||
type ExportOpts = {
|
type ExportOpts = {
|
||||||
elements: readonly NonDeleted<ExcalidrawElement>[];
|
elements: readonly NonDeleted<ExcalidrawElement>[];
|
||||||
@ -81,7 +86,7 @@ export const exportToBlob = async (
|
|||||||
mimeType?: string;
|
mimeType?: string;
|
||||||
quality?: number;
|
quality?: number;
|
||||||
},
|
},
|
||||||
): Promise<Blob | null> => {
|
): Promise<Blob> => {
|
||||||
let { mimeType = MIME_TYPES.png, quality } = opts;
|
let { mimeType = MIME_TYPES.png, quality } = opts;
|
||||||
|
|
||||||
if (mimeType === MIME_TYPES.png && typeof quality === "number") {
|
if (mimeType === MIME_TYPES.png && typeof quality === "number") {
|
||||||
@ -107,9 +112,12 @@ export const exportToBlob = async (
|
|||||||
|
|
||||||
quality = quality ? quality : /image\/jpe?g/.test(mimeType) ? 0.92 : 0.8;
|
quality = quality ? quality : /image\/jpe?g/.test(mimeType) ? 0.92 : 0.8;
|
||||||
|
|
||||||
return new Promise((resolve) => {
|
return new Promise((resolve, reject) => {
|
||||||
canvas.toBlob(
|
canvas.toBlob(
|
||||||
async (blob: Blob | null) => {
|
async (blob) => {
|
||||||
|
if (!blob) {
|
||||||
|
return reject(new Error("couldn't export to blob"));
|
||||||
|
}
|
||||||
if (
|
if (
|
||||||
blob &&
|
blob &&
|
||||||
mimeType === MIME_TYPES.png &&
|
mimeType === MIME_TYPES.png &&
|
||||||
@ -156,6 +164,34 @@ export const exportToSvg = async ({
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const exportToClipboard = async (
|
||||||
|
opts: ExportOpts & {
|
||||||
|
mimeType?: string;
|
||||||
|
quality?: number;
|
||||||
|
type: "png" | "svg" | "json";
|
||||||
|
},
|
||||||
|
) => {
|
||||||
|
if (opts.type === "svg") {
|
||||||
|
const svg = await exportToSvg(opts);
|
||||||
|
await copyTextToSystemClipboard(svg.outerHTML);
|
||||||
|
} else if (opts.type === "png") {
|
||||||
|
await copyBlobToClipboardAsPng(exportToBlob(opts));
|
||||||
|
} else if (opts.type === "json") {
|
||||||
|
const appState = {
|
||||||
|
offsetTop: 0,
|
||||||
|
offsetLeft: 0,
|
||||||
|
width: 0,
|
||||||
|
height: 0,
|
||||||
|
...getDefaultAppState(),
|
||||||
|
...opts.appState,
|
||||||
|
};
|
||||||
|
await copyToClipboard(opts.elements, appState, opts.files);
|
||||||
|
} else {
|
||||||
|
throw new Error("Invalid export type");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
export { serializeAsJSON, serializeLibraryAsJSON } from "../data/json";
|
export { serializeAsJSON, serializeLibraryAsJSON } from "../data/json";
|
||||||
export { loadFromBlob, loadLibraryFromBlob } from "../data/blob";
|
export { loadFromBlob, loadLibraryFromBlob } from "../data/blob";
|
||||||
export { getFreeDrawSvgPath } from "../renderer/renderElement";
|
export { getFreeDrawSvgPath } from "../renderer/renderElement";
|
||||||
|
export { mergeLibraryItems } from "../data/library";
|
||||||
|
@ -41,7 +41,7 @@
|
|||||||
"@babel/plugin-transform-typescript": "7.16.1",
|
"@babel/plugin-transform-typescript": "7.16.1",
|
||||||
"@babel/preset-env": "7.16.7",
|
"@babel/preset-env": "7.16.7",
|
||||||
"@babel/preset-typescript": "7.16.7",
|
"@babel/preset-typescript": "7.16.7",
|
||||||
"babel-loader": "8.2.3",
|
"babel-loader": "8.2.5",
|
||||||
"babel-plugin-transform-class-properties": "6.24.1",
|
"babel-plugin-transform-class-properties": "6.24.1",
|
||||||
"cross-env": "7.0.3",
|
"cross-env": "7.0.3",
|
||||||
"css-loader": "6.7.1",
|
"css-loader": "6.7.1",
|
||||||
|
@ -1320,13 +1320,13 @@ babel-helper-get-function-arity@^6.24.1:
|
|||||||
babel-runtime "^6.22.0"
|
babel-runtime "^6.22.0"
|
||||||
babel-types "^6.24.1"
|
babel-types "^6.24.1"
|
||||||
|
|
||||||
babel-loader@8.2.3:
|
babel-loader@8.2.5:
|
||||||
version "8.2.3"
|
version "8.2.5"
|
||||||
resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-8.2.3.tgz#8986b40f1a64cacfcb4b8429320085ef68b1342d"
|
resolved "https://registry.yarnpkg.com/babel-loader/-/babel-loader-8.2.5.tgz#d45f585e654d5a5d90f5350a779d7647c5ed512e"
|
||||||
integrity sha512-n4Zeta8NC3QAsuyiizu0GkmRcQ6clkV9WFUnUf1iXP//IeSKbWjofW3UHyZVwlOB4y039YQKefawyTn64Zwbuw==
|
integrity sha512-OSiFfH89LrEMiWd4pLNqGz4CwJDtbs2ZVc+iGu2HrkRfPxId9F2anQj38IxWpmRfsUY0aBZYi1EFcd3mhtRMLQ==
|
||||||
dependencies:
|
dependencies:
|
||||||
find-cache-dir "^3.3.1"
|
find-cache-dir "^3.3.1"
|
||||||
loader-utils "^1.4.0"
|
loader-utils "^2.0.0"
|
||||||
make-dir "^3.1.0"
|
make-dir "^3.1.0"
|
||||||
schema-utils "^2.6.5"
|
schema-utils "^2.6.5"
|
||||||
|
|
||||||
@ -2002,13 +2002,6 @@ json-schema-traverse@^0.4.1:
|
|||||||
resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660"
|
resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660"
|
||||||
integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==
|
integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==
|
||||||
|
|
||||||
json5@^1.0.1:
|
|
||||||
version "1.0.1"
|
|
||||||
resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.1.tgz#779fb0018604fa854eacbf6252180d83543e3dbe"
|
|
||||||
integrity sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==
|
|
||||||
dependencies:
|
|
||||||
minimist "^1.2.0"
|
|
||||||
|
|
||||||
json5@^2.1.2:
|
json5@^2.1.2:
|
||||||
version "2.2.0"
|
version "2.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.0.tgz#2dfefe720c6ba525d9ebd909950f0515316c89a3"
|
resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.0.tgz#2dfefe720c6ba525d9ebd909950f0515316c89a3"
|
||||||
@ -2031,15 +2024,6 @@ loader-runner@^4.2.0:
|
|||||||
resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-4.2.0.tgz#d7022380d66d14c5fb1d496b89864ebcfd478384"
|
resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-4.2.0.tgz#d7022380d66d14c5fb1d496b89864ebcfd478384"
|
||||||
integrity sha512-92+huvxMvYlMzMt0iIOukcwYBFpkYJdpl2xsZ7LrlayO7E8SOv+JJUEK17B/dJIHAOLMfh2dZZ/Y18WgmGtYNw==
|
integrity sha512-92+huvxMvYlMzMt0iIOukcwYBFpkYJdpl2xsZ7LrlayO7E8SOv+JJUEK17B/dJIHAOLMfh2dZZ/Y18WgmGtYNw==
|
||||||
|
|
||||||
loader-utils@^1.4.0:
|
|
||||||
version "1.4.0"
|
|
||||||
resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-1.4.0.tgz#c579b5e34cb34b1a74edc6c1fb36bfa371d5a613"
|
|
||||||
integrity sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==
|
|
||||||
dependencies:
|
|
||||||
big.js "^5.2.2"
|
|
||||||
emojis-list "^3.0.0"
|
|
||||||
json5 "^1.0.1"
|
|
||||||
|
|
||||||
loader-utils@^2.0.0:
|
loader-utils@^2.0.0:
|
||||||
version "2.0.0"
|
version "2.0.0"
|
||||||
resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-2.0.0.tgz#e4cace5b816d425a166b5f097e10cd12b36064b0"
|
resolved "https://registry.yarnpkg.com/loader-utils/-/loader-utils-2.0.0.tgz#e4cace5b816d425a166b5f097e10cd12b36064b0"
|
||||||
@ -2122,10 +2106,10 @@ mimic-fn@^2.1.0:
|
|||||||
resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b"
|
resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b"
|
||||||
integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==
|
integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==
|
||||||
|
|
||||||
minimist@^1.2.0, minimist@^1.2.5:
|
minimist@^1.2.5:
|
||||||
version "1.2.5"
|
version "1.2.6"
|
||||||
resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.5.tgz#67d66014b66a6a8aaa0c083c5fd58df4e4e97602"
|
resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.6.tgz#8637a5b759ea0d6e98702cfb3a9283323c93af44"
|
||||||
integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==
|
integrity sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q==
|
||||||
|
|
||||||
ms@2.0.0:
|
ms@2.0.0:
|
||||||
version "2.0.0"
|
version "2.0.0"
|
||||||
|
@ -115,6 +115,19 @@ export const exportToSvg = async (
|
|||||||
svgRoot.setAttribute("filter", THEME_FILTER);
|
svgRoot.setAttribute("filter", THEME_FILTER);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let assetPath = "https://excalidraw.com/";
|
||||||
|
|
||||||
|
// Asset path needs to be determined only when using package
|
||||||
|
if (process.env.IS_EXCALIDRAW_NPM_PACKAGE) {
|
||||||
|
assetPath =
|
||||||
|
window.EXCALIDRAW_ASSET_PATH ||
|
||||||
|
`https://unpkg.com/${process.env.PKG_NAME}@${process.env.PKG_VERSION}`;
|
||||||
|
|
||||||
|
if (assetPath?.startsWith("/")) {
|
||||||
|
assetPath = assetPath.replace("/", `${window.location.origin}/`);
|
||||||
|
}
|
||||||
|
assetPath = `${assetPath}/dist/excalidraw-assets/`;
|
||||||
|
}
|
||||||
svgRoot.innerHTML = `
|
svgRoot.innerHTML = `
|
||||||
${SVG_EXPORT_TAG}
|
${SVG_EXPORT_TAG}
|
||||||
${metadata}
|
${metadata}
|
||||||
@ -122,16 +135,15 @@ export const exportToSvg = async (
|
|||||||
<style>
|
<style>
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "Virgil";
|
font-family: "Virgil";
|
||||||
src: url("https://excalidraw.com/Virgil.woff2");
|
src: url("${assetPath}Virgil.woff2");
|
||||||
}
|
}
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "Cascadia";
|
font-family: "Cascadia";
|
||||||
src: url("https://excalidraw.com/Cascadia.woff2");
|
src: url("${assetPath}Cascadia.woff2");
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
</defs>
|
</defs>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
// render background rect
|
// render background rect
|
||||||
if (appState.exportBackground && viewBackgroundColor) {
|
if (appState.exportBackground && viewBackgroundColor) {
|
||||||
const rect = svgRoot.ownerDocument!.createElementNS(SVG_NS, "rect");
|
const rect = svgRoot.ownerDocument!.createElementNS(SVG_NS, "rect");
|
||||||
|
@ -14,12 +14,12 @@ describe("library", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("import library via drag&drop", async () => {
|
it("import library via drag&drop", async () => {
|
||||||
expect(await h.app.library.loadLibrary()).toEqual([]);
|
expect(await h.app.library.getLatestLibrary()).toEqual([]);
|
||||||
await API.drop(
|
await API.drop(
|
||||||
await API.loadFile("./fixtures/fixture_library.excalidrawlib"),
|
await API.loadFile("./fixtures/fixture_library.excalidrawlib"),
|
||||||
);
|
);
|
||||||
await waitFor(async () => {
|
await waitFor(async () => {
|
||||||
expect(await h.app.library.loadLibrary()).toEqual([
|
expect(await h.app.library.getLatestLibrary()).toEqual([
|
||||||
{
|
{
|
||||||
status: "unpublished",
|
status: "unpublished",
|
||||||
elements: [expect.objectContaining({ id: "A" })],
|
elements: [expect.objectContaining({ id: "A" })],
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { fireEvent, GlobalTestState, render } from "../test-utils";
|
import { fireEvent, GlobalTestState, render } from "../test-utils";
|
||||||
import Excalidraw from "../../packages/excalidraw/index";
|
import { Excalidraw } from "../../packages/excalidraw/index";
|
||||||
import { queryByText, queryByTestId } from "@testing-library/react";
|
import { queryByText, queryByTestId } from "@testing-library/react";
|
||||||
import { GRID_SIZE, THEME } from "../../constants";
|
import { GRID_SIZE, THEME } from "../../constants";
|
||||||
import { t } from "../../i18n";
|
import { t } from "../../i18n";
|
||||||
|
@ -7,6 +7,7 @@ import {
|
|||||||
} from "../fixtures/elementFixture";
|
} from "../fixtures/elementFixture";
|
||||||
|
|
||||||
describe("exportToSvg", () => {
|
describe("exportToSvg", () => {
|
||||||
|
window.EXCALIDRAW_ASSET_PATH = "/";
|
||||||
const ELEMENT_HEIGHT = 100;
|
const ELEMENT_HEIGHT = 100;
|
||||||
const ELEMENT_WIDTH = 100;
|
const ELEMENT_WIDTH = 100;
|
||||||
const ELEMENTS = [
|
const ELEMENTS = [
|
||||||
|
@ -4,7 +4,7 @@ import {
|
|||||||
restoreOriginalGetBoundingClientRect,
|
restoreOriginalGetBoundingClientRect,
|
||||||
waitFor,
|
waitFor,
|
||||||
} from "./test-utils";
|
} from "./test-utils";
|
||||||
import Excalidraw from "../packages/excalidraw/index";
|
import { Excalidraw } from "../packages/excalidraw/index";
|
||||||
import { API } from "./helpers/api";
|
import { API } from "./helpers/api";
|
||||||
|
|
||||||
const { h } = window;
|
const { h } = window;
|
||||||
|
@ -45,6 +45,9 @@ export type Collaborator = {
|
|||||||
background: string;
|
background: string;
|
||||||
stroke: string;
|
stroke: string;
|
||||||
};
|
};
|
||||||
|
// The url of the collaborator's avatar, defaults to username intials
|
||||||
|
// if not present
|
||||||
|
src?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type DataURL = string & { _brand: "DataURL" };
|
export type DataURL = string & { _brand: "DataURL" };
|
||||||
|
@ -575,6 +575,9 @@ export const arrayToMap = <T extends { id: string } | string>(
|
|||||||
export const isTestEnv = () =>
|
export const isTestEnv = () =>
|
||||||
typeof process !== "undefined" && process.env?.NODE_ENV === "test";
|
typeof process !== "undefined" && process.env?.NODE_ENV === "test";
|
||||||
|
|
||||||
|
export const isProdEnv = () =>
|
||||||
|
typeof process !== "undefined" && process.env?.NODE_ENV === "production";
|
||||||
|
|
||||||
export const wrapEvent = <T extends Event>(name: EVENT, nativeEvent: T) => {
|
export const wrapEvent = <T extends Event>(name: EVENT, nativeEvent: T) => {
|
||||||
return new CustomEvent(name, {
|
return new CustomEvent(name, {
|
||||||
detail: {
|
detail: {
|
||||||
|
28
yarn.lock
28
yarn.lock
@ -2971,9 +2971,9 @@ async-limiter@~1.0.0:
|
|||||||
integrity sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==
|
integrity sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==
|
||||||
|
|
||||||
async@^2.6.2:
|
async@^2.6.2:
|
||||||
version "2.6.3"
|
version "2.6.4"
|
||||||
resolved "https://registry.npmjs.org/async/-/async-2.6.3.tgz"
|
resolved "https://registry.yarnpkg.com/async/-/async-2.6.4.tgz#706b7ff6084664cd7eae713f6f965433b5504221"
|
||||||
integrity sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==
|
integrity sha512-mzo5dfJYwAn29PeiJ0zvwTo04zj8HDJj0Mn8TD7sno7q12prdbnasKJHhkm2c1LgrhlJ0teaea8860oxi51mGA==
|
||||||
dependencies:
|
dependencies:
|
||||||
lodash "^4.17.14"
|
lodash "^4.17.14"
|
||||||
|
|
||||||
@ -3394,10 +3394,10 @@ brorand@^1.0.1, brorand@^1.1.0:
|
|||||||
resolved "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz"
|
resolved "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz"
|
||||||
integrity sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=
|
integrity sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=
|
||||||
|
|
||||||
browser-fs-access@0.24.1:
|
browser-fs-access@0.29.1:
|
||||||
version "0.24.1"
|
version "0.29.1"
|
||||||
resolved "https://registry.yarnpkg.com/browser-fs-access/-/browser-fs-access-0.24.1.tgz#29b915fdcd2ef0972ebabc2b7685c6893d288a72"
|
resolved "https://registry.yarnpkg.com/browser-fs-access/-/browser-fs-access-0.29.1.tgz#8a9794c73cf86b9aec74201829999c597128379c"
|
||||||
integrity sha512-fRcwhfNej0h2Jy+Uodxjbc5PQvNkZyG9fXu3S6Mcv0kigKath5sL54GXfMtOay/A/ULMa956eQZ9lMWVPiLtMA==
|
integrity sha512-LSvVX5e21LRrXqVMhqtAwj5xPgDb+fXAIH80NsnCQ9xuZPs2xWsOREi24RKgZa1XOiQRbcmVrv87+ulOKsgjxw==
|
||||||
|
|
||||||
browser-process-hrtime@^1.0.0:
|
browser-process-hrtime@^1.0.0:
|
||||||
version "1.0.0"
|
version "1.0.0"
|
||||||
@ -8369,9 +8369,9 @@ minimatch@3.0.4, minimatch@^3.0.4:
|
|||||||
brace-expansion "^1.1.7"
|
brace-expansion "^1.1.7"
|
||||||
|
|
||||||
minimist@^1.1.1, minimist@^1.2.0, minimist@^1.2.5:
|
minimist@^1.1.1, minimist@^1.2.0, minimist@^1.2.5:
|
||||||
version "1.2.5"
|
version "1.2.6"
|
||||||
resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.5.tgz#67d66014b66a6a8aaa0c083c5fd58df4e4e97602"
|
resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.6.tgz#8637a5b759ea0d6e98702cfb3a9283323c93af44"
|
||||||
integrity sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==
|
integrity sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q==
|
||||||
|
|
||||||
minipass-collect@^1.0.2:
|
minipass-collect@^1.0.2:
|
||||||
version "1.0.2"
|
version "1.0.2"
|
||||||
@ -8510,10 +8510,10 @@ nan@^2.12.1:
|
|||||||
resolved "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz"
|
resolved "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz"
|
||||||
integrity sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==
|
integrity sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==
|
||||||
|
|
||||||
nanoid@3.1.32, nanoid@^3.1.20:
|
nanoid@3.3.3, nanoid@^3.1.20:
|
||||||
version "3.1.32"
|
version "3.3.3"
|
||||||
resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.1.32.tgz#8f96069e6239cc0a9ae8c0d3b41a3b4933a88c0a"
|
resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.3.tgz#fd8e8b7aa761fe807dba2d1b98fb7241bb724a25"
|
||||||
integrity sha512-F8mf7R3iT9bvThBoW4tGXhXFHCctyCiUUPrWF8WaTqa3h96d9QybkSeba43XVOOE3oiLfkVDe4bT8MeGmkrTxw==
|
integrity sha512-p1sjXuopFs0xg+fPASzQ28agW1oHD7xDsd9Xkf3T15H3c/cifrFHVwrh74PdoklAPi+i7MdRsE47vm2r6JoB+w==
|
||||||
|
|
||||||
nanomatch@^1.2.9:
|
nanomatch@^1.2.9:
|
||||||
version "1.2.13"
|
version "1.2.13"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user