Skip to content

Commit 97cd9ac

Browse files
committed
fix(troika-three-utils): fragmentColorTransform is now inserted prior to postprocessing chunks
Derived materials used to inject fragmentColorTransform after any of the builtin postprocessing shader chunks like tonemapping/fog/etc. This fixes that so the user's color transform is still subject to those final bits of processing. Closes issue #20.
1 parent 7ac9e95 commit 97cd9ac

File tree

1 file changed

+25
-5
lines changed

1 file changed

+25
-5
lines changed

packages/troika-three-utils/src/DerivedMaterial.js

+25-5
Original file line numberDiff line numberDiff line change
@@ -56,10 +56,13 @@ const CACHE = new WeakMap() //threejs requires WeakMap internally so should be s
5656
* @param {String} options.fragmentMainIntro - Custom GLSL code to inject at the top of the fragment
5757
* shader's `void main` function.
5858
* @param {String} options.fragmentMainOutro - Custom GLSL code to inject at the end of the fragment
59-
* shader's `void main` function.
59+
* shader's `void main` function. You can manipulate `gl_FragColor` here but keep in mind it goes
60+
* after any of ThreeJS's color postprocessing shader chunks (tonemapping, fog, etc.), so if you
61+
* want those to apply to your changes use `fragmentColorTransform` instead.
6062
* @param {String} options.fragmentColorTransform - Custom GLSL code to manipulate the `gl_FragColor`
61-
* output value. Will be injected after all other `void main` logic has executed but just before
62-
* the `fragmentMainOutro`. TODO allow injecting in other places?
63+
* output value. Will be injected near the end of the `void main` function, but before any
64+
* of ThreeJS's color postprocessing shader chunks (tonemapping, fog, etc.), and before the
65+
* `fragmentMainOutro`.
6366
* @param {function<{vertexShader,fragmentShader}>:{vertexShader,fragmentShader}} options.customRewriter - A function
6467
* for performing custom rewrites of the full shader code. Useful if you need to do something
6568
* special that's not covered by the other builtin options. This function will be executed before
@@ -250,6 +253,14 @@ function upgradeShaders({vertexShader, fragmentShader}, options, id) {
250253
vertexShader = expandShaderIncludes(vertexShader)
251254
}
252255
if (fragmentColorTransform || customRewriter) {
256+
// We need to be able to find postprocessing chunks after include expansion in order to
257+
// put them after the fragmentColorTransform, so mark them with comments first. Even if
258+
// this particular derivation doesn't have a fragmentColorTransform, other derivations may,
259+
// so we still mark them.
260+
fragmentShader = fragmentShader.replace(
261+
/^[ \t]*#include <((?:tonemapping|encodings|fog|premultiplied_alpha|dithering)_fragment)>/gm,
262+
'\n//!BEGIN_POST_CHUNK $1\n$&\n//!END_POST_CHUNK\n'
263+
)
253264
fragmentShader = expandShaderIncludes(fragmentShader)
254265
}
255266

@@ -260,9 +271,18 @@ function upgradeShaders({vertexShader, fragmentShader}, options, id) {
260271
fragmentShader = res.fragmentShader
261272
}
262273

263-
// Treat fragmentColorTransform as an outro
274+
// The fragmentColorTransform needs to go before any postprocessing chunks, so extract
275+
// those and re-insert them into the outro in the correct place:
264276
if (fragmentColorTransform) {
265-
fragmentMainOutro = `${fragmentColorTransform}\n${fragmentMainOutro}`
277+
let postChunks = []
278+
fragmentShader = fragmentShader.replace(
279+
/^\/\/!BEGIN_POST_CHUNK[^]+?^\/\/!END_POST_CHUNK/gm, // [^]+? = non-greedy match of any chars including newlines
280+
match => {
281+
postChunks.push(match)
282+
return ''
283+
}
284+
)
285+
fragmentMainOutro = `${fragmentColorTransform}\n${postChunks.join('\n')}\n${fragmentMainOutro}`
266286
}
267287

268288
// Inject auto-updating time uniform if requested

0 commit comments

Comments
 (0)