@@ -56,10 +56,13 @@ const CACHE = new WeakMap() //threejs requires WeakMap internally so should be s
56
56
* @param {String } options.fragmentMainIntro - Custom GLSL code to inject at the top of the fragment
57
57
* shader's `void main` function.
58
58
* @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.
60
62
* @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`.
63
66
* @param {function<{vertexShader,fragmentShader}>:{vertexShader,fragmentShader} } options.customRewriter - A function
64
67
* for performing custom rewrites of the full shader code. Useful if you need to do something
65
68
* 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) {
250
253
vertexShader = expandShaderIncludes ( vertexShader )
251
254
}
252
255
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 ] * # i n c l u d e < ( (?: t o n e m a p p i n g | e n c o d i n g s | f o g | p r e m u l t i p l i e d _ a l p h a | d i t h e r i n g ) _ f r a g m e n t ) > / gm,
262
+ '\n//!BEGIN_POST_CHUNK $1\n$&\n//!END_POST_CHUNK\n'
263
+ )
253
264
fragmentShader = expandShaderIncludes ( fragmentShader )
254
265
}
255
266
@@ -260,9 +271,18 @@ function upgradeShaders({vertexShader, fragmentShader}, options, id) {
260
271
fragmentShader = res . fragmentShader
261
272
}
262
273
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:
264
276
if ( fragmentColorTransform ) {
265
- fragmentMainOutro = `${ fragmentColorTransform } \n${ fragmentMainOutro } `
277
+ let postChunks = [ ]
278
+ fragmentShader = fragmentShader . replace (
279
+ / ^ \/ \/ ! B E G I N _ P O S T _ C H U N K [ ^ ] + ?^ \/ \/ ! E N D _ P O S T _ C H U N K / 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 } `
266
286
}
267
287
268
288
// Inject auto-updating time uniform if requested
0 commit comments