Skip to content

Commit 5589ee2

Browse files
committed
fix(troika-xr): destroy WristMountedUI contents when hidden
WristMountedUI contents are now destroyed when the UI is in inactive state, by default, unless the user specifies keepContentAlive:true. This fixes issues with instanced objects in the content remaining visible and contributing to render times. Also fixed issue where the wristband/container were sometimes rendered without a valid XR gripPose.
1 parent b3f14ac commit 5589ee2

File tree

2 files changed

+36
-17
lines changed

2 files changed

+36
-17
lines changed

packages/troika-xr/src/facade/wrist-mounted-ui/ContentContainer.js

+11-8
Original file line numberDiff line numberDiff line change
@@ -17,13 +17,15 @@ export class ContentContainer extends Object3DFacade {
1717
this.projectionSourcePosition = null //worldspace vec3
1818
this.gripPose = null
1919
this.active = false
20+
this.keepContentAlive = false
2021
}
2122

2223
/**
2324
* Sync to the current XRFrame's gripPose - all matrix syncing is localized here
2425
* to avoid a full afterUpdate pass on every frame.
2526
*/
2627
syncPose(gripPose) {
28+
let visible = false
2729
if (gripPose) {
2830
// Get current posed camera position, relative to the parent
2931
let cam = this.getCameraFacade().threeObject
@@ -50,9 +52,9 @@ export class ContentContainer extends Object3DFacade {
5052
let pos = this.threeObject.position
5153
pos.lerp(targetPos, 0.05) //move by 5% of distance each frame)
5254
this.scale += (targetScale - this.scale) * 0.3
53-
this.visible = this.scale > 0.01 //hide below a certain size
55+
visible = this.scale > 0.01 //hide below a certain size
5456

55-
if (this.visible) {
57+
if (visible) {
5658
// Rotate to face camera
5759
this.rotateY = Math.atan2(camPos.x - pos.x, camPos.z - pos.z)
5860

@@ -65,16 +67,17 @@ export class ContentContainer extends Object3DFacade {
6567
// Sync all matrices
6668
this.traverse(updateMatrices)
6769
}
68-
} else {
69-
this.visible = false
7070
}
71-
}
72-
73-
shouldUpdateChildren () {
74-
return !!this.visible
71+
if (visible !== this.visible) {
72+
this.update({visible})
73+
}
7574
}
7675

7776
describeChildren() {
77+
if (!this.visible && !this.keepContentAlive) {
78+
return null
79+
}
80+
7881
let kids = this._kidsTpl || (this._kidsTpl = [
7982
{
8083
key: '$platform',

packages/troika-xr/src/facade/wrist-mounted-ui/WristMountedUI.js

+25-9
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { Group3DFacade } from 'troika-3d'
1+
import { Facade, Group3DFacade } from 'troika-3d'
22
import {
33
Matrix4,
44
Quaternion,
@@ -24,18 +24,29 @@ const upVec3 = new Vector3(0, 1, 0)
2424
export class WristMountedUI extends Group3DFacade {
2525
constructor (parent) {
2626
super(parent)
27-
this.active = false
27+
// Config:
2828
this.activeUpAngle = Math.PI / 7
2929
this.preferredHand = 'left'
3030
this.platformRadius = 0.25
3131
this.platformColor = 0x333333
3232
this.projectionColor = 0x3399ff
33+
this.keepContentAlive = false
34+
this.onActiveChange = null
35+
36+
// Internal state:
37+
this.gripPose = null
38+
this.active = false
3339

3440
this._cogPos = new Vector3()
3541
this.addEventListener('xrframe', this.onXRFrame.bind(this))
3642
}
3743

3844
describeChildren () {
45+
// Only render children if we have a valid gripPose
46+
if (!this.gripPose) {
47+
return null
48+
}
49+
3950
let children = this._childTpl || (this._childTpl = [
4051
{
4152
key: 'wristband',
@@ -62,6 +73,7 @@ export class WristMountedUI extends Group3DFacade {
6273
contentDef.platformColor = this.platformColor
6374
contentDef.projectionColor = this.projectionColor
6475
contentDef.projectionSourcePosition = this._cogPos
76+
contentDef.keepContentAlive = this.keepContentAlive
6577
contentDef.children = this.children
6678

6779
return children
@@ -75,9 +87,9 @@ export class WristMountedUI extends Group3DFacade {
7587

7688
onXRFrame (time, xrFrame) {
7789
let gripPose = null
90+
let active = false
7891
let inputSources = xrFrame.session.inputSources
7992
if (inputSources) {
80-
let active = false
8193
let gripSpace = null
8294
for (let i = 0, len = inputSources.length; i < len; i++) {
8395
if (inputSources[i].handedness === this.preferredHand) {
@@ -101,18 +113,22 @@ export class WristMountedUI extends Group3DFacade {
101113
active = tempVec3.angleTo(upVec3) < this.activeUpAngle
102114
}
103115
}
104-
if (active !== this.active) {
105-
this.active = active
106-
this.afterUpdate()
116+
}
117+
118+
if (active !== this.active) {
119+
if (this.onActiveChange) {
120+
this.onActiveChange(active)
107121
}
122+
this.update({active})
108123
}
109124

110-
if (gripPose || !this.gripPose) {
111-
this.gripPose = gripPose
125+
if (!!gripPose !== !!this.gripPose) {
126+
this.update({gripPose})
127+
}
128+
else if (gripPose) {
112129
// Skip full afterUpdate pass, just give the gripPose to children - they both have
113130
// a syncPose method to handle syncing matrices without a full afterUpdate.
114131
this.forEachChild(child => child.syncPose(gripPose))
115132
}
116133
}
117-
118134
}

0 commit comments

Comments
 (0)