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

GLTFLoader: Remove onBeforeRender dependency #14099

Merged
merged 6 commits into from
Jan 24, 2020

Conversation

pailhead
Copy link
Contributor

@pailhead pailhead commented May 20, 2018

Per conversation with @mrdoob.

Refactor GLTFLoader to remove the onBeforeRender dependency with onBeforeCompile dependency. It seems to make more sense to have the material manage the uniforms, rather than a scene graph node.

Ideally i would not use onBeforeCompile but an alternate approach where different systems of the WebGLRenderer could be leveraged. But this is a step.

The linter should maybe be run in this file, as it's got a lot of complaints in the dev branch.

ping @takahirox @donmccurdy

@WestLangley
Copy link
Collaborator

extend StandardMaterial with onBeforeCompile

Yes, I am inclined to agree that doing so is better than creating a custom ShaderMaterial, but it is still a bit tedious.

@pailhead pailhead changed the title Remove on before render dep gltf loader Remove onBeforeRender dep gltf loader May 20, 2018
@pailhead
Copy link
Contributor Author

pailhead commented May 20, 2018

I agree, however, I mustered up an API example that is slightly less tedious and better suited for automation. #14031.

Unfortunately i didn't realize that there were lint issues with the GLTF loader so when i linted the entire file, it made the diff look scary. But this example should be cleaner.

Slightly cleaner PR to look at might be #13198

I think for the time being, it's a way to leverage more of the existing code, remove some redundancy and and decouple some concerns within the existing system. onBeforeCompile is a way to do so currently available.

examples/js/loaders/GLTFLoader.js Outdated Show resolved Hide resolved
shader.fragmentShader = shader.fragmentShader.replace( '#include <metalnessmap_fragment>', params.chunks.glossinessMapFragmentChunk );
shader.fragmentShader = shader.fragmentShader.replace( '#include <lights_physical_fragment>', params.chunks.lightPhysicalFragmentChunk );

};
Copy link
Collaborator

Choose a reason for hiding this comment

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

Would this still work if we extend MeshStandardMaterial with a new type, GLTFMeshStandardSGMaterial, and set GLTFMeshStandardSGMaterial.prototype.onBeforeCompile?

function GLTFMeshStandardSGMaterial (params) {
  MeshStandardMaterial.call(this, params);
  material.userData.extraUniforms = {
    KHR_MATERIALS_PBR_SPECULAR_GLOSSINESS: params.uniforms
  };
}

It would be nice to be able to return a normal constructor from getMaterialType, or at minimum to have an .isGLTFMeshStandardSGMaterial===true flag on the material for GLTFExporter to find.

Copy link
Contributor Author

@pailhead pailhead May 21, 2018

Choose a reason for hiding this comment

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

I'm not sure about the prototype, i think it might have to be bound in the constructor. This is one of those gotchas that i always have to console.log. I don't think onBeforeCompile exists on the prototype object at all.

Ideally i would like to not have to keep onBeforeCompile for too long in lieu of material includes. But for now, if this works, yes i think a new class is the best way to handle this.

glossiness: { value: 0.5 },
specularMap: { value: null },
glossinessMap: { value: null }
};
Copy link
Collaborator

Choose a reason for hiding this comment

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

I realize this isn't new with your changes, but let's use the defaults from the extension spec:

specular: 0xffffff
glossiness: 1.0

https://github.com/KhronosGroup/glTF/tree/master/extensions/2.0/Khronos/KHR_materials_pbrSpecularGlossiness

Copy link
Contributor Author

@pailhead pailhead May 21, 2018

Choose a reason for hiding this comment

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

Will do, if this looks good i can proceed to clean this diff up a bit. This is being set to that default somewhere else though, but that's probably cruft, it should be here. I also realized i haven't actually removed the onBeforeRender, there should be no mention of it in this file.

@takahirox
Copy link
Collaborator

takahirox commented May 22, 2018

Nice work.

I don't really like the current specular-glossiness material implementation, especially refreshUniforms. I'm sure we should remove duplicated refreshUniforms from GLTFLoader.

I tried onBeforeCompile, too, to simplify the design but I gave up because it can't handle additional uniforms. Piping the value from material to uniforms with setter/getter would be a good solution to solve this issue. (Let me know if there's any concerns about it)

The current implementation has another issue. refreshUniforms is bound to mesh.onBeforeRender, not material. When user wants to share the material with other mesh, one needs to be ware of the necessity of copying .onBeforeRender.

mesh2.material = mesh.material;
mesh2.onBeforeRender = mesh.onBeforeRender;

This solution can also solve this issue.

@pailhead
Copy link
Contributor Author

It's as tricky to clone something with onBeforeCompile as it is with onBeforeRender but at least the responsibility is in a more appropriate realm (material manages uniform, not an Object3D). I'll clean this up today or tomorrow.

I do have concerns though about onBeforeCompile i don't like how it behaves nor how tedious it is to use it. GLTFLoader happened to be a really good example where shader injection in general can better solve the problem, BUT it doesn't have to be with onBeforeCompile. This IMO is a superior way:
#13198

Would you be open to giving this other API some support after this lands?

@mrdoob mrdoob added this to the r94 milestone May 22, 2018
@@ -728,15 +759,20 @@ THREE.GLTFLoader = ( function () {
material.glossinessMap = params.glossinessMap === undefined ? null : params.glossinessMap;
material.glossiness = params.glossiness;

if ( material.glossinessMap ) {
Copy link
Contributor Author

Choose a reason for hiding this comment

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

im not sure what to do about this

@pailhead
Copy link
Contributor Author

pailhead commented May 26, 2018

Mkay, i took a stab at this hack for the roughness thing. I think with the alternative to onBeforeCompile this could be simplified. This feels hacky but works, should possibly set needsUpdate to true, but it would make it into a pretty side-effecty setter.

With this code:

var clone = object.clone();
clone.position.set(.025,0,0);
scene.add(clone);
clone.children[0].material = clone.children[0].material.clone()
console.log(clone.children[0].material)
clone.children[0].material.color.setStyle(`#ff0000`)

I was able to achieve this with spec gloss.
gltfclone

It would be cool if someone reviewed this, since this may be the most I could pick up, i'm not that fmiliar with the rest of the loader. There might also be some leftover cruft.

@takahirox @donmccurdy

@pailhead
Copy link
Contributor Author

pailhead commented May 26, 2018

And maybe link this here:
#14009

I'm not a CS person, but i'm trying to learn, i've been curious about the callback, why it makes sense, why it doesn't. I'm not sure where this implementation falls.

@pailhead
Copy link
Contributor Author

The further this is pushed out the more work it causes. Conflicts were solved already.

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.

Agreed that this improves the loader.

I want to test this on a few models still, but would like to work toward merging this.

* @pailhead
*/

function SpecularGlossinessPbrMaterial( params ) {
Copy link
Collaborator

Choose a reason for hiding this comment

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

I think we discussed possible names if a spec/gloss PBR material were added to three.js once before, and MeshStandardSGMaterial was the candidate? Perhaps let's use that.

].join( '\n' );

var uniforms = {
specular: { value: new THREE.Color().setHex( 0x111111 ) },
Copy link
Collaborator

Choose a reason for hiding this comment

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

Not a new issue with your code I realize, but could you make this 0xFFFFFF to match the spec?


var uniforms = {
specular: { value: new THREE.Color().setHex( 0x111111 ) },
glossiness: { value: 0.5 },
Copy link
Collaborator

Choose a reason for hiding this comment

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

Similarly, default in glTF should be 1.0 here.

this._extraUniforms = uniforms;

// please see #14031 or #13198 for an alternate approach
this.onBeforeCompile = function ( shader ) {
Copy link
Collaborator

Choose a reason for hiding this comment

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

If we put onBeforeCompile on the SpecularGlossinessPbrMaterial prototype, does it still work? Having it support cloning would probably make sense in this case.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I’m not sure I think it won’t like the prototype. But cloning this should still work since it calls the constructor. I’ll rename these look at the conflict and see where it leads us. Thank you.

Copy link
Collaborator

Choose a reason for hiding this comment

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

Ok, this looks good as-is.

@pailhead pailhead force-pushed the remove-onBeforeRender-dep-GLTFLoader branch from 6204e49 to c9b7607 Compare July 21, 2018 09:53
@pailhead
Copy link
Contributor Author

pailhead commented Jul 21, 2018

rebase yeey. With the messed up branch i had, it was even more fun. I think it's fixed now, and i've addressed the naming. @donmccurdy

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.

Looks good to me, thanks! 👍

You can also remove the now-unused cloneMaterial helper function.

this._extraUniforms = uniforms;

// please see #14031 or #13198 for an alternate approach
this.onBeforeCompile = function ( shader ) {
Copy link
Collaborator

Choose a reason for hiding this comment

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

Ok, this looks good as-is.

@pailhead
Copy link
Contributor Author

@donmccurdy

nuked cloneMaterial nice, shaved off more code


};

Object.defineProperties(
Copy link
Contributor Author

Choose a reason for hiding this comment

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

@donmccurdy

do you think it would be ok to add /*eslint-disable*/ for these lines, they're reaaaaly spaced out for getters and setters?

Copy link
Collaborator

Choose a reason for hiding this comment

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

Perhaps use one-liners like get: function () { return uniforms.specularMap.value; }? I'd be fine with that yeah. 👍


uniforms.glossinessMap.value = v;
//how about something like this - @pailhead
if ( v ) {
Copy link
Contributor Author

Choose a reason for hiding this comment

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

@donmccurdy did you catch this? It's a little bit quirky i think, since needsUpdate is also probably needed. I see two issues here, the general pattern with three.js when refreshing shaders, and then these defines in particular.

Copy link
Collaborator

@donmccurdy donmccurdy Jul 23, 2018

Choose a reason for hiding this comment

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

Oh thanks for calling that out, and the comment about USE_ROUGHNESSMAP. Requiring needsUpdate seems fine / I don't have any better ideas. 👍

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Excellent! Call it done for now :)

@donmccurdy donmccurdy changed the title Remove onBeforeRender dep gltf loader GLTFLoader: Remove onBeforeRender dep Jul 23, 2018
@pailhead
Copy link
Contributor Author

Is this still relevant? I can fix the conflicts but sinceI've fixed them several times in the past 15 months, it'd be cool if one of these milestones were actually upheld :)

@mrdoob
Copy link
Owner

mrdoob commented Jul 31, 2019

I'll let @takahirox and @donmccurdy decide if it's still relevant.

@mrdoob mrdoob modified the milestones: r107, r108 Jul 31, 2019
@pailhead
Copy link
Contributor Author

Damn, sorry I didn’t know this would be such a tough one.

@mrdoob mrdoob modified the milestones: r108, r109 Aug 27, 2019
@mrdoob mrdoob modified the milestones: r109, r110 Sep 23, 2019
@mrdoob mrdoob modified the milestones: r110, r111 Oct 30, 2019
@mrdoob mrdoob modified the milestones: r111, r112 Nov 27, 2019
@Mugen87
Copy link
Collaborator

Mugen87 commented Dec 18, 2019

I'll let @takahirox and @donmccurdy decide if it's still relevant.

Um, seems it's not. If this refactoring is still wanted, I suggest to create a new PR from scratch.

@Mugen87 Mugen87 closed this Dec 18, 2019
@Mugen87 Mugen87 removed this from the r112 milestone Dec 18, 2019
@pailhead
Copy link
Contributor Author

pailhead commented Dec 18, 2019

Could it please be elaborated a bit what made this irrelevant and what should the new PR created from scratch contain? Is it just the conflicts? I think it was fine conflict wise during these R103 period and was approved but didn’t make it. Some pointers for the new PR if it’s needed at all would be useful :) Thank you!

@donmccurdy @takahiro

@donmccurdy
Copy link
Collaborator

I've approved the PR, as it meaningfully simplifies the current code — I have no concerns with the changes.

@mrdoob
Copy link
Owner

mrdoob commented Dec 18, 2019

Is it just the conflicts?

Yes.

@mrdoob
Copy link
Owner

mrdoob commented Dec 18, 2019

Also, I feel @takahirox no longer has time to review GLTFLoader related PRs.

@pailhead
Copy link
Contributor Author

Sweet thank you for the info! I can look into this by the end of this week.

@pailhead
Copy link
Contributor Author

It looks like i'm unable to reopen this though.

@elalish
Copy link
Contributor

elalish commented Jan 24, 2020

@pailhead Thanks for fixing my issue before it even appeared! I fixed your conflicts and your work was just merged at long last, so you're welcome to close this one. Sorry for the wait.

@mrdoob
Copy link
Owner

mrdoob commented Jan 24, 2020

@elalish Merging your PR merged this PR 😀

@WestLangley
Copy link
Collaborator

@pailhead Thank you ! :-)

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.

7 participants