Skip to content
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

BufferGeometry: Implemented morphTargetsRelative #17649

Merged
merged 16 commits into from
Nov 5, 2019
Merged

BufferGeometry: Implemented morphTargetsRelative #17649

merged 16 commits into from
Nov 5, 2019

Conversation

zeux
Copy link
Contributor

@zeux zeux commented Oct 3, 2019

This change implements an addition to core BufferGeometry class, morphTargetsRelative.

By default, morphTargetsRelative is false which means that the morph targets contain absolute position/normal data. Setting this boolean to true forces three.js to interpret the target data as deltas from the base.

To implement rendering support, instead of duplicating shader code, the base mesh gets a weight that is set to 1 for relative targets and 1 - sum(weights) for absolute targets. The math ends up matching the original math with this:

base + w1 * (target1 - base) + w2 * (target2 - base) = base - w1 * base - w2 * base + w1 * target1 + w2 * target2 = base * (1 - w1 - w2) + w1 * target1 + w2 * target2

In JavaScript code we need to occasionally check the morphTargetsRelative value to determine the correct action.

As a result, glTF loader (in theory FBX loader as well, but I'm not an expert in FBX loader code so I didn't want to change this in this PR) doesn't need to duplicate morph target data. This fixes some issues with morph target format encodings, saves memory in some cases, and removes unnecessary data copying from the loading step.

I've tested this change on built-in examples that exercise morph target support as well as some custom glTF files processed by gltfpack. Also note that the change is backwards-compatible in that using an older version of GLTFLoader.js works as expected (by using the absolute morph target mode).

Fixes #17608.

@looeee
Copy link
Collaborator

looeee commented Oct 3, 2019

This would make implementation of morphtargets in the FBXLoader simpler.

@donmccurdy
Copy link
Collaborator

I'd like to do some performance profiling to quantify the difference, but my hunch is that this will be a significant parse time improvement for glTF models relying on morph targets, such as https://mrdoob.github.io/rome-gltf/.

@takahirox
Copy link
Collaborator

takahirox commented Oct 3, 2019

Nice update. I'm sure this change simplifies the glTF loader/exporter and reduces memory consumption.

I haven't looked into the implementation in the detail yet but I guess serialization/deserialization (.toJSON() and BufferGeometryLoader) should update to support morphTargetsRelative?

@zeux
Copy link
Contributor Author

zeux commented Oct 3, 2019

I missed the BufferGeometryLoader somehow - will fix!

@donmccurdy
Copy link
Collaborator

donmccurdy commented Oct 4, 2019

Some performance numbers for context. Testing against the glTF files in this folder, all of which contain morph targets, I found a couple interesting things.

First, parsing time was actually dominated by the embedded Data URIs in the .gltf files. I'd realized those were wasting file size, but hadn't realized they affected parsing time so much. So the takeway there is that it's better to either use .gltf with separate .bin files, or just use embedded binary .glb files. And usually .glb would be the better choice, to save network roundtrips.

With the slower .gltf files, rewriting morph targets represented 13% of parse time. With the faster .glb files, it was about 34%. In either case, using relative morph targets cut that to pretty much 0%. I don't necessarily know that these are representative files, and it was surprising to see how much variance there was in the individual runs.

A couple charts, showing the 7 models tested 3x each. Time spent processing morph targets is indicated in red. Y axis is milliseconds parsing, excluding network time.

gltf_parse

glb_parse

The important chart here is the one in the bottom left – I'd take that as a pretty good indicator that this makes a measurable difference on parsing time of real models.

EDIT: Just realized my charts are stacking morph parse time on top of total parse time, so the height of the bars is actually greater than total parse time. Mistake on my part, but not going to regenerate them at this point. 😇

@zeux
Copy link
Contributor Author

zeux commented Oct 4, 2019

I've updated the change to fix a few stray examples / code that slipped my initial notice, and also fix the computeBoundingBox/Sphere functions to use precise (I think) formulas.

@donmccurdy Thanks for gathering the numbers!

@zeux zeux requested a review from donmccurdy October 6, 2019 19:48
@donmccurdy
Copy link
Collaborator

@mrdoob per #17649 (comment), this would make GLTFLoader parsing more efficient. What do you think about (a) whether to add the property, and (b) how it should be located and named?

