@plait/core 0.0.13 → 0.0.17
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/board/board.component.d.ts +3 -1
- package/esm2020/board/board.component.mjs +14 -7
- package/esm2020/interfaces/board.mjs +1 -1
- package/esm2020/interfaces/history.mjs +2 -0
- package/esm2020/interfaces/index.mjs +2 -1
- package/esm2020/interfaces/operation.mjs +79 -2
- package/esm2020/plugins/create-board.mjs +8 -1
- package/esm2020/plugins/with-history.mjs +123 -0
- package/esm2020/transfroms/node.mjs +3 -2
- package/esm2020/utils/history.mjs +28 -0
- package/esm2020/utils/index.mjs +2 -1
- package/fesm2015/plait-core.mjs +389 -167
- package/fesm2015/plait-core.mjs.map +1 -1
- package/fesm2020/plait-core.mjs +408 -170
- package/fesm2020/plait-core.mjs.map +1 -1
- package/interfaces/board.d.ts +6 -0
- package/interfaces/history.d.ts +5 -0
- package/interfaces/index.d.ts +1 -0
- package/interfaces/operation.d.ts +7 -3
- package/package.json +1 -1
- package/plugins/with-history.d.ts +24 -0
- package/utils/history.d.ts +13 -0
- package/utils/index.d.ts +1 -0
- package/styles/styles.scss +0 -62
- package/styles/theme.scss +0 -14
|
@@ -21,6 +21,7 @@ export declare class PlaitBoardComponent implements OnInit, AfterViewInit, OnDes
|
|
|
21
21
|
plaitViewport: Viewport;
|
|
22
22
|
plaitPlugins: PlaitPlugin[];
|
|
23
23
|
plaitReadonly: boolean;
|
|
24
|
+
plaitAllowClearBoard: boolean;
|
|
24
25
|
plaitChange: EventEmitter<PlaitBoardChangeEvent>;
|
|
25
26
|
plaitBoardInitialized: EventEmitter<PlaitBoard>;
|
|
26
27
|
constructor(cdr: ChangeDetectorRef, renderer2: Renderer2);
|
|
@@ -28,6 +29,7 @@ export declare class PlaitBoardComponent implements OnInit, AfterViewInit, OnDes
|
|
|
28
29
|
ngAfterViewInit(): void;
|
|
29
30
|
initializePlugins(): void;
|
|
30
31
|
initializeEvents(): void;
|
|
32
|
+
refreshViewport(): void;
|
|
31
33
|
updateViewport(): void;
|
|
32
34
|
zoomIn(event: MouseEvent): void;
|
|
33
35
|
zoomOut(event: MouseEvent): void;
|
|
@@ -35,5 +37,5 @@ export declare class PlaitBoardComponent implements OnInit, AfterViewInit, OnDes
|
|
|
35
37
|
trackBy: (index: number, element: PlaitElement) => number;
|
|
36
38
|
ngOnDestroy(): void;
|
|
37
39
|
static ɵfac: i0.ɵɵFactoryDeclaration<PlaitBoardComponent, never>;
|
|
38
|
-
static ɵcmp: i0.ɵɵComponentDeclaration<PlaitBoardComponent, "plait-board", never, { "plaitValue": "plaitValue"; "plaitViewport": "plaitViewport"; "plaitPlugins": "plaitPlugins"; "plaitReadonly": "plaitReadonly"; }, { "plaitChange": "plaitChange"; "plaitBoardInitialized": "plaitBoardInitialized"; }, never, ["*"]>;
|
|
40
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<PlaitBoardComponent, "plait-board", never, { "plaitValue": "plaitValue"; "plaitViewport": "plaitViewport"; "plaitPlugins": "plaitPlugins"; "plaitReadonly": "plaitReadonly"; "plaitAllowClearBoard": "plaitAllowClearBoard"; }, { "plaitChange": "plaitChange"; "plaitBoardInitialized": "plaitBoardInitialized"; }, never, ["*"]>;
|
|
39
41
|
}
|
|
@@ -9,6 +9,7 @@ import { Transforms } from '../transfroms';
|
|
|
9
9
|
import { withSelection } from '../plugins/with-selection';
|
|
10
10
|
import { PlaitOperation } from '../interfaces/operation';
|
|
11
11
|
import { getViewBox } from '../utils/board';
|
|
12
|
+
import { withHistroy } from '../plugins/with-history';
|
|
12
13
|
import * as i0 from "@angular/core";
|
|
13
14
|
import * as i1 from "../core/element/element.component";
|
|
14
15
|
import * as i2 from "@angular/common";
|
|
@@ -22,6 +23,7 @@ export class PlaitBoardComponent {
|
|
|
22
23
|
this.plaitValue = [];
|
|
23
24
|
this.plaitPlugins = [];
|
|
24
25
|
this.plaitReadonly = false;
|
|
26
|
+
this.plaitAllowClearBoard = false;
|
|
25
27
|
this.plaitChange = new EventEmitter();
|
|
26
28
|
this.plaitBoardInitialized = new EventEmitter();
|
|
27
29
|
this.trackBy = (index, element) => {
|
|
@@ -59,8 +61,8 @@ export class PlaitBoardComponent {
|
|
|
59
61
|
this.plaitBoardInitialized.emit(this.board);
|
|
60
62
|
}
|
|
61
63
|
initializePlugins() {
|
|
62
|
-
const options = { readonly: this.plaitReadonly };
|
|
63
|
-
let board = withSelection(withBoard(createBoard(this.host, this.plaitValue, options)));
|
|
64
|
+
const options = { readonly: this.plaitReadonly, allowClearBoard: this.plaitAllowClearBoard };
|
|
65
|
+
let board = withHistroy(withSelection(withBoard(createBoard(this.host, this.plaitValue, options))));
|
|
64
66
|
this.plaitPlugins.forEach(plugin => {
|
|
65
67
|
board = plugin(board);
|
|
66
68
|
});
|
|
@@ -118,11 +120,14 @@ export class PlaitBoardComponent {
|
|
|
118
120
|
this.board?.keyup(event);
|
|
119
121
|
});
|
|
120
122
|
window.onresize = () => {
|
|
121
|
-
|
|
122
|
-
const viewBoxValues = this.host.getAttribute('viewBox')?.split(',');
|
|
123
|
-
this.renderer2.setAttribute(this.host, 'viewBox', `${viewBoxValues[0].trim()}, ${viewBoxValues[1].trim()}, ${viewBoxModel.width}, ${viewBoxModel.height}`);
|
|
123
|
+
this.refreshViewport();
|
|
124
124
|
};
|
|
125
125
|
}
|
|
126
|
+
refreshViewport() {
|
|
127
|
+
const viewBoxModel = getViewBox(this.board);
|
|
128
|
+
const viewBoxValues = this.host.getAttribute('viewBox')?.split(',');
|
|
129
|
+
this.renderer2.setAttribute(this.host, 'viewBox', `${viewBoxValues[0].trim()}, ${viewBoxValues[1].trim()}, ${viewBoxModel.width}, ${viewBoxModel.height}`);
|
|
130
|
+
}
|
|
126
131
|
updateViewport() {
|
|
127
132
|
this.zoom = Math.floor(this.board.viewport.zoom * 100);
|
|
128
133
|
const viewBox = getViewBox(this.board);
|
|
@@ -158,7 +163,7 @@ export class PlaitBoardComponent {
|
|
|
158
163
|
}
|
|
159
164
|
}
|
|
160
165
|
PlaitBoardComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: PlaitBoardComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component });
|
|
161
|
-
PlaitBoardComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.11", type: PlaitBoardComponent, selector: "plait-board", inputs: { plaitValue: "plaitValue", plaitViewport: "plaitViewport", plaitPlugins: "plaitPlugins", plaitReadonly: "plaitReadonly" }, outputs: { plaitChange: "plaitChange", plaitBoardInitialized: "plaitBoardInitialized" }, host: { properties: { "class": "this.hostClass" } }, viewQueries: [{ propertyName: "svg", first: true, predicate: ["svg"], descendants: true, static: true }], ngImport: i0, template: `
|
|
166
|
+
PlaitBoardComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.11", type: PlaitBoardComponent, selector: "plait-board", inputs: { plaitValue: "plaitValue", plaitViewport: "plaitViewport", plaitPlugins: "plaitPlugins", plaitReadonly: "plaitReadonly", plaitAllowClearBoard: "plaitAllowClearBoard" }, outputs: { plaitChange: "plaitChange", plaitBoardInitialized: "plaitBoardInitialized" }, host: { properties: { "class": "this.hostClass" } }, viewQueries: [{ propertyName: "svg", first: true, predicate: ["svg"], descendants: true, static: true }], ngImport: i0, template: `
|
|
162
167
|
<svg #svg width="100%" height="100%" style="position: relative"></svg>
|
|
163
168
|
<div *ngIf="isFocused" class="plait-toolbar island zoom-toolbar plait-board-attached">
|
|
164
169
|
<button class="item" (mousedown)="zoomOut($event)">-</button>
|
|
@@ -214,9 +219,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImpo
|
|
|
214
219
|
type: Input
|
|
215
220
|
}], plaitReadonly: [{
|
|
216
221
|
type: Input
|
|
222
|
+
}], plaitAllowClearBoard: [{
|
|
223
|
+
type: Input
|
|
217
224
|
}], plaitChange: [{
|
|
218
225
|
type: Output
|
|
219
226
|
}], plaitBoardInitialized: [{
|
|
220
227
|
type: Output
|
|
221
228
|
}] } });
|
|
222
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"board.component.js","sourceRoot":"","sources":["../../../../packages/plait/src/board/board.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterViewInit,\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    Renderer2,\n    ViewChild\n} from '@angular/core';\nimport { BOARD_TO_ON_CHANGE, HOST_TO_ROUGH_SVG, IS_TEXT_EDITABLE } from '../utils/weak-maps';\nimport { PlaitBoardChangeEvent, PlaitBoard, PlaitBoardOptions } from '../interfaces/board';\nimport { PlaitElement } from '../interfaces/element';\nimport { createBoard } from '../plugins/create-board';\nimport { withBoard } from '../plugins/with-board';\nimport { fromEvent, Subject } from 'rxjs';\nimport { filter, takeUntil } from 'rxjs/operators';\nimport { PlaitPlugin } from '../interfaces/plugin';\nimport { RoughSVG } from 'roughjs/bin/svg';\nimport rough from 'roughjs/bin/rough';\nimport { Transforms } from '../transfroms';\nimport { withSelection } from '../plugins/with-selection';\nimport { PlaitOperation } from '../interfaces/operation';\nimport { getViewBox, isNoSelectionElement } from '../utils/board';\nimport { Viewport } from '../interfaces/viewport';\n\n@Component({\n    selector: 'plait-board',\n    template: `\n        <svg #svg width=\"100%\" height=\"100%\" style=\"position: relative\"></svg>\n        <div *ngIf=\"isFocused\" class=\"plait-toolbar island zoom-toolbar plait-board-attached\">\n            <button class=\"item\" (mousedown)=\"zoomOut($event)\">-</button>\n            <button class=\"item zoom-value\" (mousedown)=\"resetZoom($event)\">{{ zoom }}%</button>\n            <button class=\"item\" (mousedown)=\"zoomIn($event)\">+</button>\n        </div>\n        <plait-element\n            *ngFor=\"let item of board.children; let index = index; trackBy: trackBy\"\n            [index]=\"index\"\n            [element]=\"item\"\n            [board]=\"board\"\n            [viewport]=\"board.viewport\"\n            [selection]=\"board.selection\"\n            [host]=\"host\"\n        ></plait-element>\n        <ng-content></ng-content>\n    `,\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PlaitBoardComponent implements OnInit, AfterViewInit, OnDestroy {\n    zoom = 100;\n\n    @HostBinding('class') hostClass = `plait-board-container`;\n\n    board!: PlaitBoard;\n\n    roughSVG!: RoughSVG;\n\n    destroy$: Subject<any> = new Subject();\n\n    @ViewChild('svg', { static: true })\n    svg!: ElementRef;\n\n    get host(): SVGElement {\n        return this.svg.nativeElement;\n    }\n\n    get isFocused() {\n        return this.board?.selection;\n    }\n\n    @Input() plaitValue: PlaitElement[] = [];\n\n    @Input() plaitViewport!: Viewport;\n\n    @Input() plaitPlugins: PlaitPlugin[] = [];\n\n    @Input() plaitReadonly = false;\n\n    @Output() plaitChange: EventEmitter<PlaitBoardChangeEvent> = new EventEmitter();\n\n    @Output() plaitBoardInitialized: EventEmitter<PlaitBoard> = new EventEmitter();\n\n    constructor(private cdr: ChangeDetectorRef, private renderer2: Renderer2) {}\n\n    ngOnInit(): void {\n        const roughSVG = rough.svg(this.host as SVGSVGElement, { options: { roughness: 0, strokeWidth: 1 } });\n        HOST_TO_ROUGH_SVG.set(this.host, roughSVG);\n        this.initializePlugins();\n        this.initializeEvents();\n        this.updateViewport();\n        BOARD_TO_ON_CHANGE.set(this.board, () => {\n            this.cdr.detectChanges();\n            const changeEvent: PlaitBoardChangeEvent = {\n                children: this.board.children,\n                operations: this.board.operations,\n                viewport: this.board.viewport,\n                selection: this.board.selection\n            };\n            // update viewBox\n            if (this.board.operations.some(op => PlaitOperation.isSetViewportOperation(op))) {\n                this.updateViewport();\n            }\n            this.plaitChange.emit(changeEvent);\n        });\n    }\n\n    ngAfterViewInit(): void {\n        this.plaitBoardInitialized.emit(this.board);\n    }\n\n    initializePlugins() {\n        const options: PlaitBoardOptions = { readonly: this.plaitReadonly };\n        let board = withSelection(withBoard(createBoard(this.host, this.plaitValue, options)));\n        this.plaitPlugins.forEach(plugin => {\n            board = plugin(board);\n        });\n        this.board = board;\n        if (this.plaitViewport) {\n            this.board.viewport = this.plaitViewport;\n        }\n    }\n\n    initializeEvents() {\n        fromEvent<MouseEvent>(this.host, 'mousedown')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: MouseEvent) => {\n                this.board.mousedown(event);\n            });\n\n        fromEvent<MouseEvent>(this.host, 'mousemove')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: MouseEvent) => {\n                this.board.mousemove(event);\n            });\n\n        fromEvent<MouseEvent>(document, 'mouseup')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: MouseEvent) => {\n                this.board.mouseup(event);\n            });\n\n        fromEvent<MouseEvent>(this.host, 'dblclick')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: MouseEvent) => {\n                this.board.dblclick(event);\n            });\n\n        fromEvent<WheelEvent>(this.host, 'wheel')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: WheelEvent) => {\n                if (this.isFocused) {\n                    event.preventDefault();\n                    const viewport = this.board.viewport;\n                    Transforms.setViewport(this.board, {\n                        ...viewport,\n                        offsetX: viewport?.offsetX - event.deltaX,\n                        offsetY: viewport?.offsetY - event.deltaY\n                    });\n                }\n            });\n\n        fromEvent<KeyboardEvent>(document, 'keydown')\n            .pipe(\n                takeUntil(this.destroy$),\n                filter(() => {\n                    return !IS_TEXT_EDITABLE.get(this.board) && !!this.board.selection;\n                })\n            )\n            .subscribe((event: KeyboardEvent) => {\n                this.board?.keydown(event);\n            });\n\n        fromEvent<KeyboardEvent>(document, 'keyup')\n            .pipe(\n                takeUntil(this.destroy$),\n                filter(() => {\n                    return !IS_TEXT_EDITABLE.get(this.board) && !!this.board.selection;\n                })\n            )\n            .subscribe((event: KeyboardEvent) => {\n                this.board?.keyup(event);\n            });\n\n        window.onresize = () => {\n            const viewBoxModel = getViewBox(this.board);\n            const viewBoxValues = this.host.getAttribute('viewBox')?.split(',') as string[];\n            this.renderer2.setAttribute(\n                this.host,\n                'viewBox',\n                `${viewBoxValues[0].trim()}, ${viewBoxValues[1].trim()}, ${viewBoxModel.width}, ${viewBoxModel.height}`\n            );\n        };\n    }\n\n    updateViewport() {\n        this.zoom = Math.floor(this.board.viewport.zoom * 100);\n        const viewBox = getViewBox(this.board);\n        this.renderer2.setAttribute(this.host, 'viewBox', `${viewBox.minX}, ${viewBox.minY}, ${viewBox.width}, ${viewBox.height}`);\n    }\n\n    // 放大\n    zoomIn(event: MouseEvent) {\n        const viewport = this.board?.viewport as Viewport;\n        Transforms.setViewport(this.board, {\n            ...viewport,\n            zoom: viewport.zoom + 0.1\n        });\n    }\n\n    // 缩小\n    zoomOut(event: MouseEvent) {\n        const viewport = this.board?.viewport as Viewport;\n        Transforms.setViewport(this.board, {\n            ...viewport,\n            zoom: viewport.zoom - 0.1\n        });\n    }\n\n    resetZoom(event: MouseEvent) {\n        const viewport = this.board?.viewport as Viewport;\n        Transforms.setViewport(this.board, {\n            ...viewport,\n            zoom: 1\n        });\n    }\n\n    trackBy = (index: number, element: PlaitElement) => {\n        return index;\n    };\n\n    ngOnDestroy(): void {\n        this.destroy$.next();\n        this.destroy$.complete();\n        HOST_TO_ROUGH_SVG.delete(this.host);\n    }\n}\n"]}
|
|
229
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"board.component.js","sourceRoot":"","sources":["../../../../packages/plait/src/board/board.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    AfterViewInit,\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    Renderer2,\n    ViewChild\n} from '@angular/core';\nimport { BOARD_TO_ON_CHANGE, HOST_TO_ROUGH_SVG, IS_TEXT_EDITABLE } from '../utils/weak-maps';\nimport { PlaitBoardChangeEvent, PlaitBoard, PlaitBoardOptions } from '../interfaces/board';\nimport { PlaitElement } from '../interfaces/element';\nimport { createBoard } from '../plugins/create-board';\nimport { withBoard } from '../plugins/with-board';\nimport { fromEvent, Subject } from 'rxjs';\nimport { filter, takeUntil } from 'rxjs/operators';\nimport { PlaitPlugin } from '../interfaces/plugin';\nimport { RoughSVG } from 'roughjs/bin/svg';\nimport rough from 'roughjs/bin/rough';\nimport { Transforms } from '../transfroms';\nimport { withSelection } from '../plugins/with-selection';\nimport { PlaitOperation } from '../interfaces/operation';\nimport { getViewBox, isNoSelectionElement } from '../utils/board';\nimport { Viewport } from '../interfaces/viewport';\nimport { withHistroy } from '../plugins/with-history';\n\n@Component({\n    selector: 'plait-board',\n    template: `\n        <svg #svg width=\"100%\" height=\"100%\" style=\"position: relative\"></svg>\n        <div *ngIf=\"isFocused\" class=\"plait-toolbar island zoom-toolbar plait-board-attached\">\n            <button class=\"item\" (mousedown)=\"zoomOut($event)\">-</button>\n            <button class=\"item zoom-value\" (mousedown)=\"resetZoom($event)\">{{ zoom }}%</button>\n            <button class=\"item\" (mousedown)=\"zoomIn($event)\">+</button>\n        </div>\n        <plait-element\n            *ngFor=\"let item of board.children; let index = index; trackBy: trackBy\"\n            [index]=\"index\"\n            [element]=\"item\"\n            [board]=\"board\"\n            [viewport]=\"board.viewport\"\n            [selection]=\"board.selection\"\n            [host]=\"host\"\n        ></plait-element>\n        <ng-content></ng-content>\n    `,\n    changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class PlaitBoardComponent implements OnInit, AfterViewInit, OnDestroy {\n    zoom = 100;\n\n    @HostBinding('class') hostClass = `plait-board-container`;\n\n    board!: PlaitBoard;\n\n    roughSVG!: RoughSVG;\n\n    destroy$: Subject<any> = new Subject();\n\n    @ViewChild('svg', { static: true })\n    svg!: ElementRef;\n\n    get host(): SVGElement {\n        return this.svg.nativeElement;\n    }\n\n    get isFocused() {\n        return this.board?.selection;\n    }\n\n    @Input() plaitValue: PlaitElement[] = [];\n\n    @Input() plaitViewport!: Viewport;\n\n    @Input() plaitPlugins: PlaitPlugin[] = [];\n\n    @Input() plaitReadonly = false;\n\n    @Input() plaitAllowClearBoard = false;\n\n    @Output() plaitChange: EventEmitter<PlaitBoardChangeEvent> = new EventEmitter();\n\n    @Output() plaitBoardInitialized: EventEmitter<PlaitBoard> = new EventEmitter();\n\n    constructor(private cdr: ChangeDetectorRef, private renderer2: Renderer2) {}\n\n    ngOnInit(): void {\n        const roughSVG = rough.svg(this.host as SVGSVGElement, { options: { roughness: 0, strokeWidth: 1 } });\n        HOST_TO_ROUGH_SVG.set(this.host, roughSVG);\n        this.initializePlugins();\n        this.initializeEvents();\n        this.updateViewport();\n        BOARD_TO_ON_CHANGE.set(this.board, () => {\n            this.cdr.detectChanges();\n            const changeEvent: PlaitBoardChangeEvent = {\n                children: this.board.children,\n                operations: this.board.operations,\n                viewport: this.board.viewport,\n                selection: this.board.selection\n            };\n            // update viewBox\n            if (this.board.operations.some(op => PlaitOperation.isSetViewportOperation(op))) {\n                this.updateViewport();\n            }\n            this.plaitChange.emit(changeEvent);\n        });\n    }\n\n    ngAfterViewInit(): void {\n        this.plaitBoardInitialized.emit(this.board);\n    }\n\n    initializePlugins() {\n        const options: PlaitBoardOptions = { readonly: this.plaitReadonly, allowClearBoard: this.plaitAllowClearBoard };\n        let board = withHistroy(withSelection(withBoard(createBoard(this.host, this.plaitValue, options))));\n        this.plaitPlugins.forEach(plugin => {\n            board = plugin(board);\n        });\n        this.board = board;\n        if (this.plaitViewport) {\n            this.board.viewport = this.plaitViewport;\n        }\n    }\n\n    initializeEvents() {\n        fromEvent<MouseEvent>(this.host, 'mousedown')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: MouseEvent) => {\n                this.board.mousedown(event);\n            });\n\n        fromEvent<MouseEvent>(this.host, 'mousemove')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: MouseEvent) => {\n                this.board.mousemove(event);\n            });\n\n        fromEvent<MouseEvent>(document, 'mouseup')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: MouseEvent) => {\n                this.board.mouseup(event);\n            });\n\n        fromEvent<MouseEvent>(this.host, 'dblclick')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: MouseEvent) => {\n                this.board.dblclick(event);\n            });\n\n        fromEvent<WheelEvent>(this.host, 'wheel')\n            .pipe(takeUntil(this.destroy$))\n            .subscribe((event: WheelEvent) => {\n                if (this.isFocused) {\n                    event.preventDefault();\n                    const viewport = this.board.viewport;\n                    Transforms.setViewport(this.board, {\n                        ...viewport,\n                        offsetX: viewport?.offsetX - event.deltaX,\n                        offsetY: viewport?.offsetY - event.deltaY\n                    });\n                }\n            });\n\n        fromEvent<KeyboardEvent>(document, 'keydown')\n            .pipe(\n                takeUntil(this.destroy$),\n                filter(() => {\n                    return !IS_TEXT_EDITABLE.get(this.board) && !!this.board.selection;\n                })\n            )\n            .subscribe((event: KeyboardEvent) => {\n                this.board?.keydown(event);\n            });\n\n        fromEvent<KeyboardEvent>(document, 'keyup')\n            .pipe(\n                takeUntil(this.destroy$),\n                filter(() => {\n                    return !IS_TEXT_EDITABLE.get(this.board) && !!this.board.selection;\n                })\n            )\n            .subscribe((event: KeyboardEvent) => {\n                this.board?.keyup(event);\n            });\n\n        window.onresize = () => {\n            this.refreshViewport();\n        };\n    }\n\n    refreshViewport() {\n        const viewBoxModel = getViewBox(this.board);\n        const viewBoxValues = this.host.getAttribute('viewBox')?.split(',') as string[];\n        this.renderer2.setAttribute(\n            this.host,\n            'viewBox',\n            `${viewBoxValues[0].trim()}, ${viewBoxValues[1].trim()}, ${viewBoxModel.width}, ${viewBoxModel.height}`\n        );\n    }\n\n    updateViewport() {\n        this.zoom = Math.floor(this.board.viewport.zoom * 100);\n        const viewBox = getViewBox(this.board);\n        this.renderer2.setAttribute(this.host, 'viewBox', `${viewBox.minX}, ${viewBox.minY}, ${viewBox.width}, ${viewBox.height}`);\n    }\n\n    // 放大\n    zoomIn(event: MouseEvent) {\n        const viewport = this.board?.viewport as Viewport;\n        Transforms.setViewport(this.board, {\n            ...viewport,\n            zoom: viewport.zoom + 0.1\n        });\n    }\n\n    // 缩小\n    zoomOut(event: MouseEvent) {\n        const viewport = this.board?.viewport as Viewport;\n        Transforms.setViewport(this.board, {\n            ...viewport,\n            zoom: viewport.zoom - 0.1\n        });\n    }\n\n    resetZoom(event: MouseEvent) {\n        const viewport = this.board?.viewport as Viewport;\n        Transforms.setViewport(this.board, {\n            ...viewport,\n            zoom: 1\n        });\n    }\n\n    trackBy = (index: number, element: PlaitElement) => {\n        return index;\n    };\n\n    ngOnDestroy(): void {\n        this.destroy$.next();\n        this.destroy$.complete();\n        HOST_TO_ROUGH_SVG.delete(this.host);\n    }\n}\n"]}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYm9hcmQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9wbGFpdC9zcmMvaW50ZXJmYWNlcy9ib2FyZC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgU2ltcGxlQ2hhbmdlcyB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ3Vyc29yU3RhdHVzIH0gZnJvbSAnLi9jdXJzb3InO1xuaW1wb3J0IHsgUGxhaXRFbGVtZW50IH0gZnJvbSAnLi9lbGVtZW50JztcbmltcG9ydCB7IFBsYWl0RWxlbWVudENvbnRleHQgfSBmcm9tICcuL2VsZW1lbnQtY29udGV4dCc7XG5pbXBvcnQgeyBQbGFpdEhpc3RvcnkgfSBmcm9tICcuL2hpc3RvcnknO1xuaW1wb3J0IHsgUGxhaXRPcGVyYXRpb24gfSBmcm9tICcuL29wZXJhdGlvbic7XG5pbXBvcnQgeyBTZWxlY3Rpb24gfSBmcm9tICcuL3NlbGVjdGlvbic7XG5pbXBvcnQgeyBWaWV3cG9ydCB9IGZyb20gJy4vdmlld3BvcnQnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFBsYWl0Qm9hcmQge1xuICAgIGhvc3Q6IFNWR0VsZW1lbnQ7XG4gICAgdmlld3BvcnQ6IFZpZXdwb3J0O1xuICAgIGNoaWxkcmVuOiBQbGFpdEVsZW1lbnRbXTtcbiAgICBvcGVyYXRpb25zOiBQbGFpdE9wZXJhdGlvbltdO1xuICAgIHNlbGVjdGlvbjogU2VsZWN0aW9uIHwgbnVsbDtcbiAgICBjdXJzb3I6IEN1cnNvclN0YXR1cztcbiAgICByZWFkb25seTogYm9vbGVhbjtcbiAgICBhbGxvd0NsZWFyQm9hcmQ6IGJvb2xlYW47XG4gICAgaGlzdG9yeTogUGxhaXRIaXN0b3J5O1xuICAgIHVuZG86ICgpID0+IHZvaWQ7XG4gICAgcmVkbzogKCkgPT4gdm9pZDtcbiAgICBhcHBseTogKG9wZXJhdGlvbjogUGxhaXRPcGVyYXRpb24pID0+IHZvaWQ7XG4gICAgb25DaGFuZ2U6ICgpID0+IHZvaWQ7XG4gICAgbW91c2Vkb3duOiAoZXZlbnQ6IE1vdXNlRXZlbnQpID0+IHZvaWQ7XG4gICAgbW91c2V1cDogKGV2ZW50OiBNb3VzZUV2ZW50KSA9PiB2b2lkO1xuICAgIG1vdXNlbW92ZTogKGV2ZW50OiBNb3VzZUV2ZW50KSA9PiB2b2lkO1xuICAgIGtleWRvd246IChldmVudDogS2V5Ym9hcmRFdmVudCkgPT4gdm9pZDtcbiAgICBrZXl1cDogKGV2ZW50OiBLZXlib2FyZEV2ZW50KSA9PiB2b2lkO1xuICAgIGRibGNsaWNrOiAoZXZlbnQ6IE1vdXNlRXZlbnQpID0+IHZvaWQ7XG4gICAgZHJhd0VsZW1lbnQ6IChjb250ZXh0OiBQbGFpdEVsZW1lbnRDb250ZXh0KSA9PiBTVkdHRWxlbWVudFtdO1xuICAgIHJlZHJhd0VsZW1lbnQ6IChjb250ZXh0OiBQbGFpdEVsZW1lbnRDb250ZXh0LCBjaGFuZ2VzOiBTaW1wbGVDaGFuZ2VzKSA9PiBTVkdHRWxlbWVudFtdO1xuICAgIGRlc3Ryb3lFbGVtZW50OiAoKSA9PiB2b2lkO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFBsYWl0Qm9hcmRDaGFuZ2VFdmVudCB7XG4gICAgY2hpbGRyZW46IFBsYWl0RWxlbWVudFtdO1xuICAgIG9wZXJhdGlvbnM6IFBsYWl0T3BlcmF0aW9uW107XG4gICAgdmlld3BvcnQ6IFZpZXdwb3J0O1xuICAgIHNlbGVjdGlvbjogU2VsZWN0aW9uIHwgbnVsbDtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQbGFpdEJvYXJkT3B0aW9ucyB7XG4gICAgcmVhZG9ubHk6IGJvb2xlYW47XG4gICAgYWxsb3dDbGVhckJvYXJkOiBib29sZWFuO1xufVxuIl19
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGlzdG9yeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL3BsYWl0L3NyYy9pbnRlcmZhY2VzL2hpc3RvcnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBsYWl0T3BlcmF0aW9uIH0gZnJvbSAnLi9vcGVyYXRpb24nO1xuXG5leHBvcnQgaW50ZXJmYWNlIFBsYWl0SGlzdG9yeSB7XG4gICAgcmVkb3M6IFBsYWl0T3BlcmF0aW9uW11bXTtcbiAgICB1bmRvczogUGxhaXRPcGVyYXRpb25bXVtdO1xufVxuIl19
|
|
@@ -11,4 +11,5 @@ export * from './plugin';
|
|
|
11
11
|
export * from './point';
|
|
12
12
|
export * from './selection';
|
|
13
13
|
export * from './viewport';
|
|
14
|
-
|
|
14
|
+
export * from './history';
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9wbGFpdC9zcmMvaW50ZXJmYWNlcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLFNBQVMsQ0FBQztBQUN4QixjQUFjLFVBQVUsQ0FBQztBQUN6QixjQUFjLG1CQUFtQixDQUFDO0FBQ2xDLGNBQWMsZ0JBQWdCLENBQUM7QUFDL0IsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxZQUFZLENBQUM7QUFDM0IsY0FBYyxXQUFXLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2JvYXJkJztcbmV4cG9ydCAqIGZyb20gJy4vY3Vyc29yJztcbmV4cG9ydCAqIGZyb20gJy4vZWxlbWVudC1jb250ZXh0JztcbmV4cG9ydCAqIGZyb20gJy4vY3VzdG9tLXR5cGVzJztcbmV4cG9ydCAqIGZyb20gJy4vZWxlbWVudCc7XG5leHBvcnQgKiBmcm9tICcuL2dyYXBoJztcbmV4cG9ydCAqIGZyb20gJy4vbm9kZSc7XG5leHBvcnQgKiBmcm9tICcuL29wZXJhdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL3BhdGgnO1xuZXhwb3J0ICogZnJvbSAnLi9wbHVnaW4nO1xuZXhwb3J0ICogZnJvbSAnLi9wb2ludCc7XG5leHBvcnQgKiBmcm9tICcuL3NlbGVjdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL3ZpZXdwb3J0JztcbmV4cG9ydCAqIGZyb20gJy4vaGlzdG9yeSc7XG4iXX0=
|
|
@@ -1,7 +1,84 @@
|
|
|
1
|
+
import { Path } from './path';
|
|
1
2
|
export const isSetViewportOperation = (value) => {
|
|
2
3
|
return value.type === 'set_viewport';
|
|
3
4
|
};
|
|
5
|
+
export const inverse = (op) => {
|
|
6
|
+
switch (op.type) {
|
|
7
|
+
case 'insert_node': {
|
|
8
|
+
return { ...op, type: 'remove_node' };
|
|
9
|
+
}
|
|
10
|
+
case 'remove_node': {
|
|
11
|
+
return { ...op, type: 'insert_node' };
|
|
12
|
+
}
|
|
13
|
+
case 'move_node': {
|
|
14
|
+
const { newPath, path } = op;
|
|
15
|
+
// PERF: in this case the move operation is a no-op anyways.
|
|
16
|
+
if (Path.equals(newPath, path)) {
|
|
17
|
+
return op;
|
|
18
|
+
}
|
|
19
|
+
// If the move happens completely within a single parent the path and
|
|
20
|
+
// newPath are stable with respect to each other.
|
|
21
|
+
if (Path.isSibling(path, newPath)) {
|
|
22
|
+
return { ...op, path: newPath, newPath: path };
|
|
23
|
+
}
|
|
24
|
+
// If the move does not happen within a single parent it is possible
|
|
25
|
+
// for the move to impact the true path to the location where the node
|
|
26
|
+
// was removed from and where it was inserted. We have to adjust for this
|
|
27
|
+
// and find the original path. We can accomplish this (only in non-sibling)
|
|
28
|
+
// moves by looking at the impact of the move operation on the node
|
|
29
|
+
// after the original move path.
|
|
30
|
+
const inversePath = Path.transform(path, op);
|
|
31
|
+
const inverseNewPath = Path.transform(Path.next(path), op);
|
|
32
|
+
return { ...op, path: inversePath, newPath: inverseNewPath };
|
|
33
|
+
}
|
|
34
|
+
case 'set_node': {
|
|
35
|
+
const { properties, newProperties } = op;
|
|
36
|
+
return { ...op, properties: newProperties, newProperties: properties };
|
|
37
|
+
}
|
|
38
|
+
case 'set_selection': {
|
|
39
|
+
const { properties, newProperties } = op;
|
|
40
|
+
if (properties == null) {
|
|
41
|
+
return {
|
|
42
|
+
...op,
|
|
43
|
+
properties: newProperties,
|
|
44
|
+
newProperties: null
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
else if (newProperties == null) {
|
|
48
|
+
return {
|
|
49
|
+
...op,
|
|
50
|
+
properties: null,
|
|
51
|
+
newProperties: properties
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
else {
|
|
55
|
+
return { ...op, properties: newProperties, newProperties: properties };
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
case 'set_viewport': {
|
|
59
|
+
const { properties, newProperties } = op;
|
|
60
|
+
if (properties == null) {
|
|
61
|
+
return {
|
|
62
|
+
...op,
|
|
63
|
+
properties: newProperties,
|
|
64
|
+
newProperties: newProperties
|
|
65
|
+
};
|
|
66
|
+
}
|
|
67
|
+
else if (newProperties == null) {
|
|
68
|
+
return {
|
|
69
|
+
...op,
|
|
70
|
+
properties: properties,
|
|
71
|
+
newProperties: properties
|
|
72
|
+
};
|
|
73
|
+
}
|
|
74
|
+
else {
|
|
75
|
+
return { ...op, properties: newProperties, newProperties: properties };
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
};
|
|
4
80
|
export const PlaitOperation = {
|
|
5
|
-
isSetViewportOperation
|
|
81
|
+
isSetViewportOperation,
|
|
82
|
+
inverse
|
|
6
83
|
};
|
|
7
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
84
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"operation.js","sourceRoot":"","sources":["../../../../packages/plait/src/interfaces/operation.ts"],"names":[],"mappings":"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","sourcesContent":["import { PlaitNode } from './node';\nimport { Path } from './path';\nimport { Selection } from './selection';\nimport { Viewport } from './viewport';\n\nexport type InsertNodeOperation = {\n    type: 'insert_node';\n    path: Path;\n    node: PlaitNode;\n};\n\nexport type RemoveNodeOperation = {\n    type: 'remove_node';\n    path: Path;\n    node: PlaitNode;\n};\n\nexport type MoveNodeOperation = {\n    type: 'move_node';\n    path: Path;\n    newPath: Path;\n};\n\nexport type SetViewportOperation = {\n    type: 'set_viewport';\n    properties: Partial<Viewport>;\n    newProperties: Partial<Viewport>;\n};\n\nexport type SetSelectionOperation = {\n    type: 'set_selection';\n    properties: Selection | null;\n    newProperties: Selection | null;\n};\n\nexport type SetNodeOperation = {\n    type: 'set_node';\n    path: Path;\n    properties: Partial<PlaitNode>;\n    newProperties: Partial<PlaitNode>;\n};\n\nexport type PlaitOperation =\n    | InsertNodeOperation\n    | SetViewportOperation\n    | SetSelectionOperation\n    | SetNodeOperation\n    | RemoveNodeOperation\n    | MoveNodeOperation;\n\nexport interface PlaitOperationInterface {\n    inverse: (op: PlaitOperation) => PlaitOperation;\n    isSetViewportOperation: (value: any) => boolean;\n}\n\nexport const isSetViewportOperation = (value: any): value is SetViewportOperation => {\n    return value.type === 'set_viewport';\n};\n\nexport const inverse = (op: PlaitOperation): PlaitOperation => {\n    switch (op.type) {\n        case 'insert_node': {\n            return { ...op, type: 'remove_node' };\n        }\n\n        case 'remove_node': {\n            return { ...op, type: 'insert_node' };\n        }\n\n        case 'move_node': {\n            const { newPath, path } = op;\n\n            // PERF: in this case the move operation is a no-op anyways.\n            if (Path.equals(newPath, path)) {\n                return op;\n            }\n\n            // If the move happens completely within a single parent the path and\n            // newPath are stable with respect to each other.\n            if (Path.isSibling(path, newPath)) {\n                return { ...op, path: newPath, newPath: path };\n            }\n\n            // If the move does not happen within a single parent it is possible\n            // for the move to impact the true path to the location where the node\n            // was removed from and where it was inserted. We have to adjust for this\n            // and find the original path. We can accomplish this (only in non-sibling)\n            // moves by looking at the impact of the move operation on the node\n            // after the original move path.\n            const inversePath = Path.transform(path, op)!;\n            const inverseNewPath = Path.transform(Path.next(path), op)!;\n            return { ...op, path: inversePath, newPath: inverseNewPath };\n        }\n\n        case 'set_node': {\n            const { properties, newProperties } = op;\n            return { ...op, properties: newProperties, newProperties: properties };\n        }\n\n        case 'set_selection': {\n            const { properties, newProperties } = op;\n\n            if (properties == null) {\n                return {\n                    ...op,\n                    properties: newProperties,\n                    newProperties: null\n                };\n            } else if (newProperties == null) {\n                return {\n                    ...op,\n                    properties: null,\n                    newProperties: properties\n                };\n            } else {\n                return { ...op, properties: newProperties, newProperties: properties };\n            }\n        }\n\n        case 'set_viewport': {\n            const { properties, newProperties } = op;\n            if (properties == null) {\n                return {\n                    ...op,\n                    properties: newProperties,\n                    newProperties: newProperties\n                };\n            } else if (newProperties == null) {\n                return {\n                    ...op,\n                    properties: properties,\n                    newProperties: properties\n                };\n            } else {\n                return { ...op, properties: newProperties, newProperties: properties };\n            }\n        }\n    }\n};\n\nexport const PlaitOperation: PlaitOperationInterface = {\n    isSetViewportOperation,\n    inverse\n};\n"]}
|
|
@@ -12,9 +12,16 @@ export function createBoard(host, children, options) {
|
|
|
12
12
|
},
|
|
13
13
|
children,
|
|
14
14
|
operations: [],
|
|
15
|
+
history: {
|
|
16
|
+
redos: [],
|
|
17
|
+
undos: []
|
|
18
|
+
},
|
|
15
19
|
selection: null,
|
|
16
20
|
cursor: BaseCursorStatus.select,
|
|
17
21
|
readonly: options.readonly,
|
|
22
|
+
allowClearBoard: options.allowClearBoard,
|
|
23
|
+
undo: () => { },
|
|
24
|
+
redo: () => { },
|
|
18
25
|
apply: (operation) => {
|
|
19
26
|
board.operations.push(operation);
|
|
20
27
|
Transforms.transform(board, operation);
|
|
@@ -40,4 +47,4 @@ export function createBoard(host, children, options) {
|
|
|
40
47
|
};
|
|
41
48
|
return board;
|
|
42
49
|
}
|
|
43
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
50
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY3JlYXRlLWJvYXJkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcGFja2FnZXMvcGxhaXQvc3JjL3BsdWdpbnMvY3JlYXRlLWJvYXJkLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUtBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUM5QyxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNDLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBRXhELE1BQU0sVUFBVSxXQUFXLENBQUMsSUFBZ0IsRUFBRSxRQUF3QixFQUFFLE9BQTBCO0lBQzlGLE1BQU0sS0FBSyxHQUFlO1FBQ3RCLElBQUk7UUFDSixRQUFRLEVBQUU7WUFDTixPQUFPLEVBQUUsQ0FBQztZQUNWLE9BQU8sRUFBRSxDQUFDO1lBQ1YsSUFBSSxFQUFFLENBQUM7WUFDUCxtQkFBbUIsRUFBRSxNQUFNO1NBQzlCO1FBQ0QsUUFBUTtRQUNSLFVBQVUsRUFBRSxFQUFFO1FBQ2QsT0FBTyxFQUFFO1lBQ0wsS0FBSyxFQUFFLEVBQUU7WUFDVCxLQUFLLEVBQUUsRUFBRTtTQUNaO1FBQ0QsU0FBUyxFQUFFLElBQUk7UUFDZixNQUFNLEVBQUUsZ0JBQWdCLENBQUMsTUFBTTtRQUMvQixRQUFRLEVBQUUsT0FBTyxDQUFDLFFBQVE7UUFDMUIsZUFBZSxFQUFFLE9BQU8sQ0FBQyxlQUFlO1FBQ3hDLElBQUksRUFBRSxHQUFHLEVBQUUsR0FBRSxDQUFDO1FBQ2QsSUFBSSxFQUFFLEdBQUcsRUFBRSxHQUFFLENBQUM7UUFDZCxLQUFLLEVBQUUsQ0FBQyxTQUF5QixFQUFFLEVBQUU7WUFDakMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7WUFFakMsVUFBVSxDQUFDLFNBQVMsQ0FBQyxLQUFLLEVBQUUsU0FBUyxDQUFDLENBQUM7WUFFdkMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUU7Z0JBQ3RCLFFBQVEsQ0FBQyxHQUFHLENBQUMsS0FBSyxFQUFFLElBQUksQ0FBQyxDQUFDO2dCQUUxQixPQUFPLENBQUMsT0FBTyxFQUFFLENBQUMsSUFBSSxDQUFDLEdBQUcsRUFBRTtvQkFDeEIsUUFBUSxDQUFDLEdBQUcsQ0FBQyxLQUFLLEVBQUUsS0FBSyxDQUFDLENBQUM7b0JBQzNCLEtBQUssQ0FBQyxRQUFRLEVBQUUsQ0FBQztvQkFDakIsS0FBSyxDQUFDLFVBQVUsR0FBRyxFQUFFLENBQUM7Z0JBQzFCLENBQUMsQ0FBQyxDQUFDO2FBQ047UUFDTCxDQUFDO1FBQ0QsUUFBUSxFQUFFLEdBQUcsRUFBRSxHQUFFLENBQUM7UUFDbEIsU0FBUyxFQUFFLENBQUMsS0FBaUIsRUFBRSxFQUFFLEdBQUUsQ0FBQztRQUNwQyxPQUFPLEVBQUUsQ0FBQyxLQUFpQixFQUFFLEVBQUUsR0FBRSxDQUFDO1FBQ2xDLFNBQVMsRUFBRSxDQUFDLEtBQWlCLEVBQUUsRUFBRSxHQUFFLENBQUM7UUFDcEMsT0FBTyxFQUFFLENBQUMsS0FBb0IsRUFBRSxFQUFFLEdBQUUsQ0FBQztRQUNyQyxLQUFLLEVBQUUsQ0FBQyxLQUFvQixFQUFFLEVBQUUsR0FBRSxDQUFDO1FBQ25DLFFBQVEsRUFBRSxDQUFDLEtBQWlCLEVBQUUsRUFBRSxHQUFFLENBQUM7UUFDbkMsV0FBVyxFQUFFLENBQUMsT0FBNEIsRUFBRSxFQUFFLENBQUMsRUFBRTtRQUNqRCxhQUFhLEVBQUUsQ0FBQyxPQUE0QixFQUFFLE9BQXNCLEVBQUUsRUFBRSxDQUFDLEVBQUU7UUFDM0UsY0FBYyxFQUFFLEdBQUcsRUFBRSxHQUFFLENBQUM7S0FDM0IsQ0FBQztJQUNGLE9BQU8sS0FBSyxDQUFDO0FBQ2pCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBQbGFpdEVsZW1lbnQgfSBmcm9tICcuLi9pbnRlcmZhY2VzL2VsZW1lbnQnO1xuaW1wb3J0IHsgUGxhaXRPcGVyYXRpb24gfSBmcm9tICcuLi9pbnRlcmZhY2VzL29wZXJhdGlvbic7XG5pbXBvcnQgeyBQbGFpdEJvYXJkLCBQbGFpdEJvYXJkT3B0aW9ucyB9IGZyb20gJy4uL2ludGVyZmFjZXMvYm9hcmQnO1xuaW1wb3J0IHsgUGxhaXRFbGVtZW50Q29udGV4dCB9IGZyb20gJy4uL2ludGVyZmFjZXMvZWxlbWVudC1jb250ZXh0JztcbmltcG9ydCB7IFNpbXBsZUNoYW5nZXMgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEZMVVNISU5HIH0gZnJvbSAnLi4vdXRpbHMvd2Vhay1tYXBzJztcbmltcG9ydCB7IFRyYW5zZm9ybXMgfSBmcm9tICcuLi90cmFuc2Zyb21zJztcbmltcG9ydCB7IEJhc2VDdXJzb3JTdGF0dXMgfSBmcm9tICcuLi9pbnRlcmZhY2VzL2N1cnNvcic7XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVCb2FyZChob3N0OiBTVkdFbGVtZW50LCBjaGlsZHJlbjogUGxhaXRFbGVtZW50W10sIG9wdGlvbnM6IFBsYWl0Qm9hcmRPcHRpb25zKTogUGxhaXRCb2FyZCB7XG4gICAgY29uc3QgYm9hcmQ6IFBsYWl0Qm9hcmQgPSB7XG4gICAgICAgIGhvc3QsXG4gICAgICAgIHZpZXdwb3J0OiB7XG4gICAgICAgICAgICBvZmZzZXRYOiAwLFxuICAgICAgICAgICAgb2Zmc2V0WTogMCxcbiAgICAgICAgICAgIHpvb206IDEsXG4gICAgICAgICAgICB2aWV3QmFja2dyb3VuZENvbG9yOiAnIzAwMCdcbiAgICAgICAgfSxcbiAgICAgICAgY2hpbGRyZW4sXG4gICAgICAgIG9wZXJhdGlvbnM6IFtdLFxuICAgICAgICBoaXN0b3J5OiB7XG4gICAgICAgICAgICByZWRvczogW10sXG4gICAgICAgICAgICB1bmRvczogW11cbiAgICAgICAgfSxcbiAgICAgICAgc2VsZWN0aW9uOiBudWxsLFxuICAgICAgICBjdXJzb3I6IEJhc2VDdXJzb3JTdGF0dXMuc2VsZWN0LFxuICAgICAgICByZWFkb25seTogb3B0aW9ucy5yZWFkb25seSxcbiAgICAgICAgYWxsb3dDbGVhckJvYXJkOiBvcHRpb25zLmFsbG93Q2xlYXJCb2FyZCxcbiAgICAgICAgdW5kbzogKCkgPT4ge30sXG4gICAgICAgIHJlZG86ICgpID0+IHt9LFxuICAgICAgICBhcHBseTogKG9wZXJhdGlvbjogUGxhaXRPcGVyYXRpb24pID0+IHtcbiAgICAgICAgICAgIGJvYXJkLm9wZXJhdGlvbnMucHVzaChvcGVyYXRpb24pO1xuXG4gICAgICAgICAgICBUcmFuc2Zvcm1zLnRyYW5zZm9ybShib2FyZCwgb3BlcmF0aW9uKTtcblxuICAgICAgICAgICAgaWYgKCFGTFVTSElORy5nZXQoYm9hcmQpKSB7XG4gICAgICAgICAgICAgICAgRkxVU0hJTkcuc2V0KGJvYXJkLCB0cnVlKTtcblxuICAgICAgICAgICAgICAgIFByb21pc2UucmVzb2x2ZSgpLnRoZW4oKCkgPT4ge1xuICAgICAgICAgICAgICAgICAgICBGTFVTSElORy5zZXQoYm9hcmQsIGZhbHNlKTtcbiAgICAgICAgICAgICAgICAgICAgYm9hcmQub25DaGFuZ2UoKTtcbiAgICAgICAgICAgICAgICAgICAgYm9hcmQub3BlcmF0aW9ucyA9IFtdO1xuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfVxuICAgICAgICB9LFxuICAgICAgICBvbkNoYW5nZTogKCkgPT4ge30sXG4gICAgICAgIG1vdXNlZG93bjogKGV2ZW50OiBNb3VzZUV2ZW50KSA9PiB7fSxcbiAgICAgICAgbW91c2V1cDogKGV2ZW50OiBNb3VzZUV2ZW50KSA9PiB7fSxcbiAgICAgICAgbW91c2Vtb3ZlOiAoZXZlbnQ6IE1vdXNlRXZlbnQpID0+IHt9LFxuICAgICAgICBrZXlkb3duOiAoZXZlbnQ6IEtleWJvYXJkRXZlbnQpID0+IHt9LFxuICAgICAgICBrZXl1cDogKGV2ZW50OiBLZXlib2FyZEV2ZW50KSA9PiB7fSxcbiAgICAgICAgZGJsY2xpY2s6IChldmVudDogTW91c2VFdmVudCkgPT4ge30sXG4gICAgICAgIGRyYXdFbGVtZW50OiAoY29udGV4dDogUGxhaXRFbGVtZW50Q29udGV4dCkgPT4gW10sXG4gICAgICAgIHJlZHJhd0VsZW1lbnQ6IChjb250ZXh0OiBQbGFpdEVsZW1lbnRDb250ZXh0LCBjaGFuZ2VzOiBTaW1wbGVDaGFuZ2VzKSA9PiBbXSxcbiAgICAgICAgZGVzdHJveUVsZW1lbnQ6ICgpID0+IHt9XG4gICAgfTtcbiAgICByZXR1cm4gYm9hcmQ7XG59XG4iXX0=
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import { PlaitOperation } from '../interfaces';
|
|
2
|
+
import { isHotkey } from 'is-hotkey';
|
|
3
|
+
import { shouldClear, shouldMerge, shouldSave } from '../utils';
|
|
4
|
+
export function withHistroy(board) {
|
|
5
|
+
const { apply, keydown } = board;
|
|
6
|
+
board.history = { undos: [], redos: [] };
|
|
7
|
+
board.redo = () => {
|
|
8
|
+
const { history } = board;
|
|
9
|
+
const { redos } = history;
|
|
10
|
+
if (redos.length > 0) {
|
|
11
|
+
const batch = redos[redos.length - 1];
|
|
12
|
+
PlaitHistoryBoard.withoutSaving(board, () => {
|
|
13
|
+
for (const op of batch) {
|
|
14
|
+
board.apply(op);
|
|
15
|
+
}
|
|
16
|
+
});
|
|
17
|
+
history.redos.pop();
|
|
18
|
+
history.undos.push(batch);
|
|
19
|
+
}
|
|
20
|
+
};
|
|
21
|
+
board.undo = () => {
|
|
22
|
+
const { history } = board;
|
|
23
|
+
const { undos } = history;
|
|
24
|
+
if (undos.length > 0) {
|
|
25
|
+
const batch = undos[undos.length - 1];
|
|
26
|
+
PlaitHistoryBoard.withoutSaving(board, () => {
|
|
27
|
+
const inverseOps = batch.map(PlaitOperation.inverse).reverse();
|
|
28
|
+
for (const op of inverseOps) {
|
|
29
|
+
board.apply(op);
|
|
30
|
+
}
|
|
31
|
+
});
|
|
32
|
+
history.redos.push(batch);
|
|
33
|
+
history.undos.pop();
|
|
34
|
+
}
|
|
35
|
+
};
|
|
36
|
+
board.apply = (op) => {
|
|
37
|
+
const { operations, history } = board;
|
|
38
|
+
const { undos } = history;
|
|
39
|
+
const lastBatch = undos[undos.length - 1];
|
|
40
|
+
const lastOp = lastBatch && lastBatch[lastBatch.length - 1];
|
|
41
|
+
let save = PlaitHistoryBoard.isSaving(board);
|
|
42
|
+
let merge = PlaitHistoryBoard.isMerging(board);
|
|
43
|
+
if (save == null) {
|
|
44
|
+
save = shouldSave(op, lastOp);
|
|
45
|
+
}
|
|
46
|
+
if (save) {
|
|
47
|
+
if (merge == null) {
|
|
48
|
+
if (lastBatch == null) {
|
|
49
|
+
merge = false;
|
|
50
|
+
}
|
|
51
|
+
else if (operations.length !== 0) {
|
|
52
|
+
merge = true;
|
|
53
|
+
}
|
|
54
|
+
else {
|
|
55
|
+
merge = shouldMerge(op, lastOp);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
if (lastBatch && merge) {
|
|
59
|
+
lastBatch.push(op);
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
const batch = [op];
|
|
63
|
+
undos.push(batch);
|
|
64
|
+
}
|
|
65
|
+
while (undos.length > 100) {
|
|
66
|
+
undos.shift();
|
|
67
|
+
}
|
|
68
|
+
if (shouldClear(op)) {
|
|
69
|
+
history.redos = [];
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
apply(op);
|
|
73
|
+
};
|
|
74
|
+
board.keydown = (event) => {
|
|
75
|
+
if (isHotkey('mod+z', event)) {
|
|
76
|
+
board.undo();
|
|
77
|
+
return;
|
|
78
|
+
}
|
|
79
|
+
if (isHotkey('mod+shift+z', event)) {
|
|
80
|
+
board.redo();
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
keydown(event);
|
|
84
|
+
};
|
|
85
|
+
return board;
|
|
86
|
+
}
|
|
87
|
+
export const SAVING = new WeakMap();
|
|
88
|
+
export const MERGING = new WeakMap();
|
|
89
|
+
export const PlaitHistoryBoard = {
|
|
90
|
+
/**
|
|
91
|
+
* Get the saving flag's current value.
|
|
92
|
+
*/
|
|
93
|
+
isSaving(board) {
|
|
94
|
+
return SAVING.get(board);
|
|
95
|
+
},
|
|
96
|
+
/**
|
|
97
|
+
* Get the merge flag's current value.
|
|
98
|
+
*/
|
|
99
|
+
isMerging(board) {
|
|
100
|
+
return MERGING.get(board);
|
|
101
|
+
},
|
|
102
|
+
/**
|
|
103
|
+
* Apply a series of changes inside a synchronous `fn`, without merging any of
|
|
104
|
+
* the new operations into previous save point in the history.
|
|
105
|
+
*/
|
|
106
|
+
withoutMerging(editor, fn) {
|
|
107
|
+
const prev = PlaitHistoryBoard.isMerging(editor);
|
|
108
|
+
MERGING.set(editor, false);
|
|
109
|
+
fn();
|
|
110
|
+
MERGING.set(editor, prev);
|
|
111
|
+
},
|
|
112
|
+
/**
|
|
113
|
+
* Apply a series of changes inside a synchronous `fn`, without saving any of
|
|
114
|
+
* their operations into the history.
|
|
115
|
+
*/
|
|
116
|
+
withoutSaving(editor, fn) {
|
|
117
|
+
const prev = PlaitHistoryBoard.isSaving(editor);
|
|
118
|
+
SAVING.set(editor, false);
|
|
119
|
+
fn();
|
|
120
|
+
SAVING.set(editor, prev);
|
|
121
|
+
}
|
|
122
|
+
};
|
|
123
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"with-history.js","sourceRoot":"","sources":["../../../../packages/plait/src/plugins/with-history.ts"],"names":[],"mappings":"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","sourcesContent":["import { PlaitBoard, PlaitOperation } from '../interfaces';\nimport { isHotkey } from 'is-hotkey';\nimport { shouldClear, shouldMerge, shouldSave } from '../utils';\n\nexport function withHistroy<T extends PlaitBoard>(board: T) {\n    const { apply, keydown } = board;\n    board.history = { undos: [], redos: [] };\n\n    board.redo = () => {\n        const { history } = board;\n        const { redos } = history;\n\n        if (redos.length > 0) {\n            const batch = redos[redos.length - 1];\n\n            PlaitHistoryBoard.withoutSaving(board, () => {\n                for (const op of batch) {\n                    board.apply(op);\n                }\n            });\n\n            history.redos.pop();\n            history.undos.push(batch);\n        }\n    };\n\n    board.undo = () => {\n        const { history } = board;\n        const { undos } = history;\n\n        if (undos.length > 0) {\n            const batch = undos[undos.length - 1];\n\n            PlaitHistoryBoard.withoutSaving(board, () => {\n                const inverseOps = batch.map(PlaitOperation.inverse).reverse();\n                for (const op of inverseOps) {\n                    board.apply(op);\n                }\n            });\n\n            history.redos.push(batch);\n            history.undos.pop();\n        }\n    };\n\n    board.apply = (op: PlaitOperation) => {\n        const { operations, history } = board;\n        const { undos } = history;\n        const lastBatch = undos[undos.length - 1];\n        const lastOp = lastBatch && lastBatch[lastBatch.length - 1];\n        let save = PlaitHistoryBoard.isSaving(board);\n        let merge = PlaitHistoryBoard.isMerging(board);\n\n        if (save == null) {\n            save = shouldSave(op, lastOp);\n        }\n\n        if (save) {\n            if (merge == null) {\n                if (lastBatch == null) {\n                    merge = false;\n                } else if (operations.length !== 0) {\n                    merge = true;\n                } else {\n                    merge = shouldMerge(op, lastOp);\n                }\n            }\n\n            if (lastBatch && merge) {\n                lastBatch.push(op);\n            } else {\n                const batch = [op];\n                undos.push(batch);\n            }\n\n            while (undos.length > 100) {\n                undos.shift();\n            }\n\n            if (shouldClear(op)) {\n                history.redos = [];\n            }\n        }\n        apply(op);\n    };\n\n    board.keydown = (event: KeyboardEvent) => {\n        if (isHotkey('mod+z', event)) {\n            board.undo();\n            return;\n        }\n        if (isHotkey('mod+shift+z', event)) {\n            board.redo();\n            return;\n        }\n        keydown(event);\n    };\n\n    return board;\n}\n\nexport const SAVING = new WeakMap<PlaitBoard, boolean | undefined>();\nexport const MERGING = new WeakMap<PlaitBoard, boolean | undefined>();\n\nexport const PlaitHistoryBoard = {\n    /**\n     * Get the saving flag's current value.\n     */\n    isSaving(board: PlaitBoard): boolean | undefined {\n        return SAVING.get(board);\n    },\n\n    /**\n     * Get the merge flag's current value.\n     */\n\n    isMerging(board: PlaitBoard): boolean | undefined {\n        return MERGING.get(board);\n    },\n\n    /**\n     * Apply a series of changes inside a synchronous `fn`, without merging any of\n     * the new operations into previous save point in the history.\n     */\n\n    withoutMerging(editor: PlaitBoard, fn: () => void): void {\n        const prev = PlaitHistoryBoard.isMerging(editor);\n        MERGING.set(editor, false);\n        fn();\n        MERGING.set(editor, prev);\n    },\n    /**\n     * Apply a series of changes inside a synchronous `fn`, without saving any of\n     * their operations into the history.\n     */\n\n    withoutSaving(editor: PlaitBoard, fn: () => void): void {\n        const prev = PlaitHistoryBoard.isSaving(editor);\n        SAVING.set(editor, false);\n        fn();\n        SAVING.set(editor, prev);\n    }\n};\n"]}
|
|
@@ -20,7 +20,8 @@ export function setNode(board, props, path) {
|
|
|
20
20
|
board.apply(operation);
|
|
21
21
|
}
|
|
22
22
|
export function removeNode(board, path) {
|
|
23
|
-
const
|
|
23
|
+
const node = PlaitNode.get(board, path);
|
|
24
|
+
const operation = { type: 'remove_node', path, node };
|
|
24
25
|
board.apply(operation);
|
|
25
26
|
}
|
|
26
27
|
export function moveNode(board, path, newPath) {
|
|
@@ -33,4 +34,4 @@ export const NodeTransforms = {
|
|
|
33
34
|
removeNode,
|
|
34
35
|
moveNode
|
|
35
36
|
};
|
|
36
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL3BsYWl0L3NyYy90cmFuc2Zyb21zL25vZGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBRy9DLE1BQU0sVUFBVSxVQUFVLENBQUMsS0FBaUIsRUFBRSxJQUFlLEVBQUUsSUFBVTtJQUNyRSxNQUFNLFNBQVMsR0FBd0IsRUFBRSxJQUFJLEVBQUUsYUFBYSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsQ0FBQztJQUMzRSxLQUFLLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQyxDQUFDO0FBQzNCLENBQUM7QUFFRCxNQUFNLFVBQVUsT0FBTyxDQUFDLEtBQWlCLEVBQUUsS0FBeUIsRUFBRSxJQUFVO0lBQzVFLE1BQU0sVUFBVSxHQUF1QixFQUFFLENBQUM7SUFDMUMsTUFBTSxhQUFhLEdBQXVCLEVBQUUsQ0FBQztJQUM3QyxNQUFNLElBQUksR0FBRyxTQUFTLENBQUMsR0FBRyxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsQ0FBQztJQUN4QyxLQUFLLE1BQU0sQ0FBQyxJQUFJLEtBQUssRUFBRTtRQUNuQixJQUFJLElBQUksQ0FBQyxDQUFDLENBQUMsS0FBSyxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUU7WUFDdEIsSUFBSSxJQUFJLENBQUMsY0FBYyxDQUFDLENBQUMsQ0FBQyxFQUFFO2dCQUN4QixVQUFVLENBQUMsQ0FBQyxDQUFDLEdBQUcsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO2FBQzNCO1lBQ0QsSUFBSSxLQUFLLENBQUMsQ0FBQyxDQUFDLElBQUksSUFBSTtnQkFBRSxhQUFhLENBQUMsQ0FBQyxDQUFDLEdBQUcsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1NBQ3JEO0tBQ0o7SUFDRCxNQUFNLFNBQVMsR0FBcUIsRUFBRSxJQUFJLEVBQUUsVUFBVSxFQUFFLFVBQVUsRUFBRSxhQUFhLEVBQUUsSUFBSSxFQUFFLENBQUM7SUFDMUYsS0FBSyxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQztBQUMzQixDQUFDO0FBRUQsTUFBTSxVQUFVLFVBQVUsQ0FBQyxLQUFpQixFQUFFLElBQVU7SUFDcEQsTUFBTSxJQUFJLEdBQUcsU0FBUyxDQUFDLEdBQUcsQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDeEMsTUFBTSxTQUFTLEdBQXdCLEVBQUUsSUFBSSxFQUFFLGFBQWEsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLENBQUM7SUFDM0UsS0FBSyxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQztBQUMzQixDQUFDO0FBRUQsTUFBTSxVQUFVLFFBQVEsQ0FBQyxLQUFpQixFQUFFLElBQVUsRUFBRSxPQUFhO0lBQ2pFLE1BQU0sU0FBUyxHQUFzQixFQUFFLElBQUksRUFBRSxXQUFXLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxDQUFDO0lBQzFFLEtBQUssQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDLENBQUM7QUFDM0IsQ0FBQztBQVNELE1BQU0sQ0FBQyxNQUFNLGNBQWMsR0FBbUI7SUFDMUMsVUFBVTtJQUNWLE9BQU87SUFDUCxVQUFVO0lBQ1YsUUFBUTtDQUNYLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbnNlcnROb2RlT3BlcmF0aW9uLCBNb3ZlTm9kZU9wZXJhdGlvbiwgUmVtb3ZlTm9kZU9wZXJhdGlvbiwgU2V0Tm9kZU9wZXJhdGlvbiB9IGZyb20gJy4uL2ludGVyZmFjZXMvb3BlcmF0aW9uJztcbmltcG9ydCB7IFBsYWl0Qm9hcmQgfSBmcm9tICcuLi9pbnRlcmZhY2VzL2JvYXJkJztcbmltcG9ydCB7IFBsYWl0Tm9kZSB9IGZyb20gJy4uL2ludGVyZmFjZXMvbm9kZSc7XG5pbXBvcnQgeyBQYXRoIH0gZnJvbSAnLi4vaW50ZXJmYWNlcy9wYXRoJztcblxuZXhwb3J0IGZ1bmN0aW9uIGluc2VydE5vZGUoYm9hcmQ6IFBsYWl0Qm9hcmQsIG5vZGU6IFBsYWl0Tm9kZSwgcGF0aDogUGF0aCkge1xuICAgIGNvbnN0IG9wZXJhdGlvbjogSW5zZXJ0Tm9kZU9wZXJhdGlvbiA9IHsgdHlwZTogJ2luc2VydF9ub2RlJywgbm9kZSwgcGF0aCB9O1xuICAgIGJvYXJkLmFwcGx5KG9wZXJhdGlvbik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBzZXROb2RlKGJvYXJkOiBQbGFpdEJvYXJkLCBwcm9wczogUGFydGlhbDxQbGFpdE5vZGU+LCBwYXRoOiBQYXRoKSB7XG4gICAgY29uc3QgcHJvcGVydGllczogUGFydGlhbDxQbGFpdE5vZGU+ID0ge307XG4gICAgY29uc3QgbmV3UHJvcGVydGllczogUGFydGlhbDxQbGFpdE5vZGU+ID0ge307XG4gICAgY29uc3Qgbm9kZSA9IFBsYWl0Tm9kZS5nZXQoYm9hcmQsIHBhdGgpO1xuICAgIGZvciAoY29uc3QgayBpbiBwcm9wcykge1xuICAgICAgICBpZiAobm9kZVtrXSAhPT0gcHJvcHNba10pIHtcbiAgICAgICAgICAgIGlmIChub2RlLmhhc093blByb3BlcnR5KGspKSB7XG4gICAgICAgICAgICAgICAgcHJvcGVydGllc1trXSA9IG5vZGVba107XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBpZiAocHJvcHNba10gIT0gbnVsbCkgbmV3UHJvcGVydGllc1trXSA9IHByb3BzW2tdO1xuICAgICAgICB9XG4gICAgfVxuICAgIGNvbnN0IG9wZXJhdGlvbjogU2V0Tm9kZU9wZXJhdGlvbiA9IHsgdHlwZTogJ3NldF9ub2RlJywgcHJvcGVydGllcywgbmV3UHJvcGVydGllcywgcGF0aCB9O1xuICAgIGJvYXJkLmFwcGx5KG9wZXJhdGlvbik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiByZW1vdmVOb2RlKGJvYXJkOiBQbGFpdEJvYXJkLCBwYXRoOiBQYXRoKSB7XG4gICAgY29uc3Qgbm9kZSA9IFBsYWl0Tm9kZS5nZXQoYm9hcmQsIHBhdGgpO1xuICAgIGNvbnN0IG9wZXJhdGlvbjogUmVtb3ZlTm9kZU9wZXJhdGlvbiA9IHsgdHlwZTogJ3JlbW92ZV9ub2RlJywgcGF0aCwgbm9kZSB9O1xuICAgIGJvYXJkLmFwcGx5KG9wZXJhdGlvbik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBtb3ZlTm9kZShib2FyZDogUGxhaXRCb2FyZCwgcGF0aDogUGF0aCwgbmV3UGF0aDogUGF0aCkge1xuICAgIGNvbnN0IG9wZXJhdGlvbjogTW92ZU5vZGVPcGVyYXRpb24gPSB7IHR5cGU6ICdtb3ZlX25vZGUnLCBwYXRoLCBuZXdQYXRoIH07XG4gICAgYm9hcmQuYXBwbHkob3BlcmF0aW9uKTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBOb2RlVHJhbnNmb3JtcyB7XG4gICAgaW5zZXJ0Tm9kZTogKGJvYXJkOiBQbGFpdEJvYXJkLCBub2RlOiBQbGFpdE5vZGUsIHBhdGg6IFBhdGgpID0+IHZvaWQ7XG4gICAgc2V0Tm9kZTogKGJvYXJkOiBQbGFpdEJvYXJkLCBub2RlOiBQYXJ0aWFsPFBsYWl0Tm9kZT4sIHBhdGg6IFBhdGgpID0+IHZvaWQ7XG4gICAgcmVtb3ZlTm9kZTogKGJvYXJkOiBQbGFpdEJvYXJkLCBwYXRoOiBQYXRoKSA9PiB2b2lkO1xuICAgIG1vdmVOb2RlOiAoYm9hcmQ6IFBsYWl0Qm9hcmQsIHBhdGg6IFBhdGgsIG5ld1BhdGg6IFBhdGgpID0+IHZvaWQ7XG59XG5cbmV4cG9ydCBjb25zdCBOb2RlVHJhbnNmb3JtczogTm9kZVRyYW5zZm9ybXMgPSB7XG4gICAgaW5zZXJ0Tm9kZSxcbiAgICBzZXROb2RlLFxuICAgIHJlbW92ZU5vZGUsXG4gICAgbW92ZU5vZGVcbn07XG4iXX0=
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Check whether to merge an operation into the previous operation.
|
|
3
|
+
*/
|
|
4
|
+
export const shouldMerge = (op, prev) => {
|
|
5
|
+
if (op.type === 'set_viewport') {
|
|
6
|
+
return true;
|
|
7
|
+
}
|
|
8
|
+
return false;
|
|
9
|
+
};
|
|
10
|
+
/**
|
|
11
|
+
* Check whether an operation needs to be saved to the history.
|
|
12
|
+
*/
|
|
13
|
+
export const shouldSave = (op, prev) => {
|
|
14
|
+
if (op.type === 'set_selection') {
|
|
15
|
+
return false;
|
|
16
|
+
}
|
|
17
|
+
return true;
|
|
18
|
+
};
|
|
19
|
+
/**
|
|
20
|
+
* Check whether an operation should clear the redos stack.
|
|
21
|
+
*/
|
|
22
|
+
export const shouldClear = (op) => {
|
|
23
|
+
if (op.type === 'set_selection') {
|
|
24
|
+
return false;
|
|
25
|
+
}
|
|
26
|
+
return true;
|
|
27
|
+
};
|
|
28
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGlzdG9yeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3BhY2thZ2VzL3BsYWl0L3NyYy91dGlscy9oaXN0b3J5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBOztHQUVHO0FBRUgsTUFBTSxDQUFDLE1BQU0sV0FBVyxHQUFHLENBQUMsRUFBa0IsRUFBRSxJQUFnQyxFQUFXLEVBQUU7SUFDekYsSUFBSSxFQUFFLENBQUMsSUFBSSxLQUFLLGNBQWMsRUFBRTtRQUM1QixPQUFPLElBQUksQ0FBQztLQUNmO0lBQ0QsT0FBTyxLQUFLLENBQUM7QUFDakIsQ0FBQyxDQUFDO0FBRUY7O0dBRUc7QUFFSCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQUcsQ0FBQyxFQUFrQixFQUFFLElBQWdDLEVBQVcsRUFBRTtJQUN4RixJQUFJLEVBQUUsQ0FBQyxJQUFJLEtBQUssZUFBZSxFQUFFO1FBQzdCLE9BQU8sS0FBSyxDQUFDO0tBQ2hCO0lBRUQsT0FBTyxJQUFJLENBQUM7QUFDaEIsQ0FBQyxDQUFDO0FBRUY7O0dBRUc7QUFFSCxNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsQ0FBQyxFQUFrQixFQUFXLEVBQUU7SUFDdkQsSUFBSSxFQUFFLENBQUMsSUFBSSxLQUFLLGVBQWUsRUFBRTtRQUM3QixPQUFPLEtBQUssQ0FBQztLQUNoQjtJQUVELE9BQU8sSUFBSSxDQUFDO0FBQ2hCLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBsYWl0T3BlcmF0aW9uIH0gZnJvbSAnLi4vaW50ZXJmYWNlcyc7XG5cbi8qKlxuICogQ2hlY2sgd2hldGhlciB0byBtZXJnZSBhbiBvcGVyYXRpb24gaW50byB0aGUgcHJldmlvdXMgb3BlcmF0aW9uLlxuICovXG5cbmV4cG9ydCBjb25zdCBzaG91bGRNZXJnZSA9IChvcDogUGxhaXRPcGVyYXRpb24sIHByZXY6IFBsYWl0T3BlcmF0aW9uIHwgdW5kZWZpbmVkKTogYm9vbGVhbiA9PiB7XG4gICAgaWYgKG9wLnR5cGUgPT09ICdzZXRfdmlld3BvcnQnKSB7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cbiAgICByZXR1cm4gZmFsc2U7XG59O1xuXG4vKipcbiAqIENoZWNrIHdoZXRoZXIgYW4gb3BlcmF0aW9uIG5lZWRzIHRvIGJlIHNhdmVkIHRvIHRoZSBoaXN0b3J5LlxuICovXG5cbmV4cG9ydCBjb25zdCBzaG91bGRTYXZlID0gKG9wOiBQbGFpdE9wZXJhdGlvbiwgcHJldjogUGxhaXRPcGVyYXRpb24gfCB1bmRlZmluZWQpOiBib29sZWFuID0+IHtcbiAgICBpZiAob3AudHlwZSA9PT0gJ3NldF9zZWxlY3Rpb24nKSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG5cbiAgICByZXR1cm4gdHJ1ZTtcbn07XG5cbi8qKlxuICogQ2hlY2sgd2hldGhlciBhbiBvcGVyYXRpb24gc2hvdWxkIGNsZWFyIHRoZSByZWRvcyBzdGFjay5cbiAqL1xuXG5leHBvcnQgY29uc3Qgc2hvdWxkQ2xlYXIgPSAob3A6IFBsYWl0T3BlcmF0aW9uKTogYm9vbGVhbiA9PiB7XG4gICAgaWYgKG9wLnR5cGUgPT09ICdzZXRfc2VsZWN0aW9uJykge1xuICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgcmV0dXJuIHRydWU7XG59O1xuIl19
|
package/esm2020/utils/index.mjs
CHANGED
|
@@ -7,4 +7,5 @@ export * from './weak-maps';
|
|
|
7
7
|
export * from './helper';
|
|
8
8
|
export * from './math';
|
|
9
9
|
export * from './board';
|
|
10
|
-
|
|
10
|
+
export * from './history';
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wYWNrYWdlcy9wbGFpdC9zcmMvdXRpbHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxPQUFPLENBQUM7QUFDdEIsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxXQUFXLENBQUM7QUFDMUIsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxVQUFVLENBQUM7QUFDekIsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxTQUFTLENBQUM7QUFDeEIsY0FBYyxXQUFXLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2RvbSc7XG5leHBvcnQgKiBmcm9tICcuL2Vudmlyb25tZW50JztcbmV4cG9ydCAqIGZyb20gJy4vZ3JhcGgnO1xuZXhwb3J0ICogZnJvbSAnLi9ob3RrZXlzJztcbmV4cG9ydCAqIGZyb20gJy4vaWQtY3JlYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3dlYWstbWFwcyc7XG5leHBvcnQgKiBmcm9tICcuL2hlbHBlcic7XG5leHBvcnQgKiBmcm9tICcuL21hdGgnO1xuZXhwb3J0ICogZnJvbSAnLi9ib2FyZCc7XG5leHBvcnQgKiBmcm9tICcuL2hpc3RvcnknO1xuIl19
|