From cb838882342146df51cfae3a5afeeaba04d92bdf Mon Sep 17 00:00:00 2001 From: Peter Date: Wed, 19 Jun 2024 16:48:27 +0200 Subject: [PATCH 01/20] test: add insert/remove tests --- src/core/nodeOps.test.ts | 313 ++++++++++++++++++++++++++++++++++++++- 1 file changed, 312 insertions(+), 1 deletion(-) diff --git a/src/core/nodeOps.test.ts b/src/core/nodeOps.test.ts index 3ec90aebf..f298a95e9 100644 --- a/src/core/nodeOps.test.ts +++ b/src/core/nodeOps.test.ts @@ -185,6 +185,50 @@ describe('nodeOps', () => { expect(parent.children.includes(child)).toBeTruthy() }) + describe('primitive :object', () => { + describe('into mesh', () => { + it.skip('inserts a mesh :object', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const object = new THREE.Mesh() + const primitive = nodeOps.createElement('primitive', undefined, undefined, { object }) + + expect(parent.material.uuid).not.toBe(object.uuid) + nodeOps.insert(primitive, parent) + expect(parent.material.uuid).toBe(object.uuid) + }) + + it.skip('inserts a material :object', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const object = new THREE.MeshNormalMaterial() + const primitive = nodeOps.createElement('primitive', undefined, undefined, { object }) + + expect(parent.material.uuid).not.toBe(object.uuid) + nodeOps.insert(primitive, parent) + expect(parent.material.uuid).toBe(object.uuid) + }) + + it.skip('inserts a geometry :object', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const object = new THREE.BoxGeometry() + const primitive = nodeOps.createElement('primitive', undefined, undefined, { object }) + + expect(parent.material.uuid).not.toBe(object.uuid) + nodeOps.insert(primitive, parent) + expect(parent.material.uuid).toBe(object.uuid) + }) + + it.skip('inserts a group :object', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const object = new THREE.Group() + const primitive = nodeOps.createElement('primitive', undefined, undefined, { object }) + + expect(parent.material.uuid).not.toBe(object.uuid) + nodeOps.insert(primitive, parent) + expect(parent.material.uuid).toBe(object.uuid) + }) + }) + }) + it('does not insert a falsy child', () => { const parent = nodeOps.createElement('Object3D', undefined, undefined, {}) for (const falsyChild of [undefined, null]) { @@ -228,7 +272,7 @@ describe('nodeOps', () => { } }) - it('calls dispose on materials', () => { + it('calls dispose on a material', () => { const parent = mockTresObjectRootInObject(nodeOps.createElement('Mesh', undefined, undefined, {})) const material = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, {}) const spy = vi.spyOn(material, 'dispose') @@ -237,6 +281,18 @@ describe('nodeOps', () => { expect(spy).toHaveBeenCalledOnce() }) + it('calls dispose on a material array', () => { + const parent = mockTresObjectRootInObject(nodeOps.createElement('Mesh', undefined, undefined, {})) + const material0 = new THREE.MeshNormalMaterial() + const material1 = new THREE.MeshNormalMaterial() + const spy0 = vi.spyOn(material0, 'dispose') + const spy1 = vi.spyOn(material1, 'dispose') + parent.material = [material0, material1] + nodeOps.remove(parent) + expect(spy0).toHaveBeenCalledOnce() + expect(spy1).toHaveBeenCalledOnce() + }) + it('calls dispose on geometries', () => { const parent = mockTresObjectRootInObject(nodeOps.createElement('Mesh', undefined, undefined, {})) const geometry = nodeOps.createElement('SphereGeometry', undefined, undefined, {}) @@ -245,6 +301,220 @@ describe('nodeOps', () => { nodeOps.remove(parent) expect(spy).toHaveBeenCalledOnce() }) + + it('calls dispose on every material/geometry in a TresMesh tree', () => { + const NUM_LEVEL = 5 + const NUM_CHILD_PER_NODE = 3 + const rootNode = mockTresObjectRootInObject(nodeOps.createElement('Mesh')) + const disposalSpies = [] + + createTreeIn(rootNode, (parent, childI, levelI) => { + if (levelI > NUM_LEVEL || childI >= NUM_CHILD_PER_NODE) { + return false + } + const { mesh, material, geometry } = createElementMesh(nodeOps) + nodeOps.insert(mesh, parent) + disposalSpies.push(vi.spyOn(geometry, 'dispose')) + disposalSpies.push(vi.spyOn(material, 'dispose')) + return mesh + }) + + nodeOps.remove(rootNode) + for (const spy of disposalSpies) { + expect(spy).toHaveBeenCalledOnce() + } + }) + + it('calls dispose on every material/geometry in a TresMesh/TresGroup tree', () => { + const NUM_LEVEL = 5 + const NUM_CHILD_PER_NODE = 3 + const rootNode = mockTresObjectRootInObject(nodeOps.createElement('Group')) + const disposalSpies = [] + + createTreeIn(rootNode, (parent, childI, levelI) => { + if (childI > NUM_CHILD_PER_NODE || levelI > NUM_LEVEL) { + return false + } + if (Math.random() > 0.3) { + const { mesh, material, geometry } = createElementMesh(nodeOps) + nodeOps.insert(mesh, parent) + disposalSpies.push(vi.spyOn(geometry, 'dispose')) + disposalSpies.push(vi.spyOn(material, 'dispose')) + return mesh + } + else { + const group = nodeOps.createElement('Group') + nodeOps.insert(group, parent) + return group + } + }) + + nodeOps.remove(rootNode) + for (const spy of disposalSpies) { + expect(spy).toHaveBeenCalledOnce() + } + }) + + it('does not dispose primitive material/geometries on remove(primitive)', () => { + const { primitive, material, geometry } = createElementPrimitiveMesh(nodeOps) + const spy0 = vi.spyOn(material, 'dispose') + const spy1 = vi.spyOn(geometry, 'dispose') + + const group = nodeOps.createElement('Group') + nodeOps.insert(primitive, group) + nodeOps.remove(primitive) + + expect(spy0).not.toBeCalled() + expect(spy1).not.toBeCalled() + }) + + it.skip('does not dispose primitive material/geometries on remove(ascestorOfPrimitive)', () => { + const { primitive, material, geometry } = createElementPrimitiveMesh(nodeOps) + const spy0 = vi.spyOn(material, 'dispose') + const spy1 = vi.spyOn(geometry, 'dispose') + + const group = nodeOps.createElement('Group') + nodeOps.insert(primitive, group) + nodeOps.remove(group) + + expect(spy0).not.toBeCalled() + expect(spy1).not.toBeCalled() + }) + + it.skip('does not call dispose on primitive materials/geometries in a tree of Mesh/Groups/Primitives created by nodeOps', () => { + const NUM_LEVEL = 5 + const NUM_CHILD_PER_NODE = 3 + const rootNode = mockTresObjectRootInObject(nodeOps.createElement('Group')) + const disposalSpies = [] + + createTreeIn(rootNode, (parent, childI, levelI) => { + if (childI > NUM_CHILD_PER_NODE || levelI > NUM_LEVEL) { + return false + } + if (Math.random() > 0.5) { + const { mesh } = createElementMesh(nodeOps) + nodeOps.insert(mesh, parent) + return mesh + } + else if (Math.random() > 0.5) { + const group = nodeOps.createElement('Group') + nodeOps.insert(group, parent) + return group + } + else { + const { primitive, material, geometry } = createElementPrimitiveMesh(nodeOps) + disposalSpies.push(vi.spyOn(geometry, 'dispose')) + disposalSpies.push(vi.spyOn(material, 'dispose')) + nodeOps.insert(primitive, parent) + return primitive + } + }) + + nodeOps.remove(rootNode) + for (const spy of disposalSpies) { + expect(spy).not.toHaveBeenCalled() + } + }) + + describe(':dispose="null"', () => { + it.skip('does not call dispose on any element in a subtree where the root :dispose==="null"', () => { + const NUM_LEVEL = 5 + const NUM_CHILD_PER_NODE = 3 + const rootNode = mockTresObjectRootInObject(nodeOps.createElement('Group')) + const disposalSpies = [] + const nullDisposeObjects = new Set() + + createTreeIn(rootNode, (parent, childI, levelI) => { + if (childI > NUM_CHILD_PER_NODE || levelI > NUM_LEVEL) { + return false + } + const { mesh, material, geometry } = createElementMesh(nodeOps) + if (nullDisposeObjects.has(parent)) { + nullDisposeObjects.add(mesh) + disposalSpies.push(vi.spyOn(geometry, 'dispose')) + disposalSpies.push(vi.spyOn(material, 'dispose')) + } + else if (levelI > 2 && Math.random() > 0.8) { + nodeOps.patchProp(mesh, 'dispose', undefined, null) + nullDisposeObjects.add(mesh) + disposalSpies.push(vi.spyOn(geometry, 'dispose')) + disposalSpies.push(vi.spyOn(material, 'dispose')) + } + nodeOps.insert(mesh, parent) + return mesh + }) + + nodeOps.remove(rootNode) + for (const spy of disposalSpies) { + expect(spy).not.toHaveBeenCalled() + } + }) + }) + + describe('in the THREE parent-child graph', () => { + it('detaches mesh from mesh', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { mesh: child } = createElementMesh(nodeOps) + nodeOps.insert(child, parent) + expect(child.parent.uuid).toBe(parent.uuid) + + nodeOps.remove(child) + expect(child.parent?.uuid).toBeFalsy() + }) + it('detaches group from mesh', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const child = nodeOps.createElement('Group') + nodeOps.insert(child, parent) + expect(child.parent.uuid).toBe(parent.uuid) + + nodeOps.remove(child) + expect(child.parent?.uuid).toBeFalsy() + }) + it('detaches mesh from group', () => { + const parent = nodeOps.createElement('Group') + const { mesh: child } = createElementMesh(nodeOps) + nodeOps.insert(child, parent) + expect(child.parent.uuid).toBe(parent.uuid) + + nodeOps.remove(child) + expect(child.parent?.uuid).toBeFalsy() + }) + it.skip('detaches mesh (in primitive :object) from mesh', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { primitive, mesh } = createElementPrimitiveMesh(nodeOps) + nodeOps.insert(primitive, parent) + expect(primitive.parent?.uuid).toBe(mesh.uuid) + + nodeOps.remove(primitive) + expect(mesh.parent?.uuid).toBeFalsy() + }) + it.skip('detaches mesh (in primitive :object) when mesh ancestor is removed', () => { + const { mesh: grandparent } = createElementMesh(nodeOps) + const { mesh: parent } = createElementMesh(nodeOps) + const { primitive, mesh: primitiveMesh } = createElementPrimitiveMesh(nodeOps) + nodeOps.insert(parent, grandparent) + nodeOps.insert(primitive, parent) + expect(primitiveMesh.parent?.uuid).toBe(parent.uuid) + + nodeOps.remove(parent) + expect(primitiveMesh.parent?.uuid).toBeFalsy() + }) + it('does not detach primitive :object descendants', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { primitive, mesh: primitiveMesh } = createElementPrimitiveMesh(nodeOps) + const grandChild0 = new THREE.Mesh() + const grandChild1 = new THREE.Group() + primitiveMesh.add(grandChild0, grandChild1) + + nodeOps.insert(primitive, parent) + expect(grandChild0.parent.uuid).toBe(primitiveMesh.uuid) + expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) + + nodeOps.remove(primitive) + expect(grandChild0.parent.uuid).toBe(primitiveMesh.uuid) + expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) + }) + }) }) describe('patchProp', () => { @@ -468,3 +738,44 @@ function mockTresContext() { deregisterCamera: () => {}, } as unknown as TresContext } + +function createElementMesh(nodeOps: ReturnType) { + const geometry = nodeOps.createElement('BoxGeometry') + const material = nodeOps.createElement('MeshNormalMaterial') + const mesh = nodeOps.createElement('Mesh') + nodeOps.insert(geometry, mesh) + nodeOps.insert(material, mesh) + return { mesh, geometry, material } +} + +function createElementPrimitiveMesh(nodeOps: ReturnType) { + const geometry = new THREE.BoxGeometry() + const material = new THREE.MeshNormalMaterial() + const mesh = new THREE.Mesh(geometry, material) + const primitive = nodeOps.createElement('primitive', undefined, undefined, { object: mesh }) + return { primitive, mesh, geometry, material } +} + +function createTreeIn(root: T, insertCallback: (parent: T, childI: number, levelI: number) => T) { + let levelII = 0 + const nextLevel = [root] as T[] + while (nextLevel.length) { + const currLevel = Array.from(nextLevel) + nextLevel.length = 0 + + while (currLevel.length) { + const parent = currLevel.shift() + let childI = 0 + while (true) { + const child = insertCallback(parent, childI++, levelII) + if (child) { + nextLevel.push(child) + } + else { + break + } + } + } + levelII++ + } +} From 700d1bac7e4b6493dac9455fad5fbd467fa4b0df Mon Sep 17 00:00:00 2001 From: Peter Date: Thu, 20 Jun 2024 17:25:28 +0200 Subject: [PATCH 02/20] feat: add filterInPlace --- src/utils/index.test.ts | 62 +++++++++++++++++++++++++++++++++++++++++ src/utils/index.ts | 17 +++++++++++ 2 files changed, 79 insertions(+) create mode 100644 src/utils/index.test.ts diff --git a/src/utils/index.test.ts b/src/utils/index.test.ts new file mode 100644 index 000000000..e39f822a7 --- /dev/null +++ b/src/utils/index.test.ts @@ -0,0 +1,62 @@ +import * as utils from './index' + +describe('filterInPlace', () => { + it('returns the passed array', () => { + const arr = [1, 2, 3] + const result = utils.filterInPlace(arr, v => v !== 0) + expect(result).toBe(arr) + }) + it('removes a single occurence', () => { + const arr = [1, 2, 3] + utils.filterInPlace(arr, v => v !== 1) + expect(arr).toStrictEqual([2, 3]) + }) + it('removes every occurence 0', () => { + const arr = [1, 1, 2, 1, 3, 1] + utils.filterInPlace(arr, v => v !== 1) + expect(arr).toStrictEqual([2, 3]) + }) + + it('removes every occurence 1', () => { + const [a, b, c] = [{}, {}, {}] + const COUNT = 400 + const arr = [] + for (const val of [a, b, c]) { + for (let i = 0; i < COUNT; i++) { + arr.push(val) + } + } + shuffle(arr) + + let filtered = [...arr] + utils.filterInPlace(arr, v => v !== b) + filtered = filtered.filter(v => v !== b) + expect(arr).toStrictEqual(filtered) + + utils.filterInPlace(arr, v => v !== c) + filtered = filtered.filter(v => v !== c) + expect(arr).toStrictEqual(filtered) + + utils.filterInPlace(arr, v => v !== a) + expect(arr).toStrictEqual([]) + }) + + it('sends an index to the callbackFn', () => { + const arr = 'abcdefghi'.split('') + utils.filterInPlace(arr, (_, i) => i % 2 === 0) + expect(arr).toStrictEqual('acegi'.split('')) + }) +}) + +function shuffle(array: any[]) { + let currentIndex = array.length + while (currentIndex !== 0) { + const randomIndex = Math.floor(Math.random() * currentIndex) + currentIndex--; + [array[currentIndex], array[randomIndex]] = [ + array[randomIndex], + array[currentIndex], + ] + } + return array +}; diff --git a/src/utils/index.ts b/src/utils/index.ts index 77faf0019..f85c5f04b 100644 --- a/src/utils/index.ts +++ b/src/utils/index.ts @@ -313,3 +313,20 @@ export function disposeObject3D(object: TresObject): void { } } } + +/** + * Like Array.filter, but modifies the array in place. + * @param array - Array to modify + * @param callbackFn - A function called for each element of the array. It should return a truthy value to keep the element in the array. + */ +export function filterInPlace(array: T[], callbackFn: (element: T, index: number) => unknown) { + let i = 0 + for (let ii = 0; ii < array.length; ii++) { + if (callbackFn(array[ii], ii)) { + array[i] = array[ii] + i++ + } + } + array.length = i + return array +} From 3d1541f0e278186144354d9a9adce773b667390b Mon Sep 17 00:00:00 2001 From: Peter Date: Thu, 20 Jun 2024 17:59:26 +0200 Subject: [PATCH 03/20] refactor: make some LocalState fields non-optional --- src/types/index.ts | 23 ++++++++++++++++------- 1 file changed, 16 insertions(+), 7 deletions(-) diff --git a/src/types/index.ts b/src/types/index.ts index 0f7550d93..34beb158e 100644 --- a/src/types/index.ts +++ b/src/types/index.ts @@ -42,15 +42,22 @@ interface TresBaseObject { export interface LocalState { type: string - // objects and parent are used when children are added with `attach` instead of being added to the Object3D scene graph - objects?: TresObject3D[] - parent?: TresObject3D | null + eventCount: number + root: TresContext + handlers: Partial + memoizedProps: { [key: string]: any } + // NOTE: + // LocalState holds information about the parent/child relationship + // in the Vue graph. If a child is `insert`ed into a parent using + // anything but THREE's `add`, it's put into the parent's `objects`. + // objects and parent are used when children are added with `attach` + // instead of being added to the Object3D scene graph + objects: TresObject[] + parent: TresObject | null + // NOTE: End graph info + primitive?: boolean - eventCount?: number - handlers?: Partial - memoizedProps?: { [key: string]: any } disposable?: boolean - root?: TresContext } // Custom type for geometry and material properties in Object3D @@ -62,6 +69,8 @@ export interface TresObject3D extends THREE.Object3D { export type TresObject = TresBaseObject & (TresObject3D | THREE.BufferGeometry | THREE.Material | THREE.Fog) & { __tres?: LocalState } +export type TresInstance = TresObject & { __tres: LocalState } + export interface TresScene extends THREE.Scene { __tres: { root: TresContext From 808e08205bcda837437f22ffb676363607881140 Mon Sep 17 00:00:00 2001 From: Peter Date: Thu, 20 Jun 2024 18:02:45 +0200 Subject: [PATCH 04/20] test: add LocalState graph tests --- src/core/nodeOps.test.ts | 78 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 78 insertions(+) diff --git a/src/core/nodeOps.test.ts b/src/core/nodeOps.test.ts index f298a95e9..1e71920ee 100644 --- a/src/core/nodeOps.test.ts +++ b/src/core/nodeOps.test.ts @@ -255,6 +255,55 @@ describe('nodeOps', () => { expect(parent.children.length).toBe(0) } }) + + it('adds a material to parent.__tres.objects', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const material = nodeOps.createElement('MeshNormalMaterial') + nodeOps.insert(material, parent) + expect(parent.__tres.objects.map(child => child.uuid)).toStrictEqual([material.uuid]) + }) + + it('adds a geometry to parent.__tres.objects', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const geometry = nodeOps.createElement('BoxGeometry') + nodeOps.insert(geometry, parent) + expect(parent.__tres.objects.map(child => child.uuid)).toStrictEqual([geometry.uuid]) + }) + + it('adds a fog to parent.__tres.objects', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const fog = nodeOps.createElement('Fog') + nodeOps.insert(fog, parent) + expect(parent.__tres.objects.map(child => child.uuid)).toStrictEqual([fog.uuid]) + }) + + it('adds parent to child.__tres.parent', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const material = nodeOps.createElement('MeshNormalMaterial') + const geometry = nodeOps.createElement('BoxGeometry') + const fog = nodeOps.createElement('Fog') + nodeOps.insert(material, parent) + nodeOps.insert(geometry, parent) + nodeOps.insert(fog, parent) + expect(material.__tres.parent).toBe(parent) + expect(geometry.__tres.parent).toBe(parent) + expect(fog.__tres.parent).toBe(parent) + }) + + it('adds non-Object3D children to parent.__tres.objects, but no more than once', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const material = nodeOps.createElement('MeshNormalMaterial') + const geometry = nodeOps.createElement('BoxGeometry') + const fog = nodeOps.createElement('Fog') + nodeOps.insert(material, parent) + nodeOps.insert(geometry, parent) + nodeOps.insert(fog, parent) + expect(parent.__tres.objects.length).toBe(3) + const objectSet = new Set(parent.__tres.objects) + expect(objectSet.has(material)).toBe(true) + expect(objectSet.has(geometry)).toBe(true) + expect(objectSet.has(fog)).toBe(true) + }) }) describe('remove', () => { @@ -515,6 +564,35 @@ describe('nodeOps', () => { expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) }) }) + describe('in the __tres parent-object graph', () => { + it('removes parent-object relationship when object is removed', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const material = nodeOps.createElement('MeshNormalMaterial') + const geometry = nodeOps.createElement('BoxGeometry') + const fog = nodeOps.createElement('Fog') + nodeOps.insert(material, parent) + nodeOps.insert(geometry, parent) + nodeOps.insert(fog, parent) + expect(material.__tres.parent).toBe(parent) + expect(geometry.__tres.parent).toBe(parent) + expect(fog.__tres.parent).toBe(parent) + + nodeOps.remove(fog) + expect(fog.__tres.parent).toBe(null) + expect(parent.__tres.objects.length).toBe(2) + expect(parent.__tres.objects.includes(fog)).toBe(false) + + nodeOps.remove(material) + expect(material.__tres.parent).toBe(null) + expect(parent.__tres.objects.length).toBe(1) + expect(parent.__tres.objects.includes(material)).toBe(false) + + nodeOps.remove(geometry) + expect(geometry.__tres.parent).toBe(null) + expect(parent.__tres.objects.length).toBe(0) + expect(parent.__tres.objects.includes(geometry)).toBe(false) + }) + }) }) describe('patchProp', () => { From 9af075a907de11c5bd9b42b1c68bc025f59d226e Mon Sep 17 00:00:00 2001 From: Peter Date: Thu, 20 Jun 2024 22:37:32 +0200 Subject: [PATCH 05/20] refactor: add prepare() to add __tres field --- src/core/nodeOps.ts | 24 ++++++++++++++++++------ 1 file changed, 18 insertions(+), 6 deletions(-) diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index 4325dffd7..3cc7c5357 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -88,14 +88,14 @@ export const nodeOps: (context: TresContext) => RendererOptions RendererOptions RendererOptions noop('cloneNode'), insertStaticContent: () => noop('insertStaticContent'), } + + function prepare(obj: T, state: Partial) { + const instance = obj as unknown as TresInstance + instance.__tres = { + type: 'unknown', + eventCount: 0, + root: context, + handlers: {}, + memoizedProps: {}, + objects: [], + parent: null, + ...state, + } + return instance + } } From b2eb17acf9bb912f20086d354273176b267085a6 Mon Sep 17 00:00:00 2001 From: Peter Date: Thu, 20 Jun 2024 22:38:37 +0200 Subject: [PATCH 06/20] refactor: add TODOs --- src/core/nodeOps.ts | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index 3cc7c5357..1f17ebd91 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -121,6 +121,10 @@ export const nodeOps: (context: TresContext) => RendererOptions RendererOptions Date: Thu, 20 Jun 2024 23:06:47 +0200 Subject: [PATCH 07/20] refactor: maintain parent/objects relationship in __tres --- src/core/nodeOps.ts | 25 +++++++++++++++++++++---- 1 file changed, 21 insertions(+), 4 deletions(-) diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index 1f17ebd91..8f514a6a1 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -2,8 +2,8 @@ import type { RendererOptions } from 'vue' import { BufferAttribute, Object3D } from 'three' import type { TresContext } from '../composables' import { useLogger } from '../composables' -import { deepArrayEqual, disposeObject3D, isHTMLTag, kebabToCamel } from '../utils' -import type { InstanceProps, TresObject, TresObject3D } from '../types' +import { deepArrayEqual, disposeObject3D, filterInPlace, isHTMLTag, kebabToCamel } from '../utils' +import type { InstanceProps, LocalState, TresInstance, TresObject, TresObject3D } from '../types' import * as is from '../utils/is' import { catalogue } from './catalogue' @@ -108,7 +108,7 @@ export const nodeOps: (context: TresContext) => RendererOptions RendererOptions RendererOptions RendererOptions obj !== node) + } + if (is.object3D(node)) { node.removeFromParent?.() @@ -331,7 +348,7 @@ export const nodeOps: (context: TresContext) => RendererOptions noop('insertStaticContent'), } - function prepare(obj: T, state: Partial) { + function prepare(obj: T, state: Partial): TresInstance { const instance = obj as unknown as TresInstance instance.__tres = { type: 'unknown', From ec47e0dd0d2738a957bdcc48f8913367127d07bd Mon Sep 17 00:00:00 2001 From: Peter Date: Fri, 21 Jun 2024 03:19:58 +0200 Subject: [PATCH 08/20] test: add dispose=null test --- src/core/nodeOps.test.ts | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/src/core/nodeOps.test.ts b/src/core/nodeOps.test.ts index 1e71920ee..13548cb63 100644 --- a/src/core/nodeOps.test.ts +++ b/src/core/nodeOps.test.ts @@ -466,7 +466,18 @@ describe('nodeOps', () => { }) describe(':dispose="null"', () => { - it.skip('does not call dispose on any element in a subtree where the root :dispose==="null"', () => { + it('does not call dipose on geometry/material in a Mesh where :dispose==="null"', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { mesh, geometry, material } = createElementMesh(nodeOps) + const spy0 = vi.spyOn(geometry, 'dispose') + const spy1 = vi.spyOn(material, 'dispose') + nodeOps.patchProp(mesh, 'dispose', undefined, null) + nodeOps.insert(mesh, parent) + nodeOps.remove(mesh) + expect(spy0).not.toBeCalled() + expect(spy1).not.toBeCalled() + }) + it('does not call dispose on any element in a subtree where the root :dispose==="null"', () => { const NUM_LEVEL = 5 const NUM_CHILD_PER_NODE = 3 const rootNode = mockTresObjectRootInObject(nodeOps.createElement('Group')) From fac2a08062b7a5453f9efb88abad5fa9a9921785 Mon Sep 17 00:00:00 2001 From: Peter Date: Fri, 21 Jun 2024 14:17:36 +0200 Subject: [PATCH 09/20] feat: allow "dispose=null" to bail out tree disposal --- src/core/nodeOps.test.ts | 6 ++-- src/core/nodeOps.ts | 65 ++++++++++++++++++++++++++-------------- src/utils/is.ts | 6 +++- 3 files changed, 52 insertions(+), 25 deletions(-) diff --git a/src/core/nodeOps.test.ts b/src/core/nodeOps.test.ts index 13548cb63..5ee25ce2a 100644 --- a/src/core/nodeOps.test.ts +++ b/src/core/nodeOps.test.ts @@ -312,7 +312,7 @@ describe('nodeOps', () => { const child = mockTresObjectRootInObject(new Mesh() as unknown as TresObject) nodeOps.insert(child, parent) nodeOps.remove(child) - expect(!parent.children.includes(child)).toBeTruthy() + expect(parent.children.includes(child)).toBeFalsy() }) it('silently does not remove a falsy child', () => { @@ -330,7 +330,9 @@ describe('nodeOps', () => { expect(spy).toHaveBeenCalledOnce() }) - it('calls dispose on a material array', () => { + it.skip('calls dispose on a material array', () => { + // TODO: Make this test pass. + // No way to add a material array via nodeOps currently. const parent = mockTresObjectRootInObject(nodeOps.createElement('Mesh', undefined, undefined, {})) const material0 = new THREE.MeshNormalMaterial() const material1 = new THREE.MeshNormalMaterial() diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index 8f514a6a1..30cd14b8a 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -2,7 +2,7 @@ import type { RendererOptions } from 'vue' import { BufferAttribute, Object3D } from 'three' import type { TresContext } from '../composables' import { useLogger } from '../composables' -import { deepArrayEqual, disposeObject3D, filterInPlace, isHTMLTag, kebabToCamel } from '../utils' +import { deepArrayEqual, filterInPlace, isHTMLTag, kebabToCamel } from '../utils' import type { InstanceProps, LocalState, TresInstance, TresObject, TresObject3D } from '../types' import * as is from '../utils/is' import { catalogue } from './catalogue' @@ -145,9 +145,18 @@ export const nodeOps: (context: TresContext) => RendererOptions + // 2) it has :dispose="null" + // 3) it was bailed out by a parent passing `remove(..., false)` + const isPrimitive = node.__tres?.primitive + const isDisposeNull = node.dispose === null + const isBailedOut = dispose === false + const shouldDispose = !(isPrimitive || isDisposeNull || isBailedOut) // TODO: // Figure out why `parent` is being set on `node` here @@ -161,29 +170,41 @@ export const nodeOps: (context: TresContext) => RendererOptions obj !== node) } - if (is.object3D(node)) { - node.removeFromParent?.() - - // Remove nested child objects. Primitives should not have objects and children that are - // attached to them declaratively ... - node.traverse((child) => { - context.deregisterCamera(child) - // deregisterAtPointerEventHandlerIfRequired?.(child as TresObject) - context.eventManager?.deregisterPointerMissedObject(child) - }) + node.removeFromParent?.() - context.deregisterCamera(node) - /* deregisterAtPointerEventHandlerIfRequired?.(node as TresObject) */ - invalidateInstance(node as TresObject) + // Remove nested child objects. Primitives should not have objects and children that are + // attached to them declaratively ... + node.traverse?.((child) => { + context.deregisterCamera(child) + // deregisterAtPointerEventHandlerIfRequired?.(child as TresObject) + context.eventManager?.deregisterPointerMissedObject(child) + }) - // Dispose the object if it's disposable, primitives needs to be manually disposed by - // calling dispose from `@tresjs/core` package like this `dispose(model)` - const isPrimitive = node.__tres?.primitive + context.deregisterCamera(node) + /* deregisterAtPointerEventHandlerIfRequired?.(node as TresObject) */ + invalidateInstance(node as TresObject) - if (!isPrimitive && node.__tres?.disposable) { - disposeObject3D(node) + // TODO: support removing `attach`ed components + + // NOTE: Recursively `remove` children and objects. + // Never on primitives: + // - removing children would alter the primitive :object. + // - primitives are not expected to have declarative children + // and so should not have `objects`. + if (!isPrimitive) { + // NOTE: In recursive `remove`, the array elements will + // remove themselves from these arrays, resulting in + // skipped elements. Make shallow copies of the arrays. + if (node.children) { + [...node.children].forEach(child => remove(child, shouldDispose)) + } + if (node.__tres && 'objects' in node.__tres) { + [...node.__tres.objects].forEach(obj => remove(obj, shouldDispose)) } - node.dispose?.() + } + + if (shouldDispose && node.dispose && !is.scene(node)) { + node.dispose() } } diff --git a/src/utils/is.ts b/src/utils/is.ts index 0f1a4ef15..e75ea3a15 100644 --- a/src/utils/is.ts +++ b/src/utils/is.ts @@ -1,5 +1,5 @@ import type { TresObject } from 'src/types' -import type { BufferGeometry, Camera, Fog, Material, Object3D } from 'three' +import type { BufferGeometry, Camera, Fog, Material, Object3D, Scene } from 'three' export function arr(u: unknown) { return Array.isArray(u) @@ -33,6 +33,10 @@ export function fog(u: unknown): u is Fog { return obj(u) && 'isFog' in u && !!(u.isFog) } +export function scene(u: unknown): u is Scene { + return obj(u) && 'isScene' in u && !!(u.isScene) +} + export function tresObject(u: unknown): u is TresObject { // NOTE: TresObject is currently defined as // TresObject3D | THREE.BufferGeometry | THREE.Material | THREE.Fog From f57de05c8dcfd1ba66d30d42ca2577750de32b1f Mon Sep 17 00:00:00 2001 From: Peter Date: Mon, 24 Jun 2024 16:31:09 +0200 Subject: [PATCH 10/20] refactor: update comments --- src/core/nodeOps.ts | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index 30cd14b8a..ba97c5b70 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -146,6 +146,14 @@ export const nodeOps: (context: TresContext) => RendererOptions RendererOptions obj !== node) } + // NOTE: THREE.removeFromParent removes `node` from + // `parent.children`. node.removeFromParent?.() - // Remove nested child objects. Primitives should not have objects and children that are - // attached to them declaratively ... + // NOTE: Deregister `node` THREE.Object3D children node.traverse?.((child) => { context.deregisterCamera(child) // deregisterAtPointerEventHandlerIfRequired?.(child as TresObject) context.eventManager?.deregisterPointerMissedObject(child) }) + // NOTE: Deregister `node` context.deregisterCamera(node) /* deregisterAtPointerEventHandlerIfRequired?.(node as TresObject) */ invalidateInstance(node as TresObject) @@ -203,6 +213,7 @@ export const nodeOps: (context: TresContext) => RendererOptions Date: Mon, 24 Jun 2024 16:32:14 +0200 Subject: [PATCH 11/20] refactor: add todo --- src/core/nodeOps.ts | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index ba97c5b70..e157fbeb5 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -109,7 +109,6 @@ export const nodeOps: (context: TresContext) => RendererOptions RendererOptions RendererOptions Date: Mon, 24 Jun 2024 16:32:46 +0200 Subject: [PATCH 12/20] test: add/unskip tests --- src/core/nodeOps.test.ts | 42 ++++++++++++++++++++++++++++++++-------- 1 file changed, 34 insertions(+), 8 deletions(-) diff --git a/src/core/nodeOps.test.ts b/src/core/nodeOps.test.ts index 5ee25ce2a..546a2d113 100644 --- a/src/core/nodeOps.test.ts +++ b/src/core/nodeOps.test.ts @@ -353,6 +353,19 @@ describe('nodeOps', () => { expect(spy).toHaveBeenCalledOnce() }) + it('calls dispose on material/geometry in a TresMesh child of a TresMesh', () => { + const { mesh: grandparent } = createElementMesh(nodeOps) + const { mesh: parent } = createElementMesh(nodeOps) + const { mesh: child } = createElementMesh(nodeOps) + nodeOps.insert(parent, grandparent) + nodeOps.insert(child, parent) + const childMaterialDisposalSpy = vi.spyOn(child.material, 'dispose') + const childGeometryDisposalSpy = vi.spyOn(child.geometry, 'dispose') + nodeOps.remove(parent) + expect(childGeometryDisposalSpy).toHaveBeenCalledOnce() + expect(childMaterialDisposalSpy).toHaveBeenCalledOnce() + }) + it('calls dispose on every material/geometry in a TresMesh tree', () => { const NUM_LEVEL = 5 const NUM_CHILD_PER_NODE = 3 @@ -419,7 +432,7 @@ describe('nodeOps', () => { expect(spy1).not.toBeCalled() }) - it.skip('does not dispose primitive material/geometries on remove(ascestorOfPrimitive)', () => { + it('does not dispose primitive material/geometries on remove(ascestorOfPrimitive)', () => { const { primitive, material, geometry } = createElementPrimitiveMesh(nodeOps) const spy0 = vi.spyOn(material, 'dispose') const spy1 = vi.spyOn(geometry, 'dispose') @@ -432,7 +445,7 @@ describe('nodeOps', () => { expect(spy1).not.toBeCalled() }) - it.skip('does not call dispose on primitive materials/geometries in a tree of Mesh/Groups/Primitives created by nodeOps', () => { + it('does not call dispose on primitive materials/geometries in a tree of Mesh/Groups/Primitives created by nodeOps', () => { const NUM_LEVEL = 5 const NUM_CHILD_PER_NODE = 3 const rootNode = mockTresObjectRootInObject(nodeOps.createElement('Group')) @@ -468,7 +481,7 @@ describe('nodeOps', () => { }) describe(':dispose="null"', () => { - it('does not call dipose on geometry/material in a Mesh where :dispose==="null"', () => { + it('does not call dispose on geometry/material in a Mesh where :dispose==="null"', () => { const { mesh: parent } = createElementMesh(nodeOps) const { mesh, geometry, material } = createElementMesh(nodeOps) const spy0 = vi.spyOn(geometry, 'dispose') @@ -479,6 +492,19 @@ describe('nodeOps', () => { expect(spy0).not.toBeCalled() expect(spy1).not.toBeCalled() }) + it('does not call dispose on child\'s geometry/material, for remove()', () => { + const { mesh: grandparent } = createElementMesh(nodeOps) + const { mesh: parent } = createElementMesh(nodeOps) + const { mesh: child, geometry, material } = createElementMesh(nodeOps) + const spy0 = vi.spyOn(geometry, 'dispose') + const spy1 = vi.spyOn(material, 'dispose') + nodeOps.patchProp(child, 'dispose', undefined, null) + nodeOps.insert(parent, grandparent) + nodeOps.insert(child, parent) + nodeOps.remove(parent) + expect(spy0).not.toBeCalled() + expect(spy1).not.toBeCalled() + }) it('does not call dispose on any element in a subtree where the root :dispose==="null"', () => { const NUM_LEVEL = 5 const NUM_CHILD_PER_NODE = 3 @@ -577,8 +603,8 @@ describe('nodeOps', () => { expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) }) }) - describe('in the __tres parent-object graph', () => { - it('removes parent-object relationship when object is removed', () => { + describe('in the __tres parent-objects graph', () => { + it('removes parent-objects relationship when object is removed', () => { const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) const material = nodeOps.createElement('MeshNormalMaterial') const geometry = nodeOps.createElement('BoxGeometry') @@ -591,17 +617,17 @@ describe('nodeOps', () => { expect(fog.__tres.parent).toBe(parent) nodeOps.remove(fog) - expect(fog.__tres.parent).toBe(null) + expect(fog.__tres?.parent).toBeFalsy() expect(parent.__tres.objects.length).toBe(2) expect(parent.__tres.objects.includes(fog)).toBe(false) nodeOps.remove(material) - expect(material.__tres.parent).toBe(null) + expect(material.__tres?.parent).toBeFalsy() expect(parent.__tres.objects.length).toBe(1) expect(parent.__tres.objects.includes(material)).toBe(false) nodeOps.remove(geometry) - expect(geometry.__tres.parent).toBe(null) + expect(geometry.__tres?.parent).toBeFalsy() expect(parent.__tres.objects.length).toBe(0) expect(parent.__tres.objects.includes(geometry)).toBe(false) }) From a585a351519581dda81909358b18d9ec0bf0694b Mon Sep 17 00:00:00 2001 From: Peter Date: Mon, 24 Jun 2024 16:47:57 +0200 Subject: [PATCH 13/20] refactor(nodeOps): move helper functions to new file --- src/core/nodeOps.ts | 39 +++++---------------------------------- src/utils/nodeOpsUtils.ts | 32 ++++++++++++++++++++++++++++++++ 2 files changed, 37 insertions(+), 34 deletions(-) create mode 100644 src/utils/nodeOpsUtils.ts diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index 8f514a6a1..6623e7755 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -3,15 +3,11 @@ import { BufferAttribute, Object3D } from 'three' import type { TresContext } from '../composables' import { useLogger } from '../composables' import { deepArrayEqual, disposeObject3D, filterInPlace, isHTMLTag, kebabToCamel } from '../utils' -import type { InstanceProps, LocalState, TresInstance, TresObject, TresObject3D } from '../types' +import type { InstanceProps, TresInstance, TresObject, TresObject3D } from '../types' import * as is from '../utils/is' +import { invalidateInstance, noop, prepareTresInstance } from '../utils/nodeOpsUtils' import { catalogue } from './catalogue' -function noop(fn: string): any { - // eslint-disable-next-line no-unused-expressions - fn -} - const { logError } = useLogger() const supportedPointerEvents = [ @@ -31,16 +27,6 @@ const supportedPointerEvents = [ 'onWheel', ] -export function invalidateInstance(instance: TresObject) { - const ctx = instance?.__tres?.root - - if (!ctx) { return } - - if (ctx.render && ctx.render.canBeInvalidated.value) { - ctx.invalidate() - } -} - export const nodeOps: (context: TresContext) => RendererOptions = (context) => { const scene = context.scene.value @@ -88,14 +74,14 @@ export const nodeOps: (context: TresContext) => RendererOptions RendererOptions RendererOptions noop('cloneNode'), insertStaticContent: () => noop('insertStaticContent'), } - - function prepare(obj: T, state: Partial): TresInstance { - const instance = obj as unknown as TresInstance - instance.__tres = { - type: 'unknown', - eventCount: 0, - root: context, - handlers: {}, - memoizedProps: {}, - objects: [], - parent: null, - ...state, - } - return instance - } } diff --git a/src/utils/nodeOpsUtils.ts b/src/utils/nodeOpsUtils.ts new file mode 100644 index 000000000..2c3ada316 --- /dev/null +++ b/src/utils/nodeOpsUtils.ts @@ -0,0 +1,32 @@ +import type { TresContext } from '../composables/useTresContextProvider' +import type { LocalState, TresInstance, TresObject } from '../types' + +export function prepareTresInstance(obj: T, state: Partial, context: TresContext): TresInstance { + const instance = obj as unknown as TresInstance + instance.__tres = { + type: 'unknown', + eventCount: 0, + root: context, + handlers: {}, + memoizedProps: {}, + objects: [], + parent: null, + ...state, + } + return instance +} + +export function invalidateInstance(instance: TresObject) { + const ctx = instance?.__tres?.root + + if (!ctx) { return } + + if (ctx.render && ctx.render.canBeInvalidated.value) { + ctx.invalidate() + } +} + +export function noop(fn: string): any { + // eslint-disable-next-line no-unused-expressions + fn +} From 46e62383b67e1bc29e11c90b32895158b73b1dcc Mon Sep 17 00:00:00 2001 From: Peter Date: Thu, 27 Jun 2024 16:24:34 +0200 Subject: [PATCH 14/20] test: add primitive tests --- src/core/nodeOps.test.ts | 251 +++++++++++++++++++++++++++++++++++---- 1 file changed, 225 insertions(+), 26 deletions(-) diff --git a/src/core/nodeOps.test.ts b/src/core/nodeOps.test.ts index 546a2d113..903c3242b 100644 --- a/src/core/nodeOps.test.ts +++ b/src/core/nodeOps.test.ts @@ -185,46 +185,47 @@ describe('nodeOps', () => { expect(parent.children.includes(child)).toBeTruthy() }) - describe('primitive :object', () => { + describe.skip('primitive :object', () => { describe('into mesh', () => { - it.skip('inserts a mesh :object', () => { + it('inserts a mesh :object', () => { const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) const object = new THREE.Mesh() const primitive = nodeOps.createElement('primitive', undefined, undefined, { object }) - expect(parent.material.uuid).not.toBe(object.uuid) + expect(parent.children.length).toBe(0) nodeOps.insert(primitive, parent) - expect(parent.material.uuid).toBe(object.uuid) + expect(parent.children.length).toBe(1) + expect(parent.children[0]).toBe(object) }) - it.skip('inserts a material :object', () => { + it('inserts a material :object', () => { const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) const object = new THREE.MeshNormalMaterial() const primitive = nodeOps.createElement('primitive', undefined, undefined, { object }) - expect(parent.material.uuid).not.toBe(object.uuid) + expect(parent.material.uuid).not.toBe(object) nodeOps.insert(primitive, parent) - expect(parent.material.uuid).toBe(object.uuid) + expect(parent.material).toBe(object) }) - it.skip('inserts a geometry :object', () => { + it('inserts a geometry :object', () => { const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) const object = new THREE.BoxGeometry() const primitive = nodeOps.createElement('primitive', undefined, undefined, { object }) - expect(parent.material.uuid).not.toBe(object.uuid) + expect(parent.geometry).not.toBe(object) nodeOps.insert(primitive, parent) - expect(parent.material.uuid).toBe(object.uuid) + expect(parent.geometry).toBe(object) }) - it.skip('inserts a group :object', () => { + it('inserts a group :object', () => { const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) const object = new THREE.Group() const primitive = nodeOps.createElement('primitive', undefined, undefined, { object }) - expect(parent.material.uuid).not.toBe(object.uuid) + expect(parent.children.length).toBe(0) nodeOps.insert(primitive, parent) - expect(parent.material.uuid).toBe(object.uuid) + expect(parent.children[0]).toBe(object) }) }) }) @@ -299,10 +300,86 @@ describe('nodeOps', () => { nodeOps.insert(geometry, parent) nodeOps.insert(fog, parent) expect(parent.__tres.objects.length).toBe(3) - const objectSet = new Set(parent.__tres.objects) - expect(objectSet.has(material)).toBe(true) - expect(objectSet.has(geometry)).toBe(true) - expect(objectSet.has(fog)).toBe(true) + const childrenSet = new Set(parent.__tres.objects) + expect(childrenSet.has(material)).toBe(true) + expect(childrenSet.has(geometry)).toBe(true) + expect(childrenSet.has(fog)).toBe(true) + }) + + it.skip('can insert the same `primitive :object` in multiple places in the scene graph', () => { + const material = new THREE.MeshNormalMaterial() + const geometry = new THREE.BoxGeometry() + const otherMaterial = new THREE.MeshBasicMaterial() + const otherGeometry = new THREE.SphereGeometry() + + const grandparent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const parent0 = nodeOps.createElement('Mesh', undefined, undefined, {}) + const parent1 = nodeOps.createElement('Mesh', undefined, undefined, {}) + const parent2 = nodeOps.createElement('Mesh', undefined, undefined, {}) + + const materialPrimitive0 = nodeOps.createElement('primitive', undefined, undefined, {object: material}) + const materialPrimitive1 = nodeOps.createElement('primitive', undefined, undefined, {object: material}) + const materialPrimitive2 = nodeOps.createElement('primitive', undefined, undefined, {object: material}) + const materialPrimitiveOther = nodeOps.createElement('primitive', undefined, undefined, {object: otherMaterial}) + + const geometryPrimitive0 = nodeOps.createElement('primitive', undefined, undefined, {object: geometry}) + const geometryPrimitive1 = nodeOps.createElement('primitive', undefined, undefined, {object: geometry}) + const geometryPrimitive2 = nodeOps.createElement('primitive', undefined, undefined, {object: geometry}) + const geometryPrimitiveOther = nodeOps.createElement('primitive', undefined, undefined, {object: otherGeometry}) + + nodeOps.insert(parent0, grandparent) + nodeOps.insert(parent1, grandparent) + nodeOps.insert(parent2, grandparent) + nodeOps.insert(materialPrimitive0, parent0) + nodeOps.insert(materialPrimitive1, parent1) + nodeOps.insert(materialPrimitive2, parent2) + nodeOps.insert(geometryPrimitive0, parent0) + nodeOps.insert(geometryPrimitive1, parent1) + nodeOps.insert(geometryPrimitive2, parent2) + + expect(parent0.material).toBe(material) + expect(parent1.material).toBe(material) + expect(parent2.material).toBe(material) + expect(parent0.geometry).toBe(geometry) + expect(parent1.geometry).toBe(geometry) + expect(parent2.geometry).toBe(geometry) + + nodeOps.insert(materialPrimitiveOther, parent0) + nodeOps.insert(geometryPrimitiveOther, parent1) + + expect(parent0.material).not.toBe(material) + expect(parent1.material).toBe(material) + expect(parent2.material).toBe(material) + expect(parent0.geometry).toBe(geometry) + expect(parent1.geometry).not.toBe(geometry) + expect(parent2.geometry).toBe(geometry) + + nodeOps.insert(materialPrimitiveOther, parent1) + nodeOps.insert(geometryPrimitiveOther, parent0) + + expect(parent0.material).not.toBe(material) + expect(parent1.material).not.toBe(material) + expect(parent2.material).toBe(material) + expect(parent0.geometry).not.toBe(geometry) + expect(parent1.geometry).not.toBe(geometry) + expect(parent2.geometry).toBe(geometry) + + nodeOps.insert(materialPrimitiveOther, parent2) + nodeOps.insert(geometryPrimitiveOther, parent2) + + expect(parent0.material).not.toBe(material) + expect(parent1.material).not.toBe(material) + expect(parent2.material).not.toBe(material) + expect(parent0.geometry).not.toBe(geometry) + expect(parent1.geometry).not.toBe(geometry) + expect(parent2.geometry).not.toBe(geometry) + + expect(parent0.material).toBe(otherMaterial) + expect(parent1.material).toBe(otherMaterial) + expect(parent2.material).toBe(otherMaterial) + expect(parent0.geometry).toBe(otherGeometry) + expect(parent1.geometry).toBe(otherGeometry) + expect(parent2.geometry).toBe(otherGeometry) }) }) @@ -567,16 +644,17 @@ describe('nodeOps', () => { nodeOps.remove(child) expect(child.parent?.uuid).toBeFalsy() }) - it.skip('detaches mesh (in primitive :object) from mesh', () => { + describe.skip('primitive', () => { + it('detaches mesh (in primitive :object) from mesh', () => { const { mesh: parent } = createElementMesh(nodeOps) const { primitive, mesh } = createElementPrimitiveMesh(nodeOps) nodeOps.insert(primitive, parent) - expect(primitive.parent?.uuid).toBe(mesh.uuid) + expect(primitive.parent?.uuid).toBe(parent.uuid) nodeOps.remove(primitive) expect(mesh.parent?.uuid).toBeFalsy() }) - it.skip('detaches mesh (in primitive :object) when mesh ancestor is removed', () => { + it('detaches mesh (in primitive :object) when mesh ancestor is removed', () => { const { mesh: grandparent } = createElementMesh(nodeOps) const { mesh: parent } = createElementMesh(nodeOps) const { primitive, mesh: primitiveMesh } = createElementPrimitiveMesh(nodeOps) @@ -585,7 +663,7 @@ describe('nodeOps', () => { expect(primitiveMesh.parent?.uuid).toBe(parent.uuid) nodeOps.remove(parent) - expect(primitiveMesh.parent?.uuid).toBeFalsy() + expect(primitiveMesh.parent?.type).toBeFalsy() }) it('does not detach primitive :object descendants', () => { const { mesh: parent } = createElementMesh(nodeOps) @@ -603,6 +681,7 @@ describe('nodeOps', () => { expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) }) }) + }) describe('in the __tres parent-objects graph', () => { it('removes parent-objects relationship when object is removed', () => { const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) @@ -738,20 +817,135 @@ describe('nodeOps', () => { expect(spy).toHaveBeenCalledTimes(3) }) - describe('patch `:object` on primitives', () => { + describe.skip('patch `:object` on primitives', () => { it('replaces original object', () => { const material0 = new THREE.MeshNormalMaterial() - const material1 = new THREE.MeshNormalMaterial() + const material1 = new THREE.MeshBasicMaterial() const primitive = nodeOps.createElement('primitive', undefined, undefined, { object: material0 }) nodeOps.patchProp(primitive, 'object', material0, material1) expect(primitive.object).toBe(material1) }) + + it('does not alter __tres on another primitive sharing the same object', () => { + const materialA = new THREE.MeshNormalMaterial() + const materialB = new THREE.MeshNormalMaterial() + const primitive0 = nodeOps.createElement('primitive', undefined, undefined, { object: materialA }) + const primitive0TresJson = JSON.stringify(primitive0.__tres) + const primitive1 = nodeOps.createElement('primitive', undefined, undefined, { object: materialA }) + + expect(primitive0.__tres).not.toBe(primitive1.__tres) + expect(JSON.stringify(primitive0.__tres)).toBe(primitive0TresJson) + + nodeOps.patchProp(primitive1, 'object', undefined, materialB) + expect(primitive0.__tres).not.toBe(primitive1.__tres) + expect(JSON.stringify(primitive0.__tres)).toBe(primitive0TresJson) + + nodeOps.patchProp(primitive1, 'object', undefined, materialA) + expect(primitive0.__tres).not.toBe(primitive1.__tres) + expect(JSON.stringify(primitive0.__tres)).toBe(primitive0TresJson) + }) + + it('does not replace the object in other primitives who point to the same object', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { mesh: child0 } = createElementMesh(nodeOps) + const { mesh: child1 } = createElementMesh(nodeOps) + const materialA = new THREE.MeshNormalMaterial() + const materialB = new THREE.MeshBasicMaterial() + const primitive1 = nodeOps.createElement('primitive', undefined, undefined, { object: materialA }) + const primitive0 = nodeOps.createElement('primitive', undefined, undefined, { object: materialA }) + + nodeOps.insert(primitive0, child0) + nodeOps.insert(primitive1, child1) + nodeOps.insert(child0, parent) + nodeOps.insert(child1, parent) + + expect(child0.material).toBe(materialA) + expect(child1.material).toBe(materialA) + + nodeOps.patchProp(primitive1, 'object', undefined, materialB) + expect(child0.material).toBe(materialA) + expect(child1.material).not.toBe(materialA) + + nodeOps.patchProp(primitive1, 'object', undefined, materialA) + expect(child0.material).toBe(materialA) + expect(child1.material).toBe(materialA) + + nodeOps.patchProp(primitive0, 'object', undefined, materialB) + expect(child0.material).not.toBe(materialA) + expect(child1.material).toBe(materialA) + + nodeOps.patchProp(primitive1, 'object', undefined, materialB) + expect(child0.material).not.toBe(materialA) + expect(child1.material).not.toBe(materialA) + expect(child0.material).toBe(materialB) + expect(child1.material).toBe(materialB) + }) + it('attaches the new object to the old object\'s parent; clears old object\'s parent', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { mesh: child0 } = createThreeBox() + const { mesh: child1 } = createThreeBox() + const primitive = nodeOps.createElement('primitive', undefined, undefined, { object: child0 }) + nodeOps.insert(primitive, parent) + expect(child0.parent).toBe(parent) + expect(parent.children[0]).toBe(child0) + expect(parent.children.length).toBe(1) + + nodeOps.patchProp(primitive, 'object', undefined, child1) + expect(child0.parent?.uuid).toBeFalsy() + expect(child1.parent?.uuid).toBe(parent.uuid) + expect(parent.children[0]).toBe(child1) + expect(parent.children.length).toBe(1) + }) + it('if old :object had been patched, those patches are applied to new :object', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { mesh: child0 } = createElementMesh(nodeOps) + const { mesh: child1 } = createElementMesh(nodeOps) + const primitive = nodeOps.createElement('primitive', undefined, undefined, { object: child0 }) + nodeOps.insert(primitive, parent) + nodeOps.patchProp(primitive, 'position-x', undefined, -999) + expect(child0.position.x).toBe(-999) + + nodeOps.patchProp(primitive, 'object', undefined, child1) + expect(child1.position.x).toBe(-999) + + nodeOps.patchProp(primitive, 'position-x', undefined, 1000) + nodeOps.patchProp(primitive, 'object', undefined, child0) + expect(child0.position.x).toBe(1000) + }) + it('does not attach old :object children to new :object', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { mesh: child0 } = createElementMesh(nodeOps) + const { mesh: child1 } = createElementMesh(nodeOps) + const grandchild0 = new THREE.Mesh() + const grandchild1 = new THREE.Mesh() + child0.add(grandchild0) + child1.add(grandchild1) + const primitive = nodeOps.createElement('primitive', undefined, undefined, { object: child0 }) + nodeOps.insert(primitive, parent) + expect(primitive.children[0]).toBe(grandchild0) + expect(primitive.children.length).toBe(1) + + nodeOps.patchProp(primitive, 'object', undefined, child1) + expect(primitive.children[0]).toBe(grandchild1) + expect(primitive.children.length).toBe(1) + + nodeOps.patchProp(primitive, 'object', undefined, child0) + expect(primitive.children[0].uuid).toBe(grandchild0.uuid) + expect(primitive.children.length).toBe(1) + + nodeOps.patchProp(primitive, 'object', undefined, child1) + expect(primitive.children[0]).toBe(grandchild1) + expect(primitive.children.length).toBe(1) + }) it('does not copy UUID', () => { const material0 = new THREE.MeshNormalMaterial() const material1 = new THREE.MeshNormalMaterial() const primitive = nodeOps.createElement('primitive', undefined, undefined, { object: material0 }) nodeOps.patchProp(primitive, 'object', material0, material1) expect(material0.uuid).not.toBe(material1.uuid) + + nodeOps.patchProp(primitive, 'object', material1, material0) + expect(material0.uuid).not.toBe(material1.uuid) }) }) @@ -856,6 +1050,13 @@ function mockTresContext() { } as unknown as TresContext } +function createThreeBox() { + const geometry = new THREE.BoxGeometry() + const material = new THREE.MeshNormalMaterial() + const mesh = new THREE.Mesh(geometry, material) + return { mesh, geometry, material } +} + function createElementMesh(nodeOps: ReturnType) { const geometry = nodeOps.createElement('BoxGeometry') const material = nodeOps.createElement('MeshNormalMaterial') @@ -866,9 +1067,7 @@ function createElementMesh(nodeOps: ReturnType) { } function createElementPrimitiveMesh(nodeOps: ReturnType) { - const geometry = new THREE.BoxGeometry() - const material = new THREE.MeshNormalMaterial() - const mesh = new THREE.Mesh(geometry, material) + const { mesh, geometry, material } = createThreeBox() const primitive = nodeOps.createElement('primitive', undefined, undefined, { object: mesh }) return { primitive, mesh, geometry, material } } From 21dded6d3b00c8062196efeaf86c584c957c97df Mon Sep 17 00:00:00 2001 From: Peter Date: Fri, 28 Jun 2024 23:35:15 +0200 Subject: [PATCH 15/20] refactor: move nodeOpsUtils to utils --- src/core/nodeOps.ts | 3 +-- src/utils/index.ts | 35 ++++++++++++++++++++++++++++++++++- src/utils/nodeOpsUtils.ts | 32 -------------------------------- 3 files changed, 35 insertions(+), 35 deletions(-) delete mode 100644 src/utils/nodeOpsUtils.ts diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index bbaad95f0..3138411cd 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -2,10 +2,9 @@ import type { RendererOptions } from 'vue' import { BufferAttribute, Object3D } from 'three' import type { TresContext } from '../composables' import { useLogger } from '../composables' -import { deepArrayEqual, filterInPlace, isHTMLTag, kebabToCamel } from '../utils' +import { attach, deepArrayEqual, detach, filterInPlace, isHTMLTag, kebabToCamel, prepareTresInstance, noop, invalidateInstance } from '../utils' import type { InstanceProps, TresInstance, TresObject, TresObject3D } from '../types' import * as is from '../utils/is' -import { invalidateInstance, noop, prepareTresInstance } from '../utils/nodeOpsUtils' import { catalogue } from './catalogue' const { logError } = useLogger() diff --git a/src/utils/index.ts b/src/utils/index.ts index f85c5f04b..0b2da4531 100644 --- a/src/utils/index.ts +++ b/src/utils/index.ts @@ -1,7 +1,9 @@ import type { Material, Mesh, Object3D, Texture } from 'three' import { DoubleSide, MeshBasicMaterial, Scene, Vector3 } from 'three' -import type { TresObject } from 'src/types' +import type { AttachType, LocalState, TresInstance, TresObject } from 'src/types' import { HightlightMesh } from '../devtools/highlight' +import type { TresContext } from '../composables/useTresContextProvider' +import * as is from './is' export function toSetMethodName(key: string) { return `set${key[0].toUpperCase()}${key.slice(1)}` @@ -330,3 +332,34 @@ export function filterInPlace(array: T[], callbackFn: (element: T, index: num array.length = i return array } + +export function prepareTresInstance(obj: T, state: Partial, context: TresContext): TresInstance { + const instance = obj as unknown as TresInstance + instance.__tres = { + type: 'unknown', + eventCount: 0, + root: context, + handlers: {}, + memoizedProps: {}, + objects: [], + parent: null, + previousAttach: null, + ...state, + } + return instance +} + +export function invalidateInstance(instance: TresObject) { + const ctx = instance?.__tres?.root + + if (!ctx) { return } + + if (ctx.render && ctx.render.canBeInvalidated.value) { + ctx.invalidate() + } +} + +export function noop(fn: string): any { + // eslint-disable-next-line no-unused-expressions + fn +} diff --git a/src/utils/nodeOpsUtils.ts b/src/utils/nodeOpsUtils.ts deleted file mode 100644 index 2c3ada316..000000000 --- a/src/utils/nodeOpsUtils.ts +++ /dev/null @@ -1,32 +0,0 @@ -import type { TresContext } from '../composables/useTresContextProvider' -import type { LocalState, TresInstance, TresObject } from '../types' - -export function prepareTresInstance(obj: T, state: Partial, context: TresContext): TresInstance { - const instance = obj as unknown as TresInstance - instance.__tres = { - type: 'unknown', - eventCount: 0, - root: context, - handlers: {}, - memoizedProps: {}, - objects: [], - parent: null, - ...state, - } - return instance -} - -export function invalidateInstance(instance: TresObject) { - const ctx = instance?.__tres?.root - - if (!ctx) { return } - - if (ctx.render && ctx.render.canBeInvalidated.value) { - ctx.invalidate() - } -} - -export function noop(fn: string): any { - // eslint-disable-next-line no-unused-expressions - fn -} From 81e90086e05c1f1d0c63ca38c51980bff5dcdeda Mon Sep 17 00:00:00 2001 From: Peter Date: Fri, 28 Jun 2024 23:36:06 +0200 Subject: [PATCH 16/20] feat: add pierced attach/detach --- .../advanced/multipleMaterials/index.vue | 41 +++ playground/src/router/routes/advanced.ts | 5 + src/core/nodeOps.test.ts | 291 +++++++++++++----- src/core/nodeOps.ts | 90 +++--- src/types/index.ts | 8 +- src/utils/index.test.ts | 75 +++++ src/utils/index.ts | 82 +++++ src/utils/is.test.ts | 30 ++ src/utils/is.ts | 8 + 9 files changed, 507 insertions(+), 123 deletions(-) create mode 100644 playground/src/pages/advanced/multipleMaterials/index.vue diff --git a/playground/src/pages/advanced/multipleMaterials/index.vue b/playground/src/pages/advanced/multipleMaterials/index.vue new file mode 100644 index 000000000..e45bb29b2 --- /dev/null +++ b/playground/src/pages/advanced/multipleMaterials/index.vue @@ -0,0 +1,41 @@ + + + diff --git a/playground/src/router/routes/advanced.ts b/playground/src/router/routes/advanced.ts index a2c5cdd96..867cd8cda 100644 --- a/playground/src/router/routes/advanced.ts +++ b/playground/src/router/routes/advanced.ts @@ -24,4 +24,9 @@ export const advancedRoutes = [ name: 'Suspense', component: () => import('../../pages/advanced/suspense/index.vue'), }, + { + path: '/advanced/multiple-materials', + name: 'Multiple materials', + component: () => import('../../pages/advanced/multipleMaterials/index.vue'), + }, ] diff --git a/src/core/nodeOps.test.ts b/src/core/nodeOps.test.ts index 903c3242b..62d9f3a16 100644 --- a/src/core/nodeOps.test.ts +++ b/src/core/nodeOps.test.ts @@ -129,32 +129,6 @@ describe('nodeOps', () => { it('throws an error if tag does not exist in catalogue', () => { expect(() => { nodeOps.createElement('THIS_TAG_DOES_NOT_EXIST', undefined, undefined, {}) }).toThrow() }) - - it('adds material with "attach" property if instance is a material', () => { - // Setup - const tag = 'TresMeshStandardMaterial' - const props = { args: [] } - - // Test - const instance = nodeOps.createElement(tag, undefined, undefined, props) - - // Assert - expect(instance?.isMaterial).toBeTruthy() - expect(instance?.attach).toBe('material') - }) - - it('adds attach geometry property if instance is a geometry', () => { - // Setup - const tag = 'TresTorusGeometry' - const props = { args: [] } - - // Test - const instance = nodeOps.createElement(tag, undefined, undefined, props) - - // Assert - expect(instance?.isBufferGeometry).toBeTruthy() - expect(instance?.attach).toBe('geometry') - }) }) describe('insert', () => { @@ -239,22 +213,179 @@ describe('nodeOps', () => { } }) - it('inserts Fog as a property', () => { - const parent = nodeOps.createElement('Object3D', undefined, undefined, {}) - const fog = nodeOps.createElement('Fog', undefined, undefined, {}) - nodeOps.insert(fog, parent) - expect(parent.fog).toBe(fog) - }) + describe('attach/detach', () => { + // NOTE: Special implementation case: `attach`/`detach` + // + // Objects that aren't added to the Scene using + // `THREE.Object3D`'s `add` will generally be inserted + // using `attach` and removed using `detach`. + // + // This way of inserting/removing has special challenges: + // - The user can specify how the object is `attach`/`detach`ed + // by setting the `attach` prop. + // - Before a new value is `attach`ed, the system must record + // the current value and restore it when the new value is + // `detach`ed. + it('if "attach" prop is provided, sets `parent[attach], even if the field does not exist on the parent`', () => { + const parent = nodeOps.createElement('Object3D', undefined, undefined, {}) + for (const attach of ['material', 'foo', 'bar', 'baz']) { + const child = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, { attach }) + nodeOps.insert(child, parent) + expect(parent[attach]).toBe(child) + expect(parent.children.length).toBe(0) + } + }) - it('if "attach" prop is provided, sets `parent[attach]`', () => { - const parent = nodeOps.createElement('Object3D', undefined, undefined, {}) - for (const attach of ['material', 'foo', 'bar', 'baz']) { - const child = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, {}) - child.attach = attach - nodeOps.insert(child, parent) - expect(parent[attach]).toBe(child) - expect(parent.children.length).toBe(0) - } + it('can attach and detach a BufferGeometry', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const previousAttach = parent.geometry + const geometry0 = nodeOps.createElement('BoxGeometry', undefined, undefined, {}) + const geometry1 = nodeOps.createElement('BoxGeometry', undefined, undefined, {}) + + nodeOps.insert(geometry0, parent) + expect(parent.geometry).not.toBe(previousAttach) + expect(parent.geometry).toBe(geometry0) + + nodeOps.remove(geometry0) + nodeOps.insert(geometry1, parent) + expect(parent.geometry).toBe(geometry1) + + nodeOps.remove(geometry1) + expect(parent.geometry).toBe(previousAttach) + }) + + it('can attach and detach a material', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const previousAttach = parent.material + const material0 = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, {}) + const material1 = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, {}) + nodeOps.insert(material0, parent) + expect(parent.material).toBe(material0) + + nodeOps.remove(material0) + nodeOps.insert(material1, parent) + expect(parent.material).toBe(material1) + + nodeOps.remove(material1) + expect(parent.material).toBe(previousAttach) + }) + + it('can attach and detach a material array', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const previousMaterial = parent.material + const material0 = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, { attach: 'material-0' }) + const material1 = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, { attach: 'material-1' }) + const material2 = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, { attach: 'material-2' }) + nodeOps.insert(material0, parent) + expect(parent.material[0]).toStrictEqual(material0) + nodeOps.insert(material1, parent) + expect(parent.material[1]).toStrictEqual(material1) + nodeOps.insert(material2, parent) + expect(parent.material[2]).toStrictEqual(material2) + + nodeOps.remove(material0) + expect(parent.material[0]).toBeUndefined() + expect(parent.material[1]).toBe(material1) + expect(parent.material[2]).toBe(material2) + nodeOps.remove(material2) + expect(parent.material[0]).toBeUndefined() + expect(parent.material[1]).toBe(material1) + expect(parent.material[2]).toBeUndefined() + + nodeOps.patchProp(material1, 'attach', undefined, 'material-2') + expect(parent.material[0]).toBeUndefined() + expect(parent.material[1]).toBeUndefined() + expect(parent.material[2]).toBe(material1) + + nodeOps.remove(material1) + expect(parent.material).toBe(previousMaterial) + }) + + it('can attach and detach fog', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const fog = nodeOps.createElement('Fog', undefined, undefined, {}) + nodeOps.insert(fog, parent) + expect(parent.fog).toBe(fog) + nodeOps.remove(fog) + expect('fog' in parent).toBe(false) + }) + + it('can attach and detach a "pierced" string', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const material = nodeOps.createElement('MeshBasicMaterial', undefined, undefined, { color: 'red' }) + const previousColor = material.color + const color = nodeOps.createElement('Color', undefined, undefined, { attach: 'material-color' }) + nodeOps.insert(material, parent) + nodeOps.insert(color, parent) + expect(parent.material.color).toBe(color) + nodeOps.remove(color) + expect(parent.material.color).toBe(previousColor) + + material.alphaMap = new THREE.Texture() + const previousAlphaMap = material.alphaMap + const alphaMap = nodeOps.createElement('Texture', undefined, undefined, { attach: 'material-alpha-map' }) + nodeOps.insert(alphaMap, parent) + expect(parent.material.alphaMap).toBe(alphaMap) + nodeOps.remove(alphaMap) + expect(parent.material.alphaMap).toBe(previousAlphaMap) + }) + + it('attach can be patched', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const previousMaterial = parent.material + const material = nodeOps.createElement('MeshBasicMaterial', undefined, undefined, { color: 'red', attach: 'material' }) + nodeOps.insert(material, parent) + expect(parent.material).toBe(material) + + nodeOps.patchProp(material, 'attach', undefined, 'foo') + expect(parent.foo).toBe(material) + expect(parent.material).toBe(previousMaterial) + + nodeOps.patchProp(material, 'attach', undefined, 'material') + expect(parent.foo).toBeUndefined() + expect(parent.material).toBe(material) + + nodeOps.patchProp(material, 'attach', undefined, 'bar') + expect(parent.bar).toBe(material) + expect(parent.material).toBe(previousMaterial) + }) + + it('can attach and detach a material array by patching `attach`', () => { + const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) + const previousMaterial = parent.material + const material0 = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, { attach: 'material-0' }) + const material1 = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, { attach: 'material-1' }) + const material2 = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, { attach: 'material-2' }) + nodeOps.insert(material0, parent) + nodeOps.insert(material1, parent) + nodeOps.insert(material2, parent) + expect(parent.material[0]).toBe(material0) + expect(parent.material[1]).toBe(material1) + expect(parent.material[2]).toBe(material2) + + nodeOps.patchProp(material1, 'attach', undefined, 'material-0') + expect(parent.material[0]).toBe(material1) + expect(parent.material[1]).toBeUndefined() + expect(parent.material[2]).toBe(material2) + + nodeOps.patchProp(material1, 'attach', undefined, 'material-2') + expect(parent.material[0]).toBe(material0) + expect(parent.material[1]).toBeUndefined() + expect(parent.material[2]).toBe(material1) + + nodeOps.patchProp(material0, 'attach', undefined, 'foo') + expect(parent.material[0]).toBeUndefined() + expect(parent.material[1]).toBeUndefined() + expect(parent.material[2]).toBe(material1) + + nodeOps.patchProp(material1, 'attach', undefined, 'foo') + expect(parent.material[0]).toBeUndefined() + expect(parent.material[1]).toBeUndefined() + expect(parent.material[2]).toBe(material2) + + nodeOps.patchProp(material2, 'attach', undefined, 'foo') + expect(parent.material).toBe(previousMaterial) + }) }) it('adds a material to parent.__tres.objects', () => { @@ -317,15 +448,15 @@ describe('nodeOps', () => { const parent1 = nodeOps.createElement('Mesh', undefined, undefined, {}) const parent2 = nodeOps.createElement('Mesh', undefined, undefined, {}) - const materialPrimitive0 = nodeOps.createElement('primitive', undefined, undefined, {object: material}) - const materialPrimitive1 = nodeOps.createElement('primitive', undefined, undefined, {object: material}) - const materialPrimitive2 = nodeOps.createElement('primitive', undefined, undefined, {object: material}) - const materialPrimitiveOther = nodeOps.createElement('primitive', undefined, undefined, {object: otherMaterial}) + const materialPrimitive0 = nodeOps.createElement('primitive', undefined, undefined, { object: material }) + const materialPrimitive1 = nodeOps.createElement('primitive', undefined, undefined, { object: material }) + const materialPrimitive2 = nodeOps.createElement('primitive', undefined, undefined, { object: material }) + const materialPrimitiveOther = nodeOps.createElement('primitive', undefined, undefined, { object: otherMaterial }) - const geometryPrimitive0 = nodeOps.createElement('primitive', undefined, undefined, {object: geometry}) - const geometryPrimitive1 = nodeOps.createElement('primitive', undefined, undefined, {object: geometry}) - const geometryPrimitive2 = nodeOps.createElement('primitive', undefined, undefined, {object: geometry}) - const geometryPrimitiveOther = nodeOps.createElement('primitive', undefined, undefined, {object: otherGeometry}) + const geometryPrimitive0 = nodeOps.createElement('primitive', undefined, undefined, { object: geometry }) + const geometryPrimitive1 = nodeOps.createElement('primitive', undefined, undefined, { object: geometry }) + const geometryPrimitive2 = nodeOps.createElement('primitive', undefined, undefined, { object: geometry }) + const geometryPrimitiveOther = nodeOps.createElement('primitive', undefined, undefined, { object: otherGeometry }) nodeOps.insert(parent0, grandparent) nodeOps.insert(parent1, grandparent) @@ -645,43 +776,43 @@ describe('nodeOps', () => { expect(child.parent?.uuid).toBeFalsy() }) describe.skip('primitive', () => { - it('detaches mesh (in primitive :object) from mesh', () => { - const { mesh: parent } = createElementMesh(nodeOps) - const { primitive, mesh } = createElementPrimitiveMesh(nodeOps) - nodeOps.insert(primitive, parent) - expect(primitive.parent?.uuid).toBe(parent.uuid) + it('detaches mesh (in primitive :object) from mesh', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { primitive, mesh } = createElementPrimitiveMesh(nodeOps) + nodeOps.insert(primitive, parent) + expect(primitive.parent?.uuid).toBe(parent.uuid) - nodeOps.remove(primitive) - expect(mesh.parent?.uuid).toBeFalsy() - }) - it('detaches mesh (in primitive :object) when mesh ancestor is removed', () => { - const { mesh: grandparent } = createElementMesh(nodeOps) - const { mesh: parent } = createElementMesh(nodeOps) - const { primitive, mesh: primitiveMesh } = createElementPrimitiveMesh(nodeOps) - nodeOps.insert(parent, grandparent) - nodeOps.insert(primitive, parent) - expect(primitiveMesh.parent?.uuid).toBe(parent.uuid) + nodeOps.remove(primitive) + expect(mesh.parent?.uuid).toBeFalsy() + }) + it('detaches mesh (in primitive :object) when mesh ancestor is removed', () => { + const { mesh: grandparent } = createElementMesh(nodeOps) + const { mesh: parent } = createElementMesh(nodeOps) + const { primitive, mesh: primitiveMesh } = createElementPrimitiveMesh(nodeOps) + nodeOps.insert(parent, grandparent) + nodeOps.insert(primitive, parent) + expect(primitiveMesh.parent?.uuid).toBe(parent.uuid) - nodeOps.remove(parent) - expect(primitiveMesh.parent?.type).toBeFalsy() - }) - it('does not detach primitive :object descendants', () => { - const { mesh: parent } = createElementMesh(nodeOps) - const { primitive, mesh: primitiveMesh } = createElementPrimitiveMesh(nodeOps) - const grandChild0 = new THREE.Mesh() - const grandChild1 = new THREE.Group() - primitiveMesh.add(grandChild0, grandChild1) + nodeOps.remove(parent) + expect(primitiveMesh.parent?.type).toBeFalsy() + }) + it('does not detach primitive :object descendants', () => { + const { mesh: parent } = createElementMesh(nodeOps) + const { primitive, mesh: primitiveMesh } = createElementPrimitiveMesh(nodeOps) + const grandChild0 = new THREE.Mesh() + const grandChild1 = new THREE.Group() + primitiveMesh.add(grandChild0, grandChild1) - nodeOps.insert(primitive, parent) - expect(grandChild0.parent.uuid).toBe(primitiveMesh.uuid) - expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) + nodeOps.insert(primitive, parent) + expect(grandChild0.parent.uuid).toBe(primitiveMesh.uuid) + expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) - nodeOps.remove(primitive) - expect(grandChild0.parent.uuid).toBe(primitiveMesh.uuid) - expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) + nodeOps.remove(primitive) + expect(grandChild0.parent.uuid).toBe(primitiveMesh.uuid) + expect(grandChild1.parent.uuid).toBe(primitiveMesh.uuid) + }) }) }) - }) describe('in the __tres parent-objects graph', () => { it('removes parent-objects relationship when object is removed', () => { const parent = nodeOps.createElement('Mesh', undefined, undefined, {}) diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index 3138411cd..d4c2e329f 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -38,13 +38,13 @@ export const nodeOps: (context: TresContext) => RendererOptions RendererOptions RendererOptions { @@ -212,6 +208,19 @@ export const nodeOps: (context: TresContext) => RendererOptions RendererOptions foo.bar = value) if (key.includes('-') && target === undefined) { + // TODO: A standalone function called `resolve` is + // available in /src/utils/index.ts. It's covered by tests. + // Refactor below to DRY. const chain = key.split('-') target = chain.reduce((acc, key) => acc[kebabToCamel(key)], root) key = chain.pop() as string diff --git a/src/types/index.ts b/src/types/index.ts index 34beb158e..afbfbd91b 100644 --- a/src/types/index.ts +++ b/src/types/index.ts @@ -6,8 +6,8 @@ import type { TresContext } from '../composables/useTresContextProvider' // Based on React Three Fiber types by Pmndrs // https://github.com/pmndrs/react-three-fiber/blob/v9/packages/fiber/src/three-types.ts -export type AttachFnType = (parent: any, self: O) => () => void -export type AttachType = string | AttachFnType +export type AttachFnType = (parent: any, self: TresInstance) => () => void +export type AttachType = string | AttachFnType export type ConstructorRepresentation = new (...args: any[]) => any export type NonFunctionKeys

= { [K in keyof P]-?: P[K] extends Function ? never : K }[keyof P] @@ -29,12 +29,10 @@ export interface InstanceProps { object?: T visible?: boolean dispose?: null - attach?: AttachType [prop: string]: any } interface TresBaseObject { - attach?: string removeFromParent?: () => void dispose?: () => void [prop: string]: any // for arbitrary properties @@ -58,6 +56,8 @@ export interface LocalState { primitive?: boolean disposable?: boolean + attach?: AttachType + previousAttach: any } // Custom type for geometry and material properties in Object3D diff --git a/src/utils/index.test.ts b/src/utils/index.test.ts index e39f822a7..861e961c7 100644 --- a/src/utils/index.test.ts +++ b/src/utils/index.test.ts @@ -60,3 +60,78 @@ function shuffle(array: any[]) { } return array }; + +describe('resolve', () => { + it('returns the first argument if it contains the key', () => { + const instance = { ab: 0 } + const { target, key } = utils.resolve(instance, 'ab') + expect(target).toBe(instance) + expect(key).toBe('ab') + }) + it('splits the key by "-" and traverses the obj using the pieces', () => { + const instance = { ab: { cd: { ef: 0 } } } + const { target, key } = utils.resolve(instance, 'ab-cd-ef') + expect(target).toBe(instance.ab.cd) + expect(key).toBe('ef') + }) + it('returns the current target holding the end of the key, and the end of the key', () => { + const instance = { ab: { cd: { ef: { gh: 0 } } } } + const { target, key } = utils.resolve(instance, 'ab-cd-ef') + expect(target).toBe(instance.ab.cd) + expect(key).toBe('ef') + }) + it('joins pierced props as camelCase, non-greedily', () => { + { + const instance = { abCdEfGh: { ij: 0 } } + const { target, key } = utils.resolve(instance, 'ab-cd-ef-gh-ij') + expect(target).toBe(instance.abCdEfGh) + expect(key).toBe('ij') + } + + { + const instance = { + abCdEfGh: { ij: 0 }, + abCdEf: { gh: { ij: 0 } }, + } + const { target, key } = utils.resolve(instance, 'ab-cd-ef-gh-ij') + expect(target).toBe(instance.abCdEf.gh) + expect(key).toBe('ij') + } + + { + const instance = { + abCdEfGh: { ij: 0 }, + abCd: { ef: { gh: { ij: 0 } }, efGh: { ij: 0 } }, + abCdEf: { gh: { ij: 0 } }, + } + const { target, key } = utils.resolve(instance, 'ab-cd-ef-gh-ij') + expect(target).toBe(instance.abCd.ef.gh) + expect(key).toBe('ij') + } + + { + const instance = { + abCdEfGh: { ij: 0 }, + abCdEf: { ghIj: 0 }, + ab: { cdEfGhIj: 0 }, + abCd: { ef: { gh: { ij: 0 } }, efGh: { ij: 0 } }, + } + const { target, key } = utils.resolve(instance, 'ab-cd-ef-gh-ij') + expect(target).toBe(instance.ab) + expect(key).toBe('cdEfGhIj') + } + }) + + it('joins my-key-and-the-unfindable-suffix as andTheUnfindableSuffix, for key suffixes that do not exist', () => { + expect(utils.resolve({}, 'zz').key).toBe('zz') + expect(utils.resolve({}, 'ab-cd-ef-gh-ij').key).toBe('abCdEfGhIj') + + const instance = { ab: { cd: { ef: { gh: { ij: 0 } } } } } + expect(utils.resolve(instance, 'ab-cd-ef-gh-ij-xx-yy-zz').key).toBe('xxYyZz') + expect(utils.resolve(instance, 'xx-yy-zz').key).toBe('xxYyZz') + expect(utils.resolve(instance, 'ab-xx-yy-zz').key).toBe('xxYyZz') + expect(utils.resolve(instance, 'ab-cd-zz').key).toBe('zz') + expect(utils.resolve(instance, 'ab-cd-xx-yy-zz').key).toBe('xxYyZz') + expect(utils.resolve(instance, 'ab-cd-xx-yy-zz').key).toBe('xxYyZz') + }) +}) diff --git a/src/utils/index.ts b/src/utils/index.ts index 0b2da4531..bde0fbd4e 100644 --- a/src/utils/index.ts +++ b/src/utils/index.ts @@ -333,6 +333,88 @@ export function filterInPlace(array: T[], callbackFn: (element: T, index: num return array } +export function resolve(obj: Record, key: string) { + let target = obj + if (key.includes('-')) { + const entries = key.split('-') + let currKey = entries.shift() as string + while (target && entries.length) { + if (!(currKey in target)) { + currKey = joinAsCamelCase(currKey, entries.shift() as string) + } + else { + target = target[currKey] + currKey = entries.shift() as string + } + } + return { target, key: joinAsCamelCase(currKey, ...entries) } + } + else { + return { target, key } + } +} + +function joinAsCamelCase(...strings: string[]): string { + return strings.map((s, i) => i === 0 ? s : s.charAt(0).toUpperCase() + s.slice(1)).join('') +} + +// Checks if a dash-cased string ends with an integer +const INDEX_REGEX = /-\d+$/ + +export function attach(parent: TresInstance, child: TresInstance, type: AttachType) { + if (is.str(type)) { + // NOTE: If attaching into an array (foo-0), create one + if (INDEX_REGEX.test(type)) { + const typeWithoutTrailingIndex = type.replace(INDEX_REGEX, '') + const { target, key } = resolve(parent, typeWithoutTrailingIndex) + if (!Array.isArray(target[key])) { + // NOTE: Create the array and augment it with a function + // that resets the original value if the array is empty or + // `[undefined, undefined, ...]`. The function will be run + // every time an element is `detach`ed from the array. + const previousAttach = target[key] + const augmentedArray: any[] & { __tresDetach?: () => void } = [] + augmentedArray.__tresDetach = () => { + if (augmentedArray.every(v => is.und(v))) { + target[key] = previousAttach + } + } + target[key] = augmentedArray + } + } + + const { target, key } = resolve(parent, type) + child.__tres.previousAttach = target[key] + target[key] = child + } + else { + child.__tres.previousAttach = type(parent, child) + } +} + +export function detach(parent: any, child: TresInstance, type: AttachType) { + if (is.str(type)) { + const { target, key } = resolve(parent, type) + const previous = child.__tres.previousAttach + // When the previous value was undefined, it means the value was never set to begin with + if (previous === undefined) { + delete target[key] + } + // NOTE: If the previous value was not an array, and `attach` turned it into an array + // then it also set `__tresOnArrayElementsUndefined`. Check for it and revert + // Otherwise set the previous value + else { + target[key] = previous + } + + if ('__tresDetach' in target) { target.__tresDetach() } + } + else { + child.__tres?.previousAttach?.(parent, child) + } + delete child.__tres?.previousAttach +} + export function prepareTresInstance(obj: T, state: Partial, context: TresContext): TresInstance { const instance = obj as unknown as TresInstance instance.__tres = { diff --git a/src/utils/is.test.ts b/src/utils/is.test.ts index f51c055e2..c60a14a0d 100644 --- a/src/utils/is.test.ts +++ b/src/utils/is.test.ts @@ -2,6 +2,36 @@ import { BufferGeometry, Fog, MeshBasicMaterial, MeshNormalMaterial, Object3D, P import * as is from './is' describe('is', () => { + describe('is.und(a: any)', () => { + describe('true', () => { + it('undefined', () => { + assert(is.und(undefined)) + }) + }) + describe('false', () => { + it('null', () => { + assert(!is.und(null)) + }) + it('number', () => { + assert(!is.und(0)) + assert(!is.und(-1)) + assert(!is.und(Math.PI)) + assert(!is.und(Number.POSITIVE_INFINITY)) + assert(!is.und(Number.NEGATIVE_INFINITY)) + assert(!is.und(42)) + }) + it('string', () => { + assert(!is.und('')) + assert(!is.und('tresObject')) + }) + it('function', () => { + assert(!is.und(() => {})) + }) + it('array', () => { + assert(!is.und([])) + }) + }) + }) describe('is.tresObject(a: any)', () => { describe('true', () => { it('object3D', () => { diff --git a/src/utils/is.ts b/src/utils/is.ts index e75ea3a15..67fa0f70c 100644 --- a/src/utils/is.ts +++ b/src/utils/is.ts @@ -1,10 +1,18 @@ import type { TresObject } from 'src/types' import type { BufferGeometry, Camera, Fog, Material, Object3D, Scene } from 'three' +export function und(u: unknown) { + return typeof u === 'undefined' +} + export function arr(u: unknown) { return Array.isArray(u) } +export function str(u: unknown): u is string { + return typeof u === 'string' +} + export function fun(u: unknown): u is Function { return typeof u === 'function' } From 5a037c42ee5ed39955e2dbe720970cfcbfdc7527 Mon Sep 17 00:00:00 2001 From: Peter Date: Sat, 29 Jun 2024 03:12:48 +0200 Subject: [PATCH 17/20] chore: clean up merge --- src/core/nodeOps.test.ts | 1 - src/core/nodeOps.ts | 11 +---------- src/types/index.ts | 16 ---------------- src/utils/nodeOpsUtils.ts | 32 -------------------------------- 4 files changed, 1 insertion(+), 59 deletions(-) delete mode 100644 src/utils/nodeOpsUtils.ts diff --git a/src/core/nodeOps.test.ts b/src/core/nodeOps.test.ts index 03382feca..62d9f3a16 100644 --- a/src/core/nodeOps.test.ts +++ b/src/core/nodeOps.test.ts @@ -529,7 +529,6 @@ describe('nodeOps', () => { } }) - it('calls dispose on a material', () => { it('calls dispose on a material', () => { const parent = mockTresObjectRootInObject(nodeOps.createElement('Mesh', undefined, undefined, {})) const material = nodeOps.createElement('MeshNormalMaterial', undefined, undefined, {}) diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index 61ce8551f..d4c2e329f 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -2,10 +2,9 @@ import type { RendererOptions } from 'vue' import { BufferAttribute, Object3D } from 'three' import type { TresContext } from '../composables' import { useLogger } from '../composables' -import { attach, deepArrayEqual, detach, filterInPlace, invalidateInstance, isHTMLTag, kebabToCamel, noop, prepareTresInstance } from '../utils' +import { attach, deepArrayEqual, detach, filterInPlace, isHTMLTag, kebabToCamel, prepareTresInstance, noop, invalidateInstance } from '../utils' import type { InstanceProps, TresInstance, TresObject, TresObject3D } from '../types' import * as is from '../utils/is' -import { invalidateInstance, noop, prepareTresInstance } from '../utils/nodeOpsUtils' import { catalogue } from './catalogue' const { logError } = useLogger() @@ -121,14 +120,6 @@ export const nodeOps: (context: TresContext) => RendererOptions - memoizedProps: { [key: string]: any } - // NOTE: - // LocalState holds information about the parent/child relationship - // in the Vue graph. If a child is `insert`ed into a parent using - // anything but THREE's `add`, it's put into the parent's `objects`. - // objects and parent are used when children are added with `attach` - // instead of being added to the Object3D scene graph - objects: TresObject[] - parent: TresObject | null - // NOTE: End graph info - primitive?: boolean disposable?: boolean attach?: AttachType @@ -85,8 +71,6 @@ export type TresObject = export type TresInstance = TresObject & { __tres: LocalState } -export type TresInstance = TresObject & { __tres: LocalState } - export interface TresScene extends THREE.Scene { __tres: { root: TresContext diff --git a/src/utils/nodeOpsUtils.ts b/src/utils/nodeOpsUtils.ts deleted file mode 100644 index 2c3ada316..000000000 --- a/src/utils/nodeOpsUtils.ts +++ /dev/null @@ -1,32 +0,0 @@ -import type { TresContext } from '../composables/useTresContextProvider' -import type { LocalState, TresInstance, TresObject } from '../types' - -export function prepareTresInstance(obj: T, state: Partial, context: TresContext): TresInstance { - const instance = obj as unknown as TresInstance - instance.__tres = { - type: 'unknown', - eventCount: 0, - root: context, - handlers: {}, - memoizedProps: {}, - objects: [], - parent: null, - ...state, - } - return instance -} - -export function invalidateInstance(instance: TresObject) { - const ctx = instance?.__tres?.root - - if (!ctx) { return } - - if (ctx.render && ctx.render.canBeInvalidated.value) { - ctx.invalidate() - } -} - -export function noop(fn: string): any { - // eslint-disable-next-line no-unused-expressions - fn -} From 6c2a4367425bab79de02e3b9215facc6c38e39f3 Mon Sep 17 00:00:00 2001 From: Peter Date: Sat, 29 Jun 2024 03:53:04 +0200 Subject: [PATCH 18/20] chore: lint --- .../advanced/multipleMaterials/index.vue | 41 ------------------- src/core/nodeOps.ts | 2 +- 2 files changed, 1 insertion(+), 42 deletions(-) delete mode 100644 playground/src/pages/advanced/multipleMaterials/index.vue diff --git a/playground/src/pages/advanced/multipleMaterials/index.vue b/playground/src/pages/advanced/multipleMaterials/index.vue deleted file mode 100644 index e45bb29b2..000000000 --- a/playground/src/pages/advanced/multipleMaterials/index.vue +++ /dev/null @@ -1,41 +0,0 @@ - - - diff --git a/src/core/nodeOps.ts b/src/core/nodeOps.ts index d4c2e329f..622473015 100644 --- a/src/core/nodeOps.ts +++ b/src/core/nodeOps.ts @@ -2,7 +2,7 @@ import type { RendererOptions } from 'vue' import { BufferAttribute, Object3D } from 'three' import type { TresContext } from '../composables' import { useLogger } from '../composables' -import { attach, deepArrayEqual, detach, filterInPlace, isHTMLTag, kebabToCamel, prepareTresInstance, noop, invalidateInstance } from '../utils' +import { attach, deepArrayEqual, detach, filterInPlace, invalidateInstance, isHTMLTag, kebabToCamel, noop, prepareTresInstance } from '../utils' import type { InstanceProps, TresInstance, TresObject, TresObject3D } from '../types' import * as is from '../utils/is' import { catalogue } from './catalogue' From 7f4e72f240f04638bda3b46c4dfc32a9c8517b98 Mon Sep 17 00:00:00 2001 From: Peter Date: Sat, 29 Jun 2024 03:53:47 +0200 Subject: [PATCH 19/20] docs: add playground demo --- .../pages/advanced/materialArray/index.vue | 41 +++++++++++++++++++ playground/src/router/routes/advanced.ts | 6 +-- 2 files changed, 44 insertions(+), 3 deletions(-) create mode 100644 playground/src/pages/advanced/materialArray/index.vue diff --git a/playground/src/pages/advanced/materialArray/index.vue b/playground/src/pages/advanced/materialArray/index.vue new file mode 100644 index 000000000..d8fc00908 --- /dev/null +++ b/playground/src/pages/advanced/materialArray/index.vue @@ -0,0 +1,41 @@ + + + diff --git a/playground/src/router/routes/advanced.ts b/playground/src/router/routes/advanced.ts index 867cd8cda..86a96f5ae 100644 --- a/playground/src/router/routes/advanced.ts +++ b/playground/src/router/routes/advanced.ts @@ -25,8 +25,8 @@ export const advancedRoutes = [ component: () => import('../../pages/advanced/suspense/index.vue'), }, { - path: '/advanced/multiple-materials', - name: 'Multiple materials', - component: () => import('../../pages/advanced/multipleMaterials/index.vue'), + path: '/advanced/material-array', + name: 'Material array', + component: () => import('../../pages/advanced/materialArray/index.vue'), }, ] From 00e1cb4e7bdd01f3a0e50a893192209c3b9db60b Mon Sep 17 00:00:00 2001 From: Peter Date: Sat, 29 Jun 2024 13:36:27 +0200 Subject: [PATCH 20/20] chore: update demos --- .../pages/advanced/materialArray/index.vue | 24 ++++++---- playground/src/pages/models/RiggedModel.vue | 36 -------------- .../models/RiggedModel}/UgglyBunny.vue | 0 .../src/pages/models/RiggedModel/index.vue | 48 +++++++++++++++++++ playground/src/router/routes/models.ts | 2 +- 5 files changed, 64 insertions(+), 46 deletions(-) delete mode 100644 playground/src/pages/models/RiggedModel.vue rename playground/src/{components => pages/models/RiggedModel}/UgglyBunny.vue (100%) create mode 100644 playground/src/pages/models/RiggedModel/index.vue diff --git a/playground/src/pages/advanced/materialArray/index.vue b/playground/src/pages/advanced/materialArray/index.vue index d8fc00908..843276d16 100644 --- a/playground/src/pages/advanced/materialArray/index.vue +++ b/playground/src/pages/advanced/materialArray/index.vue @@ -2,7 +2,13 @@ import { TresCanvas } from '@tresjs/core' import { OrbitControls } from '@tresjs/cientos' -const previewDataUri = '' +const previewDataUri = '' + +const i = shallowRef(0) +setInterval(() => { + i.value += 1 + i.value %= 6 +}, 500)