@zeux
Copy link
Contributor Author

zeux commented Oct 9, 2019

FWIW while I'm not attached to the naming I think the property should be on BufferGeometry - this is because otherwise the data stored in BufferGeometry.morphAttributes becomes impossible to interpret outside of context. For example, if the property lived on the Material, changing materials would break the morphed shape visually if the value isn't fixed up. With this interface, BufferGeometry is "portable" - it can be migrated between objects / meshes / materials without losing the semantics.

@zeux
Copy link
Contributor Author

zeux commented Oct 9, 2019

Oh and also given that both glTF and FBX loaders use relative data it seems that relative data is an important use case. Some formats like COLLADA support both - interestingly, if you look at COLLADA spec, they use the exact same formulation as this change:

image

zeux added 10 commits October 12, 2019 08:51
This adds the attribute and the necessary plumbing for various JS
functions; there's no support for specifying this for the shader to
consume yet.
The shader code is now a bit simpler and relies on the precomputed base
influence value; JS code sets it up appropriately.
This uses the newly added functionality for loading morph target data in
GLTF loader; there's no need to clone attribute data anymore.
On further reflection it shouldn't be necessary - it's probably better
to maintain "exact" match between the shader code and CPU-side code that
may use (pos - morph) * w directly.
The code in GLTF exporter is probably suboptimal - it's not clear that
we need to clone the attribute if it's already relative.
Also fix toNonIndexed to copy the attribute
expandByVector is not correct to call here; if a morph target has a min
of [1,1,1], it should *not* change the box.min but it should expand
box.max by 1.

To avoid doing per-component math we use addVectors to get the corners
of the box, and use expandByPoint to union the box with the resulting
point.
@zeux
Copy link
Contributor Author

zeux commented Oct 12, 2019

I've rebased this against dev because of a trivial conflict with BufferGeometry.d.ts; should be cleanly mergeable again. @mrdoob please let me know if there's anything else to be done here, as per above this seems beneficial and complete. I could rename morphTargetsRelative to something like morphRelative if that seems more appropriate.

@Mugen87
Copy link
Collaborator

Mugen87 commented Oct 12, 2019

I've added two minor comments. Apart from that, the change looks good!

@zeux zeux requested a review from Mugen87 October 12, 2019 18:59
Fix default value in documentation
@zeux zeux requested a review from WestLangley October 13, 2019 03:25
Copy link
Collaborator

@donmccurdy donmccurdy left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd support making a change like this to allow relative morph targets, especially given #17608 (comment). I'm open to other APIs for it, but this PR seems like the most direct approach from where the API is today.

@zeux
Copy link
Contributor Author

zeux commented Oct 22, 2019

@mrdoob please let me know if any further changes are required

@mrdoob
Copy link
Owner

mrdoob commented Oct 24, 2019

@zeux

Sorry for the delay.

In your experience, do game engines usually support both modes? Or is it usually only relative mode? (Tempted of only supporting relative mode and have a method to convert absolute to relative to help with breakage)

Also, does it only apply to vertex positions? Or also to normals and colors somehow?

@takahirox
Copy link
Collaborator

takahirox commented Oct 24, 2019

@mrdoob #17608 (comment) may help (in case you've missed)

@zeux
Copy link
Contributor Author

zeux commented Oct 25, 2019

Yeah so resummarizing - glTF and FBX work with relative data; COLLADA allows both; keyframed animations from MD2 use absolute data.

I think there are two use cases for morph targets - one prevalent one, for blend shapes - morphing between multiple variants of the same mesh where usually only part of the mesh gets deformed and it’s common to apply many shapes with fractional coefficients - think facial expressions (where you always overlay corrections on top of base pose). And one less common one, for morphing between substantially different poses - e.g. if you want to morph between meshes with unrelated structure relative data doesn’t make as much sense.

My overall feeling is that relative targets are the most important, and absolute targets could be either not supported or emulated through absolute targets. However, this is a large breaking change.

So I would recommend doing this gradually:

  • Adding support for relative targets (this PR)
  • Changing loaders that work with relative data to use this support directly (this PR does it for GLTF but there’s also FBX and COLLADA)
  • Surveying users about whether their use cases need absolute support
  • If it’s considered unnecessary to support absolute mode, migrating loaders that expect absolute mode (MD2, possibly COLLADA when it uses absolute data) to transform absolute to relative, and removing support separately.

The cost of supporting both isn’t very large so I would err on the side of caution, but I don’t know the philosophy of Three.JS wrt breaking changes.

@zeux
Copy link
Contributor Author

zeux commented Oct 25, 2019

Oh and the morphs are usually relative or absolute for all attributes, including normals. I have not seen morph targets used for other attributes - it’s certainly possible in theory but I don’t think it’s very common.

The general form here really is a full weighted sum of all targets, with a custom weight for base target. If the fact that there are many places that have to check the relative state is a concern, I can refactor this to use the general weighting math from the shader on JavaScript side? That would remove most of the conditions. I would prefer to do it in a separate PR but I can do it as part of this one if necessary.

@mrdoob
Copy link
Owner

mrdoob commented Oct 26, 2019

Thanks for summarising!

I think I vote for making this a breaking change: support only relative morphtargets and adding a method that converts absolute morphattributes to relative.

@zeux
Copy link
Contributor Author

zeux commented Oct 26, 2019

Do you mean to do this in one step? This also means that the three.js release with this change will silently break components such as GLTFLoader if they aren't using the version from this change which seems unsatisfying. The original intent for this PR was to make a change that is safe to deploy, so that deprecation can be done separately.

@Mugen87
Copy link
Collaborator

Mugen87 commented Oct 26, 2019

This also means that the three.js release with this change will silently break components such as GLTFLoader if they aren't using the version from this change which seems unsatisfying.

We normally do not support this scenario anyway. Check out the following notes from the docs (last section):

Note: When using code from the examples directory, it's important that all files match the version of your three.js main file. For example, it's not acceptable to use GLTFLoader and OrbitControls from R96 together with three.js R103.

@mrdoob
Copy link
Owner

mrdoob commented Oct 26, 2019

Do you mean to do this in one step?

Yeah

The original intent for this PR was to make a change that is safe to deploy, so that deprecation can be done separately.

Yes, but it also introduces new API (morphTargetsRelative) which we'll have to serialise, deserialise, maintain, and eventually add to the legacy file.

Seems easier to have a breaking change and point people to a method that fixes it.

@mrdoob
Copy link
Owner

mrdoob commented Oct 28, 2019

We can do it in two steps actually.
I'll merge this and I'll do another PR with the second step so we can revert just in case.

@zeux
Copy link
Contributor Author

zeux commented Oct 29, 2019

Sorry I’ve been short on time the last few days. Two steps sound good - that’s my preference anyway. It is my understanding that JSON export isn’t a long term compatibility target either? Effectively when removing the option, we'll start treating existing JSON exports as relative instead of absolute.

@mrdoob mrdoob modified the milestones: r110, r111 Oct 30, 2019
@mrdoob
Copy link
Owner

mrdoob commented Oct 30, 2019

Better to merge it at the beginning of r111.

@mrdoob mrdoob changed the title Implement support for BufferGeometry.morphTargetsRelative BufferGeometry: Implemented morphTargetsRelative Nov 5, 2019
@mrdoob mrdoob merged commit 3276d13 into mrdoob:dev Nov 5, 2019
@mrdoob
Copy link
Owner

mrdoob commented Nov 5, 2019

Thanks!

@mrdoob
Copy link
Owner

mrdoob commented Nov 5, 2019

@looeee

This would make implementation of morphtargets in the FBXLoader simpler.

Any chance FBXLoader can be updated to use morphTargetsRelative?

@donmccurdy
Copy link
Collaborator

Thank you @zeux! :)

@looeee
Copy link
Collaborator

looeee commented Nov 6, 2019

@mrdoob Are you still considering removing the FBXLoader at some point?

If so I'm not sure that it's worth spending the time to update it.

@mrdoob
Copy link
Owner

mrdoob commented Nov 6, 2019

@looeee well... removing code from it will help make it less intimidating and maybe that way we'll get more eyes on it.

@looeee
Copy link
Collaborator

looeee commented Nov 8, 2019

maybe that way we'll get more eyes on it.

Well, the code is now about 3 lines shorter so let's see how that goes 😁

#17890

@Mugen87 Mugen87 mentioned this pull request Dec 4, 2019
15 tasks
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

GLTFLoader: Morph targets only work for matching accessor types
7 participants