This little code will patch all material classes to have a chainable onBeforeCompile pattern, so when you assign a callback to onBeforeCompile it will be pushed into a plugins array property on the material instead overriding it. Assign null to reset or use removePlugin to remove a specific. Also addPlugin (same as assining) and hasPlugin is available.
It doesn’t break the API and works without core modification, so WebGLPrograms.js will get a combined string of all functions.
You can also set a priority on the patch function, for example for a patch patching another patch that should be called before, but especially on the code itself, if you use #include ..
as entry point to insert the patch code order might matter. If you set DecalPatch.priority = 1;
to 0 you’ll see the TintBluePatch
will be applied in the shader after DecalPatch
just like the order you assigned it and turn it fully blue, with higher priority a callback dominates it’s descendants. Of course this logic depends on how code is inserted, such as before a standard line or after.
So you can do the following without overriding the other:
material.onBeforeCompile = function( shader ) { ... };
material.onBeforeCompile = function( shader ) { ... };
or
material.onBeforeCompile = [
function( shader ) { ... },
function( shader ) { ... }
];
Below you find a example how you can make a more advanced plugin.
Code (latest): https://github.com/Fyrestar/MaterialPlugin
Basic example:
Simple waving grass plugin:
Material Callbacks
When extending shaders with onBeforeCompile it only gives you acces to the uniforms once. This allows you to not only use uniforms per material but also per object, like in this example:
So you can extend materials in a actual plugin pattern and pass properties of meshes or materials to the shader.
Where you store per-mesh properties is up to you, you can either use the mesh directly like in the example (might be less optimal), store it in it’s userData object, or extend the mesh class (optimal).
Call THREE.MaterialCallback.use();
to apply it to the prototype of Mesh and SkinnedMesh, if you want it manually or need onBeforeRender yourself, just call within your callback:
THREE.MaterialCallback.call( this, renderer, scene, ... )