@tldraw/editor 3.14.0-canary.6fc604d88ca2 → 3.14.0-canary.718fcc03811e
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/dist-cjs/index.d.ts +67 -62
- package/dist-cjs/index.js +8 -10
- package/dist-cjs/index.js.map +2 -2
- package/dist-cjs/lib/config/TLSessionStateSnapshot.js +1 -12
- package/dist-cjs/lib/config/TLSessionStateSnapshot.js.map +3 -3
- package/dist-cjs/lib/editor/Editor.js +67 -72
- package/dist-cjs/lib/editor/Editor.js.map +2 -2
- package/dist-cjs/lib/editor/bindings/BindingUtil.js.map +2 -2
- package/dist-cjs/lib/editor/derivations/bindingsIndex.js +22 -22
- package/dist-cjs/lib/editor/derivations/bindingsIndex.js.map +2 -2
- package/dist-cjs/lib/editor/derivations/parentsToChildren.js +16 -16
- package/dist-cjs/lib/editor/derivations/parentsToChildren.js.map +2 -2
- package/dist-cjs/lib/editor/managers/{ClickManager.js → ClickManager/ClickManager.js} +1 -1
- package/dist-cjs/lib/editor/managers/ClickManager/ClickManager.js.map +7 -0
- package/dist-cjs/lib/editor/managers/{EdgeScrollManager.js → EdgeScrollManager/EdgeScrollManager.js} +2 -2
- package/dist-cjs/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.js.map +7 -0
- package/dist-cjs/lib/editor/managers/FocusManager/FocusManager.js.map +7 -0
- package/dist-cjs/lib/editor/managers/{FontManager.js → FontManager/FontManager.js} +1 -2
- package/dist-cjs/lib/editor/managers/FontManager/FontManager.js.map +7 -0
- package/dist-cjs/lib/editor/managers/{HistoryManager.js → HistoryManager/HistoryManager.js} +64 -6
- package/dist-cjs/lib/editor/managers/HistoryManager/HistoryManager.js.map +7 -0
- package/dist-cjs/lib/editor/managers/{ScribbleManager.js → ScribbleManager/ScribbleManager.js} +1 -1
- package/dist-cjs/lib/editor/managers/ScribbleManager/ScribbleManager.js.map +7 -0
- package/dist-cjs/lib/editor/managers/{TextManager.js → TextManager/TextManager.js} +72 -42
- package/dist-cjs/lib/editor/managers/TextManager/TextManager.js.map +7 -0
- package/dist-cjs/lib/editor/managers/{TickManager.js → TickManager/TickManager.js} +1 -1
- package/dist-cjs/lib/editor/managers/TickManager/TickManager.js.map +7 -0
- package/dist-cjs/lib/editor/managers/{UserPreferencesManager.js → UserPreferencesManager/UserPreferencesManager.js} +1 -1
- package/dist-cjs/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.js.map +7 -0
- package/dist-cjs/lib/editor/shapes/ShapeUtil.js.map +1 -1
- package/dist-cjs/lib/editor/shapes/group/GroupShapeUtil.js +1 -1
- package/dist-cjs/lib/editor/shapes/group/GroupShapeUtil.js.map +1 -1
- package/dist-cjs/lib/editor/types/external-content.js.map +1 -1
- package/dist-cjs/lib/exports/getSvgJsx.js.map +1 -1
- package/dist-cjs/lib/hooks/useCanvasEvents.js +1 -2
- package/dist-cjs/lib/hooks/useCanvasEvents.js.map +2 -2
- package/dist-cjs/lib/primitives/Box.js +0 -6
- package/dist-cjs/lib/primitives/Box.js.map +2 -2
- package/dist-cjs/lib/utils/areShapesContentEqual.js +1 -1
- package/dist-cjs/lib/utils/areShapesContentEqual.js.map +2 -2
- package/dist-cjs/lib/utils/reorderShapes.js +11 -10
- package/dist-cjs/lib/utils/reorderShapes.js.map +2 -2
- package/dist-cjs/lib/utils/richText.js +7 -2
- package/dist-cjs/lib/utils/richText.js.map +2 -2
- package/dist-cjs/version.js +3 -3
- package/dist-cjs/version.js.map +1 -1
- package/dist-esm/index.d.mts +67 -62
- package/dist-esm/index.mjs +12 -10
- package/dist-esm/index.mjs.map +2 -2
- package/dist-esm/lib/config/TLSessionStateSnapshot.mjs +1 -1
- package/dist-esm/lib/config/TLSessionStateSnapshot.mjs.map +2 -2
- package/dist-esm/lib/editor/Editor.mjs +67 -72
- package/dist-esm/lib/editor/Editor.mjs.map +2 -2
- package/dist-esm/lib/editor/bindings/BindingUtil.mjs.map +2 -2
- package/dist-esm/lib/editor/derivations/bindingsIndex.mjs +22 -22
- package/dist-esm/lib/editor/derivations/bindingsIndex.mjs.map +2 -2
- package/dist-esm/lib/editor/derivations/parentsToChildren.mjs +16 -16
- package/dist-esm/lib/editor/derivations/parentsToChildren.mjs.map +2 -2
- package/dist-esm/lib/editor/managers/{ClickManager.mjs → ClickManager/ClickManager.mjs} +1 -1
- package/dist-esm/lib/editor/managers/ClickManager/ClickManager.mjs.map +7 -0
- package/dist-esm/lib/editor/managers/{EdgeScrollManager.mjs → EdgeScrollManager/EdgeScrollManager.mjs} +2 -2
- package/dist-esm/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.mjs.map +7 -0
- package/dist-esm/lib/editor/managers/FocusManager/FocusManager.mjs.map +7 -0
- package/dist-esm/lib/editor/managers/{FontManager.mjs → FontManager/FontManager.mjs} +1 -2
- package/dist-esm/lib/editor/managers/FontManager/FontManager.mjs.map +7 -0
- package/dist-esm/lib/editor/managers/{HistoryManager.mjs → HistoryManager/HistoryManager.mjs} +60 -2
- package/dist-esm/lib/editor/managers/HistoryManager/HistoryManager.mjs.map +7 -0
- package/dist-esm/lib/editor/managers/{ScribbleManager.mjs → ScribbleManager/ScribbleManager.mjs} +1 -1
- package/dist-esm/lib/editor/managers/ScribbleManager/ScribbleManager.mjs.map +7 -0
- package/dist-esm/lib/editor/managers/{TextManager.mjs → TextManager/TextManager.mjs} +72 -42
- package/dist-esm/lib/editor/managers/TextManager/TextManager.mjs.map +7 -0
- package/dist-esm/lib/editor/managers/{TickManager.mjs → TickManager/TickManager.mjs} +1 -1
- package/dist-esm/lib/editor/managers/TickManager/TickManager.mjs.map +7 -0
- package/dist-esm/lib/editor/managers/{UserPreferencesManager.mjs → UserPreferencesManager/UserPreferencesManager.mjs} +1 -1
- package/dist-esm/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.mjs.map +7 -0
- package/dist-esm/lib/editor/shapes/ShapeUtil.mjs.map +1 -1
- package/dist-esm/lib/editor/shapes/group/GroupShapeUtil.mjs +1 -1
- package/dist-esm/lib/editor/shapes/group/GroupShapeUtil.mjs.map +1 -1
- package/dist-esm/lib/exports/getSvgJsx.mjs.map +1 -1
- package/dist-esm/lib/hooks/useCanvasEvents.mjs +1 -2
- package/dist-esm/lib/hooks/useCanvasEvents.mjs.map +2 -2
- package/dist-esm/lib/primitives/Box.mjs +0 -6
- package/dist-esm/lib/primitives/Box.mjs.map +2 -2
- package/dist-esm/lib/utils/areShapesContentEqual.mjs +1 -1
- package/dist-esm/lib/utils/areShapesContentEqual.mjs.map +2 -2
- package/dist-esm/lib/utils/reorderShapes.mjs +11 -10
- package/dist-esm/lib/utils/reorderShapes.mjs.map +2 -2
- package/dist-esm/lib/utils/richText.mjs +8 -3
- package/dist-esm/lib/utils/richText.mjs.map +2 -2
- package/dist-esm/version.mjs +3 -3
- package/dist-esm/version.mjs.map +1 -1
- package/editor.css +433 -482
- package/package.json +8 -9
- package/src/index.ts +15 -8
- package/src/lib/config/TLSessionStateSnapshot.ts +1 -1
- package/src/lib/editor/Editor.test.ts +252 -3
- package/src/lib/editor/Editor.ts +73 -71
- package/src/lib/editor/bindings/BindingUtil.ts +6 -0
- package/src/lib/editor/derivations/bindingsIndex.ts +27 -26
- package/src/lib/editor/derivations/parentsToChildren.ts +28 -25
- package/src/lib/editor/managers/ClickManager/ClickManager.test.ts +442 -0
- package/src/lib/editor/managers/{ClickManager.ts → ClickManager/ClickManager.ts} +3 -3
- package/src/lib/editor/managers/EdgeScrollManager/EdgeScrollManager.test.ts +374 -0
- package/src/lib/editor/managers/{EdgeScrollManager.ts → EdgeScrollManager/EdgeScrollManager.ts} +3 -3
- package/src/lib/editor/managers/FocusManager/FocusManager.test.ts +455 -0
- package/src/lib/editor/managers/{FocusManager.ts → FocusManager/FocusManager.ts} +1 -1
- package/src/lib/editor/managers/FontManager/FontManager.test.ts +263 -0
- package/src/lib/editor/managers/{FontManager.ts → FontManager/FontManager.ts} +2 -3
- package/src/lib/editor/managers/{HistoryManager.test.ts → HistoryManager/HistoryManager.test.ts} +388 -1
- package/src/lib/editor/managers/{HistoryManager.ts → HistoryManager/HistoryManager.ts} +73 -2
- package/src/lib/editor/managers/ScribbleManager/ScribbleManager.test.ts +624 -0
- package/src/lib/editor/managers/{ScribbleManager.ts → ScribbleManager/ScribbleManager.ts} +2 -2
- package/src/lib/editor/managers/SnapManager/SnapManager.test.ts +485 -0
- package/src/lib/editor/managers/TextManager/TextManager.test.ts +407 -0
- package/src/lib/editor/managers/{TextManager.ts → TextManager/TextManager.ts} +117 -87
- package/src/lib/editor/managers/TickManager/TickManager.test.ts +314 -0
- package/src/lib/editor/managers/{TickManager.ts → TickManager/TickManager.ts} +2 -2
- package/src/lib/editor/managers/UserPreferencesManager/UserPreferencesManager.test.ts +591 -0
- package/src/lib/editor/managers/{UserPreferencesManager.ts → UserPreferencesManager/UserPreferencesManager.ts} +2 -2
- package/src/lib/editor/shapes/ShapeUtil.ts +1 -1
- package/src/lib/editor/shapes/group/GroupShapeUtil.tsx +1 -1
- package/src/lib/editor/types/external-content.ts +11 -2
- package/src/lib/exports/getSvgJsx.tsx +1 -1
- package/src/lib/hooks/useCanvasEvents.ts +0 -1
- package/src/lib/primitives/Box.ts +0 -8
- package/src/lib/utils/areShapesContentEqual.ts +1 -2
- package/src/lib/utils/reorderShapes.ts +10 -13
- package/src/lib/utils/richText.ts +10 -4
- package/src/version.ts +3 -3
- package/dist-cjs/lib/editor/managers/ClickManager.js.map +0 -7
- package/dist-cjs/lib/editor/managers/EdgeScrollManager.js.map +0 -7
- package/dist-cjs/lib/editor/managers/FocusManager.js.map +0 -7
- package/dist-cjs/lib/editor/managers/FontManager.js.map +0 -7
- package/dist-cjs/lib/editor/managers/HistoryManager.js.map +0 -7
- package/dist-cjs/lib/editor/managers/ScribbleManager.js.map +0 -7
- package/dist-cjs/lib/editor/managers/Stack.js +0 -82
- package/dist-cjs/lib/editor/managers/Stack.js.map +0 -7
- package/dist-cjs/lib/editor/managers/TextManager.js.map +0 -7
- package/dist-cjs/lib/editor/managers/TickManager.js.map +0 -7
- package/dist-cjs/lib/editor/managers/UserPreferencesManager.js.map +0 -7
- package/dist-esm/lib/editor/managers/ClickManager.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/EdgeScrollManager.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/FocusManager.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/FontManager.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/HistoryManager.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/ScribbleManager.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/Stack.mjs +0 -62
- package/dist-esm/lib/editor/managers/Stack.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/TextManager.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/TickManager.mjs.map +0 -7
- package/dist-esm/lib/editor/managers/UserPreferencesManager.mjs.map +0 -7
- package/src/lib/editor/managers/ScribbleManager.test.ts +0 -32
- package/src/lib/editor/managers/Stack.ts +0 -71
- /package/dist-cjs/lib/editor/managers/{FocusManager.js → FocusManager/FocusManager.js} +0 -0
- /package/dist-esm/lib/editor/managers/{FocusManager.mjs → FocusManager/FocusManager.mjs} +0 -0
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@tldraw/editor",
|
|
3
3
|
"description": "A tiny little drawing app (editor).",
|
|
4
|
-
"version": "3.14.0-canary.
|
|
4
|
+
"version": "3.14.0-canary.718fcc03811e",
|
|
5
5
|
"author": {
|
|
6
6
|
"name": "tldraw Inc.",
|
|
7
7
|
"email": "hello@tldraw.com"
|
|
@@ -48,20 +48,19 @@
|
|
|
48
48
|
"@tiptap/core": "^2.9.1",
|
|
49
49
|
"@tiptap/pm": "^2.9.1",
|
|
50
50
|
"@tiptap/react": "^2.9.1",
|
|
51
|
-
"@tldraw/state": "3.14.0-canary.
|
|
52
|
-
"@tldraw/state-react": "3.14.0-canary.
|
|
53
|
-
"@tldraw/store": "3.14.0-canary.
|
|
54
|
-
"@tldraw/tlschema": "3.14.0-canary.
|
|
55
|
-
"@tldraw/utils": "3.14.0-canary.
|
|
56
|
-
"@tldraw/validate": "3.14.0-canary.
|
|
51
|
+
"@tldraw/state": "3.14.0-canary.718fcc03811e",
|
|
52
|
+
"@tldraw/state-react": "3.14.0-canary.718fcc03811e",
|
|
53
|
+
"@tldraw/store": "3.14.0-canary.718fcc03811e",
|
|
54
|
+
"@tldraw/tlschema": "3.14.0-canary.718fcc03811e",
|
|
55
|
+
"@tldraw/utils": "3.14.0-canary.718fcc03811e",
|
|
56
|
+
"@tldraw/validate": "3.14.0-canary.718fcc03811e",
|
|
57
57
|
"@types/core-js": "^2.5.8",
|
|
58
58
|
"@use-gesture/react": "^10.3.1",
|
|
59
59
|
"classnames": "^2.5.1",
|
|
60
60
|
"core-js": "^3.40.0",
|
|
61
61
|
"eventemitter3": "^4.0.7",
|
|
62
62
|
"idb": "^7.1.1",
|
|
63
|
-
"is-plain-object": "^5.0.0"
|
|
64
|
-
"lodash.isequal": "^4.5.0"
|
|
63
|
+
"is-plain-object": "^5.0.0"
|
|
65
64
|
},
|
|
66
65
|
"peerDependencies": {
|
|
67
66
|
"react": "^18.2.0 || ^19.0.0",
|
package/src/index.ts
CHANGED
|
@@ -4,7 +4,6 @@ import 'core-js/stable/array/flat-map.js'
|
|
|
4
4
|
import 'core-js/stable/array/flat.js'
|
|
5
5
|
import 'core-js/stable/string/at.js'
|
|
6
6
|
import 'core-js/stable/string/replace-all.js'
|
|
7
|
-
export { areShapesContentEqual } from './lib/utils/areShapesContentEqual'
|
|
8
7
|
|
|
9
8
|
// eslint-disable-next-line local/no-export-star
|
|
10
9
|
export * from '@tldraw/state'
|
|
@@ -148,15 +147,18 @@ export {
|
|
|
148
147
|
type BindingOnShapeIsolateOptions,
|
|
149
148
|
type TLBindingUtilConstructor,
|
|
150
149
|
} from './lib/editor/bindings/BindingUtil'
|
|
151
|
-
export { ClickManager, type TLClickState } from './lib/editor/managers/ClickManager'
|
|
152
|
-
export { EdgeScrollManager } from './lib/editor/managers/EdgeScrollManager'
|
|
150
|
+
export { ClickManager, type TLClickState } from './lib/editor/managers/ClickManager/ClickManager'
|
|
151
|
+
export { EdgeScrollManager } from './lib/editor/managers/EdgeScrollManager/EdgeScrollManager'
|
|
153
152
|
export {
|
|
154
153
|
FontManager,
|
|
155
154
|
type TLFontFace,
|
|
156
155
|
type TLFontFaceSource,
|
|
157
|
-
} from './lib/editor/managers/FontManager'
|
|
158
|
-
export { HistoryManager } from './lib/editor/managers/HistoryManager'
|
|
159
|
-
export {
|
|
156
|
+
} from './lib/editor/managers/FontManager/FontManager'
|
|
157
|
+
export { HistoryManager } from './lib/editor/managers/HistoryManager/HistoryManager'
|
|
158
|
+
export {
|
|
159
|
+
ScribbleManager,
|
|
160
|
+
type ScribbleItem,
|
|
161
|
+
} from './lib/editor/managers/ScribbleManager/ScribbleManager'
|
|
160
162
|
export {
|
|
161
163
|
BoundsSnaps,
|
|
162
164
|
type BoundsSnapGeometry,
|
|
@@ -170,8 +172,12 @@ export {
|
|
|
170
172
|
type SnapData,
|
|
171
173
|
type SnapIndicator,
|
|
172
174
|
} from './lib/editor/managers/SnapManager/SnapManager'
|
|
173
|
-
export {
|
|
174
|
-
|
|
175
|
+
export {
|
|
176
|
+
TextManager,
|
|
177
|
+
type TLMeasureTextOpts,
|
|
178
|
+
type TLMeasureTextSpanOpts,
|
|
179
|
+
} from './lib/editor/managers/TextManager/TextManager'
|
|
180
|
+
export { UserPreferencesManager } from './lib/editor/managers/UserPreferencesManager/UserPreferencesManager'
|
|
175
181
|
export { BaseBoxShapeUtil, type TLBaseBoxShape } from './lib/editor/shapes/BaseBoxShapeUtil'
|
|
176
182
|
export {
|
|
177
183
|
ShapeUtil,
|
|
@@ -247,6 +253,7 @@ export {
|
|
|
247
253
|
type TLExternalContent,
|
|
248
254
|
type TLExternalContentSource,
|
|
249
255
|
type TLFileExternalAsset,
|
|
256
|
+
type TLFileReplaceExternalContent,
|
|
250
257
|
type TLFilesExternalContent,
|
|
251
258
|
type TLSvgTextExternalContent,
|
|
252
259
|
type TLTextExternalContent,
|
|
@@ -14,12 +14,12 @@ import {
|
|
|
14
14
|
import {
|
|
15
15
|
deleteFromSessionStorage,
|
|
16
16
|
getFromSessionStorage,
|
|
17
|
+
isEqual,
|
|
17
18
|
setInSessionStorage,
|
|
18
19
|
structuredClone,
|
|
19
20
|
uniqueId,
|
|
20
21
|
} from '@tldraw/utils'
|
|
21
22
|
import { T } from '@tldraw/validate'
|
|
22
|
-
import isEqual from 'lodash.isequal'
|
|
23
23
|
import { tlenv } from '../globals/environment'
|
|
24
24
|
|
|
25
25
|
const tabIdKey = 'TLDRAW_TAB_ID_v2' as const
|
|
@@ -17,6 +17,7 @@ type ICustomShape = TLBaseShape<
|
|
|
17
17
|
w: number
|
|
18
18
|
h: number
|
|
19
19
|
text: string | undefined
|
|
20
|
+
isFilled: boolean
|
|
20
21
|
}
|
|
21
22
|
>
|
|
22
23
|
|
|
@@ -26,19 +27,21 @@ class CustomShape extends ShapeUtil<ICustomShape> {
|
|
|
26
27
|
w: T.number,
|
|
27
28
|
h: T.number,
|
|
28
29
|
text: T.string.optional(),
|
|
30
|
+
isFilled: T.boolean,
|
|
29
31
|
}
|
|
30
32
|
getDefaultProps(): ICustomShape['props'] {
|
|
31
33
|
return {
|
|
32
34
|
w: 200,
|
|
33
35
|
h: 200,
|
|
34
36
|
text: '',
|
|
37
|
+
isFilled: false,
|
|
35
38
|
}
|
|
36
39
|
}
|
|
37
40
|
getGeometry(shape: ICustomShape): Geometry2d {
|
|
38
41
|
return new Rectangle2d({
|
|
39
42
|
width: shape.props.w,
|
|
40
43
|
height: shape.props.h,
|
|
41
|
-
isFilled:
|
|
44
|
+
isFilled: shape.props.isFilled,
|
|
42
45
|
})
|
|
43
46
|
}
|
|
44
47
|
indicator() {}
|
|
@@ -81,11 +84,11 @@ describe('updateShape', () => {
|
|
|
81
84
|
props: { w: 100, h: 100, text: 'Hello' },
|
|
82
85
|
})
|
|
83
86
|
const shape = editor.getShape(id) as ICustomShape
|
|
84
|
-
expect(shape.props).toEqual({ w: 100, h: 100, text: 'Hello' })
|
|
87
|
+
expect(shape.props).toEqual({ w: 100, h: 100, text: 'Hello', isFilled: false })
|
|
85
88
|
|
|
86
89
|
editor.updateShape({ ...shape, props: { ...shape.props, text: undefined } })
|
|
87
90
|
const updatedShape = editor.getShape(id) as ICustomShape
|
|
88
|
-
expect(updatedShape.props).toEqual({ w: 100, h: 100, text: undefined })
|
|
91
|
+
expect(updatedShape.props).toEqual({ w: 100, h: 100, text: undefined, isFilled: false })
|
|
89
92
|
})
|
|
90
93
|
})
|
|
91
94
|
|
|
@@ -176,3 +179,249 @@ describe('zoomToBounds', () => {
|
|
|
176
179
|
expect(editor.setCamera).toHaveBeenCalled()
|
|
177
180
|
})
|
|
178
181
|
})
|
|
182
|
+
|
|
183
|
+
describe('getShapesAtPoint', () => {
|
|
184
|
+
const ids = {
|
|
185
|
+
shape1: createShapeId('shape1'),
|
|
186
|
+
shape2: createShapeId('shape2'),
|
|
187
|
+
shape3: createShapeId('shape3'),
|
|
188
|
+
shape4: createShapeId('shape4'),
|
|
189
|
+
shape5: createShapeId('shape5'),
|
|
190
|
+
overlap1: createShapeId('overlap1'),
|
|
191
|
+
overlap2: createShapeId('overlap2'),
|
|
192
|
+
filledShape: createShapeId('filledShape'),
|
|
193
|
+
hollowShape: createShapeId('hollowShape'),
|
|
194
|
+
hiddenShape: createShapeId('hiddenShape'),
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
beforeEach(() => {
|
|
198
|
+
// Create test shapes with different z-index positions
|
|
199
|
+
// Shape 1: Bottom layer, large square
|
|
200
|
+
editor.createShape({
|
|
201
|
+
id: ids.shape1,
|
|
202
|
+
type: 'my-custom-shape',
|
|
203
|
+
x: 0,
|
|
204
|
+
y: 0,
|
|
205
|
+
props: { w: 200, h: 200, text: 'Bottom' },
|
|
206
|
+
})
|
|
207
|
+
|
|
208
|
+
// Shape 2: Middle layer, overlapping square
|
|
209
|
+
editor.createShape({
|
|
210
|
+
id: ids.shape2,
|
|
211
|
+
type: 'my-custom-shape',
|
|
212
|
+
x: 100,
|
|
213
|
+
y: 0,
|
|
214
|
+
props: { w: 200, h: 200, text: 'Middle' },
|
|
215
|
+
})
|
|
216
|
+
|
|
217
|
+
// Shape 3: Top layer, small square
|
|
218
|
+
editor.createShape({
|
|
219
|
+
id: ids.shape3,
|
|
220
|
+
type: 'my-custom-shape',
|
|
221
|
+
x: 50,
|
|
222
|
+
y: 50,
|
|
223
|
+
props: { w: 100, h: 100, text: 'Top' },
|
|
224
|
+
})
|
|
225
|
+
|
|
226
|
+
// Shape 4: Separate area, no overlap
|
|
227
|
+
editor.createShape({
|
|
228
|
+
id: ids.shape4,
|
|
229
|
+
type: 'my-custom-shape',
|
|
230
|
+
x: 50,
|
|
231
|
+
y: 100,
|
|
232
|
+
props: { w: 100, h: 100, text: 'Separate' },
|
|
233
|
+
})
|
|
234
|
+
})
|
|
235
|
+
|
|
236
|
+
it('returns shapes at a point in reverse z-index order', () => {
|
|
237
|
+
// Point at (50, 50) should hit shape3's edge (since it's at 50,50 with size 100x100)
|
|
238
|
+
// This point is exactly at the top-left corner of shape3
|
|
239
|
+
const shapes = editor.getShapesAtPoint({ x: 50, y: 50 })
|
|
240
|
+
const shapeIds = shapes.map((s) => s.id)
|
|
241
|
+
|
|
242
|
+
expect(shapeIds).toEqual([ids.shape3])
|
|
243
|
+
expect(shapes).toHaveLength(1)
|
|
244
|
+
})
|
|
245
|
+
|
|
246
|
+
it('returns empty array when no shapes at point', () => {
|
|
247
|
+
const shapes = editor.getShapesAtPoint({ x: 1000, y: 1000 })
|
|
248
|
+
expect(shapes).toEqual([])
|
|
249
|
+
})
|
|
250
|
+
|
|
251
|
+
it('returns single shape when point hits only one shape', () => {
|
|
252
|
+
// Point at right edge of shape2 where it doesn't overlap with other shapes
|
|
253
|
+
// Shape2 is at (100,0) with size 200x200, so right edge is at x=300
|
|
254
|
+
const shapes = editor.getShapesAtPoint({ x: 300, y: 100 })
|
|
255
|
+
expect(shapes).toHaveLength(1)
|
|
256
|
+
expect(shapes[0].id).toBe(ids.shape2)
|
|
257
|
+
})
|
|
258
|
+
|
|
259
|
+
it('returns shapes on edge when point is exactly on boundary', () => {
|
|
260
|
+
// Point at exact edge of shape1
|
|
261
|
+
const shapes = editor.getShapesAtPoint({ x: 0, y: 0 })
|
|
262
|
+
expect(shapes).toHaveLength(1)
|
|
263
|
+
expect(shapes[0].id).toBe(ids.shape1)
|
|
264
|
+
})
|
|
265
|
+
|
|
266
|
+
it('respects hitInside option when false (default)', () => {
|
|
267
|
+
// Point inside shape1 (at 0,0 with size 200x200) but with hitInside false should not hit
|
|
268
|
+
const shapes = editor.getShapesAtPoint({ x: 25, y: 25 }, { hitInside: false })
|
|
269
|
+
expect(shapes).toEqual([])
|
|
270
|
+
})
|
|
271
|
+
|
|
272
|
+
it('respects hitInside option when true', () => {
|
|
273
|
+
// Point inside shape1 (at 0,0 with size 200x200) with hitInside true should hit
|
|
274
|
+
const shapes = editor.getShapesAtPoint({ x: 25, y: 25 }, { hitInside: true })
|
|
275
|
+
expect(shapes).toHaveLength(1)
|
|
276
|
+
expect(shapes[0].id).toBe(ids.shape1)
|
|
277
|
+
})
|
|
278
|
+
|
|
279
|
+
it('respects margin option', () => {
|
|
280
|
+
// Point slightly outside shape1 at bottom edge but within margin should hit only shape1
|
|
281
|
+
// Shape1 is at (0,0) with size 200x200, shape2 goes to (300,200) so avoid overlap at (200,200)
|
|
282
|
+
const shapes = editor.getShapesAtPoint({ x: 205, y: 100 }, { margin: 10 })
|
|
283
|
+
expect(shapes).toHaveLength(1)
|
|
284
|
+
expect(shapes[0].id).toBe(ids.shape1)
|
|
285
|
+
})
|
|
286
|
+
|
|
287
|
+
it('filters out hidden shapes', () => {
|
|
288
|
+
// Create a spy to mock isShapeHidden
|
|
289
|
+
const isShapeHiddenSpy = jest.spyOn(editor, 'isShapeHidden')
|
|
290
|
+
isShapeHiddenSpy.mockImplementation((shape) => {
|
|
291
|
+
return typeof shape === 'string' ? shape === ids.shape3 : shape.id === ids.shape3
|
|
292
|
+
})
|
|
293
|
+
|
|
294
|
+
const shapes = editor.getShapesAtPoint({ x: 50, y: 50 })
|
|
295
|
+
const shapeIds = shapes.map((s) => s.id)
|
|
296
|
+
|
|
297
|
+
// Should not include shape3 since it's hidden, and no other shapes are at this point
|
|
298
|
+
expect(shapeIds).toEqual([])
|
|
299
|
+
expect(shapes).toHaveLength(0)
|
|
300
|
+
|
|
301
|
+
isShapeHiddenSpy.mockRestore()
|
|
302
|
+
})
|
|
303
|
+
|
|
304
|
+
it('handles point exactly at shape corner', () => {
|
|
305
|
+
// Point at bottom-left corner of shape1 where it doesn't overlap with other shapes
|
|
306
|
+
const shapes = editor.getShapesAtPoint({ x: 0, y: 200 })
|
|
307
|
+
expect(shapes).toHaveLength(1)
|
|
308
|
+
expect(shapes[0].id).toBe(ids.shape1)
|
|
309
|
+
})
|
|
310
|
+
|
|
311
|
+
it('handles overlapping shapes with different hit areas', () => {
|
|
312
|
+
// Point that hits both shape1 and shape2 edges (they overlap at x=100,y=0)
|
|
313
|
+
const shapes = editor.getShapesAtPoint({ x: 100, y: 0 })
|
|
314
|
+
const shapeIds = shapes.map((s) => s.id)
|
|
315
|
+
|
|
316
|
+
// Both shapes should be detected at this overlapping point (reversed order - top-most first)
|
|
317
|
+
expect(shapeIds).toEqual([ids.shape2, ids.shape1])
|
|
318
|
+
expect(shapes).toHaveLength(2)
|
|
319
|
+
})
|
|
320
|
+
|
|
321
|
+
it('maintains reverse shape order and responds to z-index changes', () => {
|
|
322
|
+
// Create filled shape that overlaps with shape2
|
|
323
|
+
editor.createShape({
|
|
324
|
+
id: ids.shape5,
|
|
325
|
+
type: 'my-custom-shape',
|
|
326
|
+
x: 110,
|
|
327
|
+
y: 110,
|
|
328
|
+
props: { w: 200, h: 200, isFilled: true, text: 'Shape5' },
|
|
329
|
+
})
|
|
330
|
+
|
|
331
|
+
// Test with hitInside to detect multiple shapes
|
|
332
|
+
// Point (120,120) will hit shape1, shape2, shape3, shape4, and shape5 with hitInside: true
|
|
333
|
+
const shapes = editor.getShapesAtPoint({ x: 120, y: 120 }, { hitInside: true })
|
|
334
|
+
const shapeIds = shapes.map((s) => s.id)
|
|
335
|
+
|
|
336
|
+
// All shapes that contain this point should be returned in reverse z-index order (top-most first)
|
|
337
|
+
expect(shapeIds).toEqual([ids.shape5, ids.shape4, ids.shape3, ids.shape2, ids.shape1])
|
|
338
|
+
|
|
339
|
+
// After bringing shape2 to front, order should change (shape2 becomes top-most)
|
|
340
|
+
editor.bringToFront([ids.shape2])
|
|
341
|
+
const shapes2 = editor.getShapesAtPoint({ x: 120, y: 120 }, { hitInside: true })
|
|
342
|
+
const shapeIds2 = shapes2.map((s) => s.id)
|
|
343
|
+
expect(shapeIds2).toEqual([ids.shape2, ids.shape5, ids.shape4, ids.shape3, ids.shape1])
|
|
344
|
+
})
|
|
345
|
+
|
|
346
|
+
it('combines hitInside and margin options', () => {
|
|
347
|
+
// Point inside shape1 (at 0,0 with size 200x200) with hitInside and margin
|
|
348
|
+
const shapes = editor.getShapesAtPoint({ x: 25, y: 25 }, { hitInside: true, margin: 5 })
|
|
349
|
+
expect(shapes).toHaveLength(1)
|
|
350
|
+
expect(shapes[0].id).toBe(ids.shape1)
|
|
351
|
+
})
|
|
352
|
+
|
|
353
|
+
it('returns empty array when all shapes are hidden', () => {
|
|
354
|
+
// Mock all shapes as hidden
|
|
355
|
+
const isShapeHiddenSpy = jest.spyOn(editor, 'isShapeHidden')
|
|
356
|
+
isShapeHiddenSpy.mockReturnValue(true)
|
|
357
|
+
|
|
358
|
+
const shapes = editor.getShapesAtPoint({ x: 50, y: 50 })
|
|
359
|
+
expect(shapes).toEqual([])
|
|
360
|
+
|
|
361
|
+
isShapeHiddenSpy.mockRestore()
|
|
362
|
+
})
|
|
363
|
+
|
|
364
|
+
it('returns multiple shapes at same point in reverse z-index order', () => {
|
|
365
|
+
// Create two shapes at exactly the same position (away from existing shapes)
|
|
366
|
+
editor.createShape({
|
|
367
|
+
id: ids.overlap1,
|
|
368
|
+
type: 'my-custom-shape',
|
|
369
|
+
x: 600,
|
|
370
|
+
y: 600,
|
|
371
|
+
props: { w: 100, h: 100, text: 'First' },
|
|
372
|
+
})
|
|
373
|
+
|
|
374
|
+
editor.createShape({
|
|
375
|
+
id: ids.overlap2,
|
|
376
|
+
type: 'my-custom-shape',
|
|
377
|
+
x: 600,
|
|
378
|
+
y: 600,
|
|
379
|
+
props: { w: 100, h: 100, text: 'Second' },
|
|
380
|
+
})
|
|
381
|
+
|
|
382
|
+
// Test at corner where both shapes' edges meet
|
|
383
|
+
const shapes = editor.getShapesAtPoint({ x: 600, y: 600 })
|
|
384
|
+
const shapeIds = shapes.map((s) => s.id)
|
|
385
|
+
|
|
386
|
+
// Should return both shapes in reverse z-index order (top-most first)
|
|
387
|
+
expect(shapeIds).toEqual([ids.overlap2, ids.overlap1])
|
|
388
|
+
expect(shapes).toHaveLength(2)
|
|
389
|
+
})
|
|
390
|
+
|
|
391
|
+
it('respects isFilled property for hit detection', () => {
|
|
392
|
+
// Create a filled shape
|
|
393
|
+
editor.createShape({
|
|
394
|
+
id: ids.filledShape,
|
|
395
|
+
type: 'my-custom-shape',
|
|
396
|
+
x: 300,
|
|
397
|
+
y: 300,
|
|
398
|
+
props: { w: 100, h: 100, isFilled: true, text: 'Filled' },
|
|
399
|
+
})
|
|
400
|
+
|
|
401
|
+
// Create a hollow shape at the same position
|
|
402
|
+
editor.createShape({
|
|
403
|
+
id: ids.hollowShape,
|
|
404
|
+
type: 'my-custom-shape',
|
|
405
|
+
x: 400,
|
|
406
|
+
y: 300,
|
|
407
|
+
props: { w: 100, h: 100, isFilled: false, text: 'Hollow' },
|
|
408
|
+
})
|
|
409
|
+
|
|
410
|
+
// Test point inside filled shape - should hit without hitInside option
|
|
411
|
+
const filledShapes = editor.getShapesAtPoint({ x: 350, y: 350 })
|
|
412
|
+
expect(filledShapes).toHaveLength(1)
|
|
413
|
+
expect(filledShapes[0].id).toBe(ids.filledShape)
|
|
414
|
+
|
|
415
|
+
// Test point inside hollow shape - should not hit without hitInside option
|
|
416
|
+
const hollowShapes = editor.getShapesAtPoint({ x: 450, y: 350 })
|
|
417
|
+
expect(hollowShapes).toHaveLength(0)
|
|
418
|
+
|
|
419
|
+
// Test point inside hollow shape with hitInside - should hit
|
|
420
|
+
const hollowShapesWithHitInside = editor.getShapesAtPoint(
|
|
421
|
+
{ x: 450, y: 350 },
|
|
422
|
+
{ hitInside: true }
|
|
423
|
+
)
|
|
424
|
+
expect(hollowShapesWithHitInside).toHaveLength(1)
|
|
425
|
+
expect(hollowShapesWithHitInside[0].id).toBe(ids.hollowShape)
|
|
426
|
+
})
|
|
427
|
+
})
|
package/src/lib/editor/Editor.ts
CHANGED
|
@@ -148,16 +148,16 @@ import { bindingsIndex } from './derivations/bindingsIndex'
|
|
|
148
148
|
import { notVisibleShapes } from './derivations/notVisibleShapes'
|
|
149
149
|
import { parentsToChildren } from './derivations/parentsToChildren'
|
|
150
150
|
import { deriveShapeIdsInCurrentPage } from './derivations/shapeIdsInCurrentPage'
|
|
151
|
-
import { ClickManager } from './managers/ClickManager'
|
|
152
|
-
import { EdgeScrollManager } from './managers/EdgeScrollManager'
|
|
153
|
-
import { FocusManager } from './managers/FocusManager'
|
|
154
|
-
import { FontManager } from './managers/FontManager'
|
|
155
|
-
import { HistoryManager } from './managers/HistoryManager'
|
|
156
|
-
import { ScribbleManager } from './managers/ScribbleManager'
|
|
151
|
+
import { ClickManager } from './managers/ClickManager/ClickManager'
|
|
152
|
+
import { EdgeScrollManager } from './managers/EdgeScrollManager/EdgeScrollManager'
|
|
153
|
+
import { FocusManager } from './managers/FocusManager/FocusManager'
|
|
154
|
+
import { FontManager } from './managers/FontManager/FontManager'
|
|
155
|
+
import { HistoryManager } from './managers/HistoryManager/HistoryManager'
|
|
156
|
+
import { ScribbleManager } from './managers/ScribbleManager/ScribbleManager'
|
|
157
157
|
import { SnapManager } from './managers/SnapManager/SnapManager'
|
|
158
|
-
import { TextManager } from './managers/TextManager'
|
|
159
|
-
import { TickManager } from './managers/TickManager'
|
|
160
|
-
import { UserPreferencesManager } from './managers/UserPreferencesManager'
|
|
158
|
+
import { TextManager } from './managers/TextManager/TextManager'
|
|
159
|
+
import { TickManager } from './managers/TickManager/TickManager'
|
|
160
|
+
import { UserPreferencesManager } from './managers/UserPreferencesManager/UserPreferencesManager'
|
|
161
161
|
import { ShapeUtil, TLGeometryOpts, TLResizeMode } from './shapes/ShapeUtil'
|
|
162
162
|
import { RootState } from './tools/RootState'
|
|
163
163
|
import { StateNode, TLStateNodeConstructor } from './tools/StateNode'
|
|
@@ -328,7 +328,7 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
328
328
|
this.store = store
|
|
329
329
|
this.history = new HistoryManager<TLRecord>({
|
|
330
330
|
store,
|
|
331
|
-
annotateError: (error) => {
|
|
331
|
+
annotateError: (error: any) => {
|
|
332
332
|
this.annotateError(error, { origin: 'history.batch', willCrashApp: true })
|
|
333
333
|
this.crash(error)
|
|
334
334
|
},
|
|
@@ -348,6 +348,8 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
348
348
|
this.getContainer = getContainer
|
|
349
349
|
|
|
350
350
|
this.textMeasure = new TextManager(this)
|
|
351
|
+
this.disposables.add(() => this.textMeasure.dispose())
|
|
352
|
+
|
|
351
353
|
this.fonts = new FontManager(this, fontAssetUrls)
|
|
352
354
|
|
|
353
355
|
this._tickManager = new TickManager(this)
|
|
@@ -506,14 +508,13 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
506
508
|
shape: {
|
|
507
509
|
afterChange: (shapeBefore, shapeAfter) => {
|
|
508
510
|
for (const binding of this.getBindingsInvolvingShape(shapeAfter)) {
|
|
509
|
-
if (areShapesContentEqual(shapeBefore, shapeAfter)) continue
|
|
510
|
-
|
|
511
511
|
invalidBindingTypes.add(binding.type)
|
|
512
512
|
if (binding.fromId === shapeAfter.id) {
|
|
513
513
|
this.getBindingUtil(binding).onAfterChangeFromShape?.({
|
|
514
514
|
binding,
|
|
515
515
|
shapeBefore,
|
|
516
516
|
shapeAfter,
|
|
517
|
+
reason: 'self',
|
|
517
518
|
})
|
|
518
519
|
}
|
|
519
520
|
if (binding.toId === shapeAfter.id) {
|
|
@@ -521,6 +522,7 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
521
522
|
binding,
|
|
522
523
|
shapeBefore,
|
|
523
524
|
shapeAfter,
|
|
525
|
+
reason: 'self',
|
|
524
526
|
})
|
|
525
527
|
}
|
|
526
528
|
}
|
|
@@ -539,6 +541,7 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
539
541
|
binding,
|
|
540
542
|
shapeBefore: descendantShape,
|
|
541
543
|
shapeAfter: descendantShape,
|
|
544
|
+
reason: 'ancestry',
|
|
542
545
|
})
|
|
543
546
|
}
|
|
544
547
|
if (binding.toId === descendantShape.id) {
|
|
@@ -546,6 +549,7 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
546
549
|
binding,
|
|
547
550
|
shapeBefore: descendantShape,
|
|
548
551
|
shapeAfter: descendantShape,
|
|
552
|
+
reason: 'ancestry',
|
|
549
553
|
})
|
|
550
554
|
}
|
|
551
555
|
}
|
|
@@ -2118,6 +2122,20 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
2118
2122
|
return this.getShapesPageBounds(this.getSelectedShapeIds())
|
|
2119
2123
|
}
|
|
2120
2124
|
|
|
2125
|
+
/**
|
|
2126
|
+
* The bounds of the selection bounding box in the current page space.
|
|
2127
|
+
*
|
|
2128
|
+
* @readonly
|
|
2129
|
+
* @public
|
|
2130
|
+
*/
|
|
2131
|
+
@computed getSelectionScreenBounds(): Box | undefined {
|
|
2132
|
+
const bounds = this.getSelectionPageBounds()
|
|
2133
|
+
if (!bounds) return undefined
|
|
2134
|
+
const { x, y } = this.pageToScreen(bounds.point)
|
|
2135
|
+
const zoom = this.getZoomLevel()
|
|
2136
|
+
return new Box(x, y, bounds.width * zoom, bounds.height * zoom)
|
|
2137
|
+
}
|
|
2138
|
+
|
|
2121
2139
|
/**
|
|
2122
2140
|
* @internal
|
|
2123
2141
|
*/
|
|
@@ -4643,44 +4661,6 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
4643
4661
|
)! as T
|
|
4644
4662
|
}
|
|
4645
4663
|
|
|
4646
|
-
private _shapePageGeometryCaches: Record<string, ComputedCache<Geometry2d, TLShape>> = {}
|
|
4647
|
-
|
|
4648
|
-
/**
|
|
4649
|
-
* Get the geometry of a shape in page-space.
|
|
4650
|
-
*
|
|
4651
|
-
* @example
|
|
4652
|
-
* ```ts
|
|
4653
|
-
* editor.getShapePageGeometry(myShape)
|
|
4654
|
-
* editor.getShapePageGeometry(myShapeId)
|
|
4655
|
-
* editor.getShapePageGeometry(myShapeId, { context: "arrow" })
|
|
4656
|
-
* ```
|
|
4657
|
-
*
|
|
4658
|
-
* @param shape - The shape (or shape id) to get the geometry for.
|
|
4659
|
-
* @param opts - Additional options about the request for geometry. Passed to {@link ShapeUtil.getGeometry}.
|
|
4660
|
-
*
|
|
4661
|
-
* @public
|
|
4662
|
-
*/
|
|
4663
|
-
getShapePageGeometry<T extends Geometry2d>(shape: TLShape | TLShapeId, opts?: TLGeometryOpts): T {
|
|
4664
|
-
const context = opts?.context ?? 'none'
|
|
4665
|
-
if (!this._shapePageGeometryCaches[context]) {
|
|
4666
|
-
this._shapePageGeometryCaches[context] = this.store.createComputedCache(
|
|
4667
|
-
'bounds',
|
|
4668
|
-
(shape) => {
|
|
4669
|
-
const geometry = this.getShapeGeometry(shape.id, opts)
|
|
4670
|
-
const pageTransform = this.getShapePageTransform(shape.id)
|
|
4671
|
-
return geometry.transform(pageTransform)
|
|
4672
|
-
},
|
|
4673
|
-
{
|
|
4674
|
-
// we only depend directly on the shape id, and changing geometry/transform will update us anyway
|
|
4675
|
-
areRecordsEqual: () => true,
|
|
4676
|
-
}
|
|
4677
|
-
)
|
|
4678
|
-
}
|
|
4679
|
-
return this._shapePageGeometryCaches[context].get(
|
|
4680
|
-
typeof shape === 'string' ? shape : shape.id
|
|
4681
|
-
)! as T
|
|
4682
|
-
}
|
|
4683
|
-
|
|
4684
4664
|
/** @internal */
|
|
4685
4665
|
@computed private _getShapeHandlesCache(): ComputedCache<TLHandle[] | undefined, TLShape> {
|
|
4686
4666
|
return this.store.createComputedCache(
|
|
@@ -4793,7 +4773,10 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
4793
4773
|
/** @internal */
|
|
4794
4774
|
@computed private _getShapePageBoundsCache(): ComputedCache<Box, TLShape> {
|
|
4795
4775
|
return this.store.createComputedCache<Box, TLShape>('pageBoundsCache', (shape) => {
|
|
4796
|
-
|
|
4776
|
+
const pageTransform = this.getShapePageTransform(shape)
|
|
4777
|
+
if (!pageTransform) return undefined
|
|
4778
|
+
const geometry = this.getShapeGeometry(shape)
|
|
4779
|
+
return Box.FromPoints(pageTransform.applyToPoints(geometry.vertices))
|
|
4797
4780
|
})
|
|
4798
4781
|
}
|
|
4799
4782
|
|
|
@@ -4867,11 +4850,12 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
4867
4850
|
if (frameAncestors.length === 0) return undefined
|
|
4868
4851
|
|
|
4869
4852
|
const pageMask = frameAncestors
|
|
4870
|
-
.map<Vec[] | undefined>(
|
|
4871
|
-
|
|
4872
|
-
|
|
4873
|
-
|
|
4874
|
-
|
|
4853
|
+
.map<Vec[] | undefined>((s) => {
|
|
4854
|
+
// Apply the frame transform to the frame outline to get the frame outline in the current page space
|
|
4855
|
+
const geometry = this.getShapeGeometry(s.id)
|
|
4856
|
+
const pageTransform = this.getShapePageTransform(s.id)
|
|
4857
|
+
return pageTransform.applyToPoints(geometry.vertices)
|
|
4858
|
+
})
|
|
4875
4859
|
.reduce((acc, b) => {
|
|
4876
4860
|
if (!(b && acc)) return undefined
|
|
4877
4861
|
const intersection = intersectPolygonPolygon(acc, b)
|
|
@@ -5069,28 +5053,33 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
5069
5053
|
*
|
|
5070
5054
|
* @public
|
|
5071
5055
|
*/
|
|
5072
|
-
isShapeOrAncestorLocked(shape?: TLShape): boolean
|
|
5073
|
-
|
|
5074
|
-
|
|
5075
|
-
|
|
5076
|
-
|
|
5077
|
-
if (shape.isLocked) return true
|
|
5078
|
-
return this.isShapeOrAncestorLocked(this.getShapeParent(shape))
|
|
5056
|
+
isShapeOrAncestorLocked(shape?: TLShape | TLShapeId): boolean {
|
|
5057
|
+
const _shape = shape && this.getShape(shape)
|
|
5058
|
+
if (_shape === undefined) return false
|
|
5059
|
+
if (_shape.isLocked) return true
|
|
5060
|
+
return this.isShapeOrAncestorLocked(this.getShapeParent(_shape))
|
|
5079
5061
|
}
|
|
5080
5062
|
|
|
5063
|
+
/**
|
|
5064
|
+
* Get shapes that are outside of the viewport.
|
|
5065
|
+
*
|
|
5066
|
+
* @public
|
|
5067
|
+
*/
|
|
5081
5068
|
@computed
|
|
5082
|
-
|
|
5083
|
-
return
|
|
5069
|
+
getNotVisibleShapes() {
|
|
5070
|
+
return this._notVisibleShapes.get()
|
|
5084
5071
|
}
|
|
5085
5072
|
|
|
5073
|
+
private _notVisibleShapes = notVisibleShapes(this)
|
|
5074
|
+
|
|
5086
5075
|
/**
|
|
5087
|
-
* Get culled shapes.
|
|
5076
|
+
* Get culled shapes (those that should not render), taking into account which shapes are selected or editing.
|
|
5088
5077
|
*
|
|
5089
5078
|
* @public
|
|
5090
5079
|
*/
|
|
5091
5080
|
@computed
|
|
5092
5081
|
getCulledShapes() {
|
|
5093
|
-
const notVisibleShapes = this.
|
|
5082
|
+
const notVisibleShapes = this.getNotVisibleShapes()
|
|
5094
5083
|
const selectedShapeIds = this.getSelectedShapeIds()
|
|
5095
5084
|
const editingId = this.getEditingShapeId()
|
|
5096
5085
|
const culledShapes = new Set<TLShapeId>(notVisibleShapes)
|
|
@@ -5339,21 +5328,23 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
5339
5328
|
* @example
|
|
5340
5329
|
* ```ts
|
|
5341
5330
|
* editor.getShapesAtPoint({ x: 100, y: 100 })
|
|
5342
|
-
* editor.getShapesAtPoint({ x: 100, y: 100 }, { hitInside: true,
|
|
5331
|
+
* editor.getShapesAtPoint({ x: 100, y: 100 }, { hitInside: true, margin: 8 })
|
|
5343
5332
|
* ```
|
|
5344
5333
|
*
|
|
5345
5334
|
* @param point - The page point to test.
|
|
5346
5335
|
* @param opts - The options for the hit point testing.
|
|
5347
5336
|
*
|
|
5337
|
+
* @returns An array of shapes at the given point, sorted in reverse order of their absolute z-index (top-most shape first).
|
|
5338
|
+
*
|
|
5348
5339
|
* @public
|
|
5349
5340
|
*/
|
|
5350
5341
|
getShapesAtPoint(
|
|
5351
5342
|
point: VecLike,
|
|
5352
5343
|
opts = {} as { margin?: number; hitInside?: boolean }
|
|
5353
5344
|
): TLShape[] {
|
|
5354
|
-
return this.
|
|
5355
|
-
(shape) => !this.isShapeHidden(shape) && this.isPointInShape(shape, point, opts)
|
|
5356
|
-
|
|
5345
|
+
return this.getCurrentPageShapesSorted()
|
|
5346
|
+
.filter((shape) => !this.isShapeHidden(shape) && this.isPointInShape(shape, point, opts))
|
|
5347
|
+
.reverse()
|
|
5357
5348
|
}
|
|
5358
5349
|
|
|
5359
5350
|
/**
|
|
@@ -8836,6 +8827,7 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
8836
8827
|
} = {
|
|
8837
8828
|
text: null,
|
|
8838
8829
|
files: null,
|
|
8830
|
+
'file-replace': null,
|
|
8839
8831
|
embed: null,
|
|
8840
8832
|
'svg-text': null,
|
|
8841
8833
|
url: null,
|
|
@@ -8885,6 +8877,15 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
8885
8877
|
return this.externalContentHandlers[info.type]?.(info as any)
|
|
8886
8878
|
}
|
|
8887
8879
|
|
|
8880
|
+
/**
|
|
8881
|
+
* Handle replacing external content.
|
|
8882
|
+
*
|
|
8883
|
+
* @param info - Info about the external content.
|
|
8884
|
+
*/
|
|
8885
|
+
async replaceExternalContent<E>(info: TLExternalContent<E>): Promise<void> {
|
|
8886
|
+
return this.externalContentHandlers[info.type]?.(info as any)
|
|
8887
|
+
}
|
|
8888
|
+
|
|
8888
8889
|
/**
|
|
8889
8890
|
* Get content that can be exported for the given shape ids.
|
|
8890
8891
|
*
|
|
@@ -9302,6 +9303,7 @@ export class Editor extends EventEmitter<TLEventMap> {
|
|
|
9302
9303
|
if (rootShapes.length === 1) {
|
|
9303
9304
|
const onlyRoot = rootShapes[0] as TLFrameShape
|
|
9304
9305
|
// If the old bounds are in the viewport...
|
|
9306
|
+
// todo: replace frame references with shapes that can accept children
|
|
9305
9307
|
if (this.isShapeOfType<TLFrameShape>(onlyRoot, 'frame')) {
|
|
9306
9308
|
while (
|
|
9307
9309
|
this.getShapesAtPoint(point).some(
|