-
Notifications
You must be signed in to change notification settings - Fork 830
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Separate scaling from rotating on WebXR #2619
base: master
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -33,6 +33,7 @@ const INIT_FRAMES = 30; | |
// estimation, which will be used once available in WebXR. | ||
const AR_SHADOW_INTENSITY = 0.3; | ||
const ROTATION_RATE = 1.5; | ||
const ROTATION_THRESHOLD = 0.05; | ||
// Angle down (towards bottom of screen) from camera center ray to use for hit | ||
// testing against the floor. This makes placement faster and more intuitive | ||
// assuming the phone is in portrait mode. This seems to be a reasonable | ||
|
@@ -107,6 +108,7 @@ export class ARRenderer extends EventDispatcher { | |
private placementComplete = false; | ||
private isTranslating = false; | ||
private isRotating = false; | ||
private isScaling = false; | ||
private isTwoFingering = false; | ||
private lastDragPosition = new Vector3(); | ||
private firstRatio = 0; | ||
|
@@ -509,7 +511,7 @@ export class ARRenderer extends EventDispatcher { | |
null; | ||
} | ||
|
||
public moveToFloor(frame: XRFrame) { | ||
public moveToPlane(frame: XRFrame) { | ||
const hitSource = this.initialHitSource; | ||
if (hitSource == null) { | ||
return; | ||
|
@@ -569,19 +571,18 @@ export class ARRenderer extends EventDispatcher { | |
} else if (fingers.length === 2) { | ||
box.show = true; | ||
this.isTwoFingering = true; | ||
const {separation} = this.fingerPolar(fingers); | ||
this.firstRatio = separation / scene.scale.x; | ||
Comment on lines
-572
to
-573
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Well noticed, I had the same issue sometimes and realized I should probably rollback this change so that |
||
} | ||
}; | ||
|
||
private onSelectEnd = () => { | ||
this.isTranslating = false; | ||
this.isRotating = false; | ||
this.isScaling = false; | ||
this.isTwoFingering = false; | ||
this.inputSource = null; | ||
this.goalPosition.y += | ||
this.placementBox!.offsetHeight * this.presentedScene!.scale.x; | ||
this.placementBox!.show = false | ||
this.placementBox!.show = false; | ||
}; | ||
|
||
private fingerPolar(fingers: XRTransientInputHitTestResult[]): | ||
|
@@ -625,10 +626,17 @@ export class ARRenderer extends EventDispatcher { | |
this.isTwoFingering = false; | ||
} else { | ||
const {separation, deltaYaw} = this.fingerPolar(fingers); | ||
if (this.placeOnWall === false) { | ||
if (Math.abs(deltaYaw) > ROTATION_THRESHOLD) { | ||
this.isScaling = false; | ||
this.isRotating = true; | ||
} else if (!this.isScaling) { | ||
this.isScaling = true; | ||
this.firstRatio = separation / scene.scale.x; | ||
} | ||
if (this.placeOnWall === false && this.isRotating) { | ||
this.goalYaw += deltaYaw; | ||
} | ||
if (scene.canScale) { | ||
if (scene.canScale && this.isScaling) { | ||
const scale = separation / this.firstRatio; | ||
this.goalScale = | ||
(scale < SCALE_SNAP_HIGH && scale > SCALE_SNAP_LOW) ? 1 : scale; | ||
|
@@ -640,6 +648,7 @@ export class ARRenderer extends EventDispatcher { | |
// to scaling instead. | ||
this.isTranslating = false; | ||
this.isRotating = false; | ||
this.isScaling = true; | ||
this.isTwoFingering = true; | ||
const {separation} = this.fingerPolar(fingers); | ||
this.firstRatio = separation / scale; | ||
|
@@ -756,7 +765,7 @@ export class ARRenderer extends EventDispatcher { | |
this.updateView(view); | ||
|
||
if (isFirstView) { | ||
this.moveToFloor(frame); | ||
this.moveToPlane(frame); | ||
|
||
this.processInput(frame); | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for the PR! Can you describe why you think this UX is an improvement? I was following more the Google Maps style where both actions happen together. There is already a 30% snap on scaling to make it sticky near 100%. Also I'd always recommend setting
ar-scale: "fixed"
to disable scaling entirely for products that are meant to be seen at 100%.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We do use
ar-scale: "fixed"
whenever we need the product scale precision!I'm checking a lot on the WebXR code to make our migration and when I read the comment of actions being independent it felt awkward trying to rotate a model without scaling, mostly because it's hard to maintain a constant distance while rotating two fingers. I've also realized I personally use fingers from both hands in opposite directions to rotate, which always make the model shrink then expand. The same movement has always worked nicely on scene-viewer where movements are not interchangeable during a two-finger interaction.
On the other hand, a little bit of rotation while scaling feels acceptable so it's mainly for the reasons above.
Maybe I'm being picky on something that's actually a good feature for most users, since there's already a precise 1 finger rotation, so I don't mind leaving this PR for some extra research😁
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for the explanation! I'll leave this for some further discussion for now.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Awesome, thanks for opening it in the thread!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well, I do agree that slight rotation while scaling is not an issue, but accidentally scaling while trying to rotate is not desired. So if I understood correctly it should be definitely an improvement :) (when rotating beyond threshold scaling is disabled, right ?)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's it! I can make an online demo if it may help on deciding between both scenarios
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For me it is clear now and I support the change :) But maybe for the others :) ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Online simplified demo for testing