@tldraw/tlschema 4.2.1 → 4.2.2

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.
Files changed (117) hide show
  1. package/dist-cjs/bindings/TLBaseBinding.js.map +2 -2
  2. package/dist-cjs/createTLSchema.js.map +2 -2
  3. package/dist-cjs/index.d.ts +242 -71
  4. package/dist-cjs/index.js +4 -1
  5. package/dist-cjs/index.js.map +2 -2
  6. package/dist-cjs/misc/TLOpacity.js +1 -5
  7. package/dist-cjs/misc/TLOpacity.js.map +2 -2
  8. package/dist-cjs/misc/TLRichText.js +5 -1
  9. package/dist-cjs/misc/TLRichText.js.map +2 -2
  10. package/dist-cjs/misc/b64Vecs.js +224 -0
  11. package/dist-cjs/misc/b64Vecs.js.map +7 -0
  12. package/dist-cjs/records/TLAsset.js.map +1 -1
  13. package/dist-cjs/records/TLBinding.js.map +2 -2
  14. package/dist-cjs/records/TLShape.js.map +2 -2
  15. package/dist-cjs/shapes/ShapeWithCrop.js.map +1 -1
  16. package/dist-cjs/shapes/TLArrowShape.js +26 -13
  17. package/dist-cjs/shapes/TLArrowShape.js.map +2 -2
  18. package/dist-cjs/shapes/TLBaseShape.js.map +2 -2
  19. package/dist-cjs/shapes/TLDrawShape.js +37 -4
  20. package/dist-cjs/shapes/TLDrawShape.js.map +2 -2
  21. package/dist-cjs/shapes/TLEmbedShape.js +17 -0
  22. package/dist-cjs/shapes/TLEmbedShape.js.map +2 -2
  23. package/dist-cjs/shapes/TLGeoShape.js +12 -1
  24. package/dist-cjs/shapes/TLGeoShape.js.map +2 -2
  25. package/dist-cjs/shapes/TLHighlightShape.js +29 -2
  26. package/dist-cjs/shapes/TLHighlightShape.js.map +2 -2
  27. package/dist-cjs/shapes/TLNoteShape.js +12 -1
  28. package/dist-cjs/shapes/TLNoteShape.js.map +2 -2
  29. package/dist-cjs/shapes/TLTextShape.js +12 -1
  30. package/dist-cjs/shapes/TLTextShape.js.map +2 -2
  31. package/dist-cjs/store-migrations.js +15 -15
  32. package/dist-cjs/store-migrations.js.map +2 -2
  33. package/dist-esm/bindings/TLBaseBinding.mjs.map +2 -2
  34. package/dist-esm/createTLSchema.mjs.map +2 -2
  35. package/dist-esm/index.d.mts +242 -71
  36. package/dist-esm/index.mjs +5 -1
  37. package/dist-esm/index.mjs.map +2 -2
  38. package/dist-esm/misc/TLOpacity.mjs +1 -5
  39. package/dist-esm/misc/TLOpacity.mjs.map +2 -2
  40. package/dist-esm/misc/TLRichText.mjs +5 -1
  41. package/dist-esm/misc/TLRichText.mjs.map +2 -2
  42. package/dist-esm/misc/b64Vecs.mjs +204 -0
  43. package/dist-esm/misc/b64Vecs.mjs.map +7 -0
  44. package/dist-esm/records/TLAsset.mjs.map +1 -1
  45. package/dist-esm/records/TLBinding.mjs.map +2 -2
  46. package/dist-esm/records/TLShape.mjs.map +2 -2
  47. package/dist-esm/shapes/TLArrowShape.mjs +26 -13
  48. package/dist-esm/shapes/TLArrowShape.mjs.map +2 -2
  49. package/dist-esm/shapes/TLBaseShape.mjs.map +2 -2
  50. package/dist-esm/shapes/TLDrawShape.mjs +37 -4
  51. package/dist-esm/shapes/TLDrawShape.mjs.map +2 -2
  52. package/dist-esm/shapes/TLEmbedShape.mjs +17 -0
  53. package/dist-esm/shapes/TLEmbedShape.mjs.map +2 -2
  54. package/dist-esm/shapes/TLGeoShape.mjs +12 -1
  55. package/dist-esm/shapes/TLGeoShape.mjs.map +2 -2
  56. package/dist-esm/shapes/TLHighlightShape.mjs +29 -2
  57. package/dist-esm/shapes/TLHighlightShape.mjs.map +2 -2
  58. package/dist-esm/shapes/TLNoteShape.mjs +12 -1
  59. package/dist-esm/shapes/TLNoteShape.mjs.map +2 -2
  60. package/dist-esm/shapes/TLTextShape.mjs +12 -1
  61. package/dist-esm/shapes/TLTextShape.mjs.map +2 -2
  62. package/dist-esm/store-migrations.mjs +15 -15
  63. package/dist-esm/store-migrations.mjs.map +2 -2
  64. package/package.json +8 -8
  65. package/src/__tests__/migrationTestUtils.ts +9 -3
  66. package/src/bindings/TLBaseBinding.ts +25 -14
  67. package/src/createTLSchema.ts +8 -2
  68. package/src/index.ts +9 -0
  69. package/src/migrations.test.ts +149 -1
  70. package/src/misc/TLOpacity.ts +1 -5
  71. package/src/misc/TLRichText.ts +6 -1
  72. package/src/misc/b64Vecs.ts +308 -0
  73. package/src/records/TLAsset.ts +2 -2
  74. package/src/records/TLBinding.ts +65 -23
  75. package/src/records/TLShape.ts +100 -5
  76. package/src/shapes/ShapeWithCrop.ts +2 -2
  77. package/src/shapes/TLArrowShape.ts +28 -14
  78. package/src/shapes/TLBaseShape.ts +34 -10
  79. package/src/shapes/TLDrawShape.ts +59 -12
  80. package/src/shapes/TLEmbedShape.ts +17 -0
  81. package/src/shapes/TLGeoShape.ts +14 -1
  82. package/src/shapes/TLHighlightShape.ts +37 -0
  83. package/src/shapes/TLNoteShape.ts +15 -1
  84. package/src/shapes/TLTextShape.ts +16 -2
  85. package/src/store-migrations.ts +17 -16
  86. package/src/assets/TLBookmarkAsset.test.ts +0 -96
  87. package/src/assets/TLImageAsset.test.ts +0 -213
  88. package/src/assets/TLVideoAsset.test.ts +0 -105
  89. package/src/bindings/TLArrowBinding.test.ts +0 -55
  90. package/src/misc/id-validator.test.ts +0 -50
  91. package/src/records/TLAsset.test.ts +0 -234
  92. package/src/records/TLBinding.test.ts +0 -22
  93. package/src/records/TLCamera.test.ts +0 -19
  94. package/src/records/TLDocument.test.ts +0 -35
  95. package/src/records/TLInstance.test.ts +0 -201
  96. package/src/records/TLPage.test.ts +0 -110
  97. package/src/records/TLPageState.test.ts +0 -228
  98. package/src/records/TLPointer.test.ts +0 -63
  99. package/src/records/TLPresence.test.ts +0 -190
  100. package/src/records/TLRecord.test.ts +0 -70
  101. package/src/records/TLShape.test.ts +0 -232
  102. package/src/shapes/ShapeWithCrop.test.ts +0 -18
  103. package/src/shapes/TLArrowShape.test.ts +0 -505
  104. package/src/shapes/TLBaseShape.test.ts +0 -142
  105. package/src/shapes/TLBookmarkShape.test.ts +0 -122
  106. package/src/shapes/TLDrawShape.test.ts +0 -177
  107. package/src/shapes/TLEmbedShape.test.ts +0 -286
  108. package/src/shapes/TLFrameShape.test.ts +0 -71
  109. package/src/shapes/TLGeoShape.test.ts +0 -247
  110. package/src/shapes/TLGroupShape.test.ts +0 -59
  111. package/src/shapes/TLHighlightShape.test.ts +0 -325
  112. package/src/shapes/TLImageShape.test.ts +0 -534
  113. package/src/shapes/TLLineShape.test.ts +0 -269
  114. package/src/shapes/TLNoteShape.test.ts +0 -1568
  115. package/src/shapes/TLTextShape.test.ts +0 -407
  116. package/src/shapes/TLVideoShape.test.ts +0 -112
  117. package/src/styles/TLColorStyle.test.ts +0 -439
@@ -1,534 +0,0 @@
1
- import { T } from '@tldraw/validate'
2
- import { describe, expect, it } from 'vitest'
3
- import { getTestMigration } from '../__tests__/migrationTestUtils'
4
- import { TLAssetId } from '../records/TLAsset'
5
- import { TLShapeCrop } from './ShapeWithCrop'
6
- import { ImageShapeCrop, imageShapeProps, imageShapeVersions } from './TLImageShape'
7
-
8
- describe('TLImageShape', () => {
9
- describe('ImageShapeCrop validator', () => {
10
- it('should validate valid crop data', () => {
11
- const validCrop: TLShapeCrop = {
12
- topLeft: { x: 0.1, y: 0.1 },
13
- bottomRight: { x: 0.9, y: 0.9 },
14
- }
15
-
16
- expect(() => ImageShapeCrop.validate(validCrop)).not.toThrow()
17
- const result = ImageShapeCrop.validate(validCrop)
18
- expect(result.topLeft).toEqual({ x: 0.1, y: 0.1 })
19
- expect(result.bottomRight).toEqual({ x: 0.9, y: 0.9 })
20
- })
21
-
22
- it('should validate crop data with isCircle flag', () => {
23
- const cropWithCircle: TLShapeCrop = {
24
- topLeft: { x: 0, y: 0 },
25
- bottomRight: { x: 1, y: 1 },
26
- isCircle: true,
27
- }
28
-
29
- expect(() => ImageShapeCrop.validate(cropWithCircle)).not.toThrow()
30
- const result = ImageShapeCrop.validate(cropWithCircle)
31
- expect(result.isCircle).toBe(true)
32
- })
33
-
34
- it('should validate crop data without isCircle flag', () => {
35
- const cropWithoutCircle: TLShapeCrop = {
36
- topLeft: { x: 0.25, y: 0.25 },
37
- bottomRight: { x: 0.75, y: 0.75 },
38
- }
39
-
40
- expect(() => ImageShapeCrop.validate(cropWithoutCircle)).not.toThrow()
41
- const result = ImageShapeCrop.validate(cropWithoutCircle)
42
- expect(result.isCircle).toBeUndefined()
43
- })
44
-
45
- it('should validate crop data with explicit false isCircle', () => {
46
- const cropWithFalseCircle: TLShapeCrop = {
47
- topLeft: { x: 0, y: 0.5 },
48
- bottomRight: { x: 1, y: 1 },
49
- isCircle: false,
50
- }
51
-
52
- expect(() => ImageShapeCrop.validate(cropWithFalseCircle)).not.toThrow()
53
- const result = ImageShapeCrop.validate(cropWithFalseCircle)
54
- expect(result.isCircle).toBe(false)
55
- })
56
-
57
- it('should validate edge case coordinate values', () => {
58
- const edgeCaseCrops = [
59
- {
60
- topLeft: { x: 0, y: 0 },
61
- bottomRight: { x: 1, y: 1 },
62
- },
63
- {
64
- topLeft: { x: 0.5, y: 0.5 },
65
- bottomRight: { x: 0.5, y: 0.5 },
66
- },
67
- {
68
- topLeft: { x: 0.001, y: 0.999 },
69
- bottomRight: { x: 0.999, y: 0.001 },
70
- },
71
- ]
72
-
73
- edgeCaseCrops.forEach((crop, _index) => {
74
- expect(() => ImageShapeCrop.validate(crop)).not.toThrow()
75
- })
76
- })
77
-
78
- it('should reject invalid crop data structures', () => {
79
- const invalidCrops = [
80
- {}, // Missing required properties
81
- { topLeft: { x: 0.1, y: 0.1 } }, // Missing bottomRight
82
- { bottomRight: { x: 0.9, y: 0.9 } }, // Missing topLeft
83
- {
84
- topLeft: { x: 'invalid', y: 0.1 },
85
- bottomRight: { x: 0.9, y: 0.9 },
86
- }, // Invalid coordinate type
87
- {
88
- topLeft: { x: 0.1, y: 0.1 },
89
- bottomRight: { x: 0.9, y: 'invalid' },
90
- }, // Invalid coordinate type
91
- {
92
- topLeft: { x: 0.1 }, // Missing y
93
- bottomRight: { x: 0.9, y: 0.9 },
94
- },
95
- {
96
- topLeft: { x: 0.1, y: 0.1 },
97
- bottomRight: { y: 0.9 }, // Missing x
98
- },
99
- {
100
- topLeft: { x: 0.1, y: 0.1 },
101
- bottomRight: { x: 0.9, y: 0.9 },
102
- isCircle: 'not-boolean', // Invalid isCircle type
103
- },
104
- null,
105
- undefined,
106
- 'not-an-object',
107
- 123,
108
- [],
109
- ]
110
-
111
- invalidCrops.forEach((crop) => {
112
- expect(() => ImageShapeCrop.validate(crop)).toThrow()
113
- })
114
- })
115
- })
116
-
117
- describe('imageShapeMigrations - AddUrlProp migration', () => {
118
- const { up, down } = getTestMigration(imageShapeVersions.AddUrlProp)
119
-
120
- describe('AddUrlProp up migration', () => {
121
- it('should add url property with empty string default', () => {
122
- const oldRecord = {
123
- id: 'shape:image1',
124
- typeName: 'shape',
125
- type: 'image',
126
- x: 100,
127
- y: 200,
128
- rotation: 0,
129
- index: 'a1',
130
- parentId: 'page:main',
131
- isLocked: false,
132
- opacity: 1,
133
- props: {
134
- w: 400,
135
- h: 300,
136
- playing: true,
137
- assetId: 'asset:image123',
138
- },
139
- meta: {},
140
- }
141
-
142
- const result = up(oldRecord)
143
- expect(result.props.url).toBe('')
144
- expect(result.props.w).toBe(400) // Preserve other props
145
- expect(result.props.playing).toBe(true)
146
- })
147
-
148
- it('should preserve all existing properties during migration', () => {
149
- const oldRecord = {
150
- id: 'shape:image2',
151
- props: {
152
- w: 500,
153
- h: 400,
154
- playing: false,
155
- assetId: null,
156
- },
157
- }
158
-
159
- const result = up(oldRecord)
160
- expect(result.props.url).toBe('')
161
- expect(result.props.w).toBe(500)
162
- expect(result.props.h).toBe(400)
163
- expect(result.props.playing).toBe(false)
164
- expect(result.props.assetId).toBeNull()
165
- })
166
- })
167
-
168
- describe('AddUrlProp down migration', () => {
169
- it('should be retired (no down migration)', () => {
170
- expect(() => {
171
- down({})
172
- }).toThrow('Migration com.tldraw.shape.image/1 does not have a down function')
173
- })
174
- })
175
- })
176
-
177
- describe('imageShapeMigrations - AddCropProp migration', () => {
178
- const { up, down } = getTestMigration(imageShapeVersions.AddCropProp)
179
-
180
- describe('AddCropProp up migration', () => {
181
- it('should add crop property with null default', () => {
182
- const oldRecord = {
183
- id: 'shape:image1',
184
- props: {
185
- w: 300,
186
- h: 200,
187
- playing: true,
188
- url: 'https://example.com/image.jpg',
189
- assetId: 'asset:image123',
190
- },
191
- }
192
-
193
- const result = up(oldRecord)
194
- expect(result.props.crop).toBeNull()
195
- expect(result.props.w).toBe(300) // Preserve other props
196
- expect(result.props.url).toBe('https://example.com/image.jpg')
197
- })
198
-
199
- it('should preserve all existing properties during migration', () => {
200
- const oldRecord = {
201
- id: 'shape:image2',
202
- props: {
203
- w: 400,
204
- h: 300,
205
- playing: false,
206
- url: '',
207
- assetId: null,
208
- },
209
- }
210
-
211
- const result = up(oldRecord)
212
- expect(result.props.crop).toBeNull()
213
- expect(result.props.w).toBe(400)
214
- expect(result.props.playing).toBe(false)
215
- })
216
- })
217
-
218
- describe('AddCropProp down migration', () => {
219
- it('should remove crop property', () => {
220
- const newRecord = {
221
- id: 'shape:image1',
222
- props: {
223
- w: 300,
224
- h: 200,
225
- playing: true,
226
- url: 'https://example.com/image.jpg',
227
- assetId: 'asset:image123',
228
- crop: {
229
- topLeft: { x: 0.1, y: 0.1 },
230
- bottomRight: { x: 0.9, y: 0.9 },
231
- },
232
- },
233
- }
234
-
235
- const result = down(newRecord)
236
- expect(result.props.crop).toBeUndefined()
237
- expect(result.props.w).toBe(300) // Preserve other props
238
- })
239
- })
240
- })
241
-
242
- describe('imageShapeMigrations - MakeUrlsValid migration', () => {
243
- const { up, down } = getTestMigration(imageShapeVersions.MakeUrlsValid)
244
-
245
- describe('MakeUrlsValid up migration', () => {
246
- it('should clear invalid URLs', () => {
247
- const oldRecord = {
248
- id: 'shape:image1',
249
- props: {
250
- w: 300,
251
- h: 200,
252
- playing: true,
253
- url: 'invalid-url-format',
254
- assetId: 'asset:image123',
255
- crop: null,
256
- },
257
- }
258
-
259
- const result = up(oldRecord)
260
- expect(result.props.url).toBe('')
261
- expect(result.props.w).toBe(300) // Preserve other props
262
- })
263
-
264
- it('should preserve valid URLs', () => {
265
- const validUrls = [
266
- '',
267
- 'https://example.com/image.jpg',
268
- 'http://test.com/photo.png',
269
- 'https://subdomain.example.com/path/image.gif',
270
- ]
271
-
272
- validUrls.forEach((url) => {
273
- const oldRecord = {
274
- id: 'shape:image1',
275
- props: {
276
- w: 300,
277
- h: 200,
278
- playing: true,
279
- url,
280
- assetId: 'asset:image123',
281
- crop: null,
282
- },
283
- }
284
-
285
- const result = up(oldRecord)
286
- expect(result.props.url).toBe(url)
287
- })
288
- })
289
-
290
- it('should preserve all other properties during migration', () => {
291
- const oldRecord = {
292
- id: 'shape:image1',
293
- props: {
294
- w: 400,
295
- h: 300,
296
- playing: false,
297
- url: 'not-valid-url',
298
- assetId: null,
299
- crop: {
300
- topLeft: { x: 0.2, y: 0.2 },
301
- bottomRight: { x: 0.8, y: 0.8 },
302
- },
303
- },
304
- }
305
-
306
- const result = up(oldRecord)
307
- expect(result.props.url).toBe('')
308
- expect(result.props.w).toBe(400)
309
- expect(result.props.playing).toBe(false)
310
- expect(result.props.crop).toEqual({
311
- topLeft: { x: 0.2, y: 0.2 },
312
- bottomRight: { x: 0.8, y: 0.8 },
313
- })
314
- })
315
- })
316
-
317
- describe('MakeUrlsValid down migration', () => {
318
- it('should be a no-op migration', () => {
319
- const newRecord = {
320
- id: 'shape:image1',
321
- props: {
322
- w: 300,
323
- h: 200,
324
- playing: true,
325
- url: 'https://example.com/image.jpg',
326
- assetId: 'asset:image123',
327
- crop: null,
328
- },
329
- }
330
-
331
- const result = down(newRecord)
332
- expect(result).toEqual(newRecord)
333
- })
334
- })
335
- })
336
-
337
- describe('imageShapeMigrations - AddFlipProps migration', () => {
338
- const { up, down } = getTestMigration(imageShapeVersions.AddFlipProps)
339
-
340
- describe('AddFlipProps up migration', () => {
341
- it('should add flipX and flipY properties with false defaults', () => {
342
- const oldRecord = {
343
- id: 'shape:image1',
344
- props: {
345
- w: 300,
346
- h: 200,
347
- playing: true,
348
- url: 'https://example.com/image.jpg',
349
- assetId: 'asset:image123',
350
- crop: null,
351
- },
352
- }
353
-
354
- const result = up(oldRecord)
355
- expect(result.props.flipX).toBe(false)
356
- expect(result.props.flipY).toBe(false)
357
- expect(result.props.w).toBe(300) // Preserve other props
358
- })
359
-
360
- it('should preserve all existing properties during migration', () => {
361
- const oldRecord = {
362
- id: 'shape:image2',
363
- props: {
364
- w: 400,
365
- h: 300,
366
- playing: false,
367
- url: '',
368
- assetId: null,
369
- crop: {
370
- topLeft: { x: 0, y: 0 },
371
- bottomRight: { x: 1, y: 1 },
372
- isCircle: true,
373
- },
374
- },
375
- }
376
-
377
- const result = up(oldRecord)
378
- expect(result.props.flipX).toBe(false)
379
- expect(result.props.flipY).toBe(false)
380
- expect(result.props.w).toBe(400)
381
- expect(result.props.crop?.isCircle).toBe(true)
382
- })
383
- })
384
-
385
- describe('AddFlipProps down migration', () => {
386
- it('should remove flipX and flipY properties', () => {
387
- const newRecord = {
388
- id: 'shape:image1',
389
- props: {
390
- w: 300,
391
- h: 200,
392
- playing: true,
393
- url: 'https://example.com/image.jpg',
394
- assetId: 'asset:image123',
395
- crop: null,
396
- flipX: true,
397
- flipY: false,
398
- },
399
- }
400
-
401
- const result = down(newRecord)
402
- expect(result.props.flipX).toBeUndefined()
403
- expect(result.props.flipY).toBeUndefined()
404
- expect(result.props.w).toBe(300) // Preserve other props
405
- })
406
- })
407
- })
408
-
409
- describe('imageShapeMigrations - AddAltText migration', () => {
410
- const { up, down } = getTestMigration(imageShapeVersions.AddAltText)
411
-
412
- describe('AddAltText up migration', () => {
413
- it('should add altText property with empty string default', () => {
414
- const oldRecord = {
415
- id: 'shape:image1',
416
- props: {
417
- w: 300,
418
- h: 200,
419
- playing: true,
420
- url: 'https://example.com/image.jpg',
421
- assetId: 'asset:image123',
422
- crop: null,
423
- flipX: false,
424
- flipY: true,
425
- },
426
- }
427
-
428
- const result = up(oldRecord)
429
- expect(result.props.altText).toBe('')
430
- expect(result.props.flipY).toBe(true) // Preserve other props
431
- })
432
-
433
- it('should preserve all existing properties during migration', () => {
434
- const oldRecord = {
435
- id: 'shape:image2',
436
- props: {
437
- w: 500,
438
- h: 400,
439
- playing: false,
440
- url: '',
441
- assetId: null,
442
- crop: {
443
- topLeft: { x: 0.25, y: 0.25 },
444
- bottomRight: { x: 0.75, y: 0.75 },
445
- },
446
- flipX: true,
447
- flipY: false,
448
- },
449
- }
450
-
451
- const result = up(oldRecord)
452
- expect(result.props.altText).toBe('')
453
- expect(result.props.flipX).toBe(true)
454
- expect(result.props.crop).toEqual({
455
- topLeft: { x: 0.25, y: 0.25 },
456
- bottomRight: { x: 0.75, y: 0.75 },
457
- })
458
- })
459
- })
460
-
461
- describe('AddAltText down migration', () => {
462
- it('should remove altText property', () => {
463
- const newRecord = {
464
- id: 'shape:image1',
465
- props: {
466
- w: 300,
467
- h: 200,
468
- playing: true,
469
- url: 'https://example.com/image.jpg',
470
- assetId: 'asset:image123',
471
- crop: null,
472
- flipX: false,
473
- flipY: false,
474
- altText: 'Sample image description',
475
- },
476
- }
477
-
478
- const result = down(newRecord)
479
- expect(result.props.altText).toBeUndefined()
480
- expect(result.props.flipX).toBe(false) // Preserve other props
481
- })
482
- })
483
- })
484
-
485
- describe('integration tests', () => {
486
- it('should work with complete image shape record validation', () => {
487
- const completeValidator = T.object({
488
- id: T.string,
489
- typeName: T.literal('shape'),
490
- type: T.literal('image'),
491
- x: T.number,
492
- y: T.number,
493
- rotation: T.number,
494
- index: T.string,
495
- parentId: T.string,
496
- isLocked: T.boolean,
497
- opacity: T.number,
498
- props: T.object(imageShapeProps),
499
- meta: T.jsonValue,
500
- })
501
-
502
- const validImageShape = {
503
- id: 'shape:image123',
504
- typeName: 'shape' as const,
505
- type: 'image' as const,
506
- x: 100,
507
- y: 200,
508
- rotation: 0.5,
509
- index: 'a1',
510
- parentId: 'page:main',
511
- isLocked: false,
512
- opacity: 0.8,
513
- props: {
514
- w: 400,
515
- h: 300,
516
- playing: true,
517
- url: 'https://example.com/image.jpg',
518
- assetId: 'asset:image123' as TLAssetId,
519
- crop: {
520
- topLeft: { x: 0.1, y: 0.1 },
521
- bottomRight: { x: 0.9, y: 0.9 },
522
- isCircle: false,
523
- },
524
- flipX: false,
525
- flipY: true,
526
- altText: 'Sample image',
527
- },
528
- meta: { custom: 'data' },
529
- }
530
-
531
- expect(() => completeValidator.validate(validImageShape)).not.toThrow()
532
- })
533
- })
534
- })