Применение шагов постобработки для определенных объектов

Основываясь на этом примере, я пытаюсь создать сцену, где несколько объектов расцветают, а другие нет.

Белый куб в середине должен быть просто белым (без цветения)

Я запутался в том, как получить желаемый результат. Я, например, попытался добавить 2-ю сцену с белым кубом, но, похоже, я не могу правильно оформить заказ. Может быть, мой подход с разными сценами не так?

Каков "лучший" способ добиться такого поведения? Я всегда заканчиваю тем, что вижу только одну сцену, только белый куб или четыре цветных. (пример ниже показывает все атм)

myFiddle: https://jsfiddle.net/qwertasyx/8qw3ys4z/16/

var scene,scene2,camera, controls, pointLight, stats;    
var composer, renderer, mixer;

var params = {
  exposure: 1,
  bloomStrength: 1.5,
  bloomThreshold: 0,
  bloomRadius: 0
};

var objects = [];

var clock = new THREE.Clock();
var container = document.getElementById( 'container' );

stats = new Stats();
//container.appendChild( stats.dom );

renderer = new THREE.WebGLRenderer( { antialias: true } );
renderer.setPixelRatio( window.devicePixelRatio );
renderer.setSize( window.innerWidth, window.innerHeight );
renderer.toneMapping = THREE.ReinhardToneMapping;
container.appendChild( renderer.domElement );

scene  = new THREE.Scene();
//scene2 = new THREE.Scene();

camera = new THREE.PerspectiveCamera( 75, window.innerWidth / window.innerHeight, 1, 100 );
camera.position.set( 2.5,2.5, 10 );
scene.add( camera );
// scene2.add( camera );

controls = new THREE.OrbitControls( camera, renderer.domElement );
controls.maxPolarAngle = Math.PI * 0.5;
controls.minDistance = 1;
controls.maxDistance = 10;
controls.target.set(2.5,2.5,0)
controls.update()

// scene.add( new THREE.AmbientLight( 0x404040 ) );

pointLight = new THREE.PointLight( 0xffffff, 1 );
// camera.add( pointLight );

var renderScene  = new THREE.RenderPass( scene, camera );
//var renderScene2 = new THREE.RenderPass( scene2, camera );



var bloomPass = new THREE.UnrealBloomPass( new THREE.Vector2( window.innerWidth, window.innerHeight ), 1.5, 0.4, 0.85 );
bloomPass.renderToScreen = true;
bloomPass.threshold = params.bloomThreshold;
bloomPass.strength  = params.bloomStrength;
bloomPass.radius    = params.bloomRadius;

composer = new THREE.EffectComposer( renderer );
composer.setSize( window.innerWidth, window.innerHeight );
composer.addPass( renderScene );

composer.addPass( bloomPass );
//composer.addPass( renderScene2 );


//objects
var geometry = new THREE.BoxGeometry( 1, 1, 1 );
var material = new THREE.MeshBasicMaterial( { color: 0xffff00 } );
var cube = new THREE.Mesh( geometry, material );
cube.vrz = 0.01;
cube.position.x += 5
scene.add( cube );
objects.push(cube)

var material = new THREE.MeshBasicMaterial( { color: 0xff0000 } );
var cube = new THREE.Mesh( geometry, material );
cube.vrz = 0.01;
cube.position.x += 5
cube.position.y += 5
scene.add( cube );
objects.push(cube)

var material = new THREE.MeshBasicMaterial( { color: 0x00ff00 } );
var cube = new THREE.Mesh( geometry, material );
cube.vrz = 0.01;
cube.position.y += 5
scene.add( cube );
objects.push(cube)

var material = new THREE.MeshBasicMaterial( { color: 0x0000ff } );
var cube = new THREE.Mesh( geometry, material );
cube.vrz = 0.01;
scene.add( cube );
objects.push(cube)

// cube thats supposed to be not bloomy 
var material = new THREE.MeshBasicMaterial( { color: 0xffffff } );
var cube = new THREE.Mesh( geometry, material );
cube.vrz = 0.01;
cube.position.y += 2.5
cube.position.x += 2.5
scene.add( cube );
objects.push(cube)

var gui = new dat.GUI();

gui.add( params, 'exposure', 0.1, 2 ).onChange( function ( value ) {

  renderer.toneMappingExposure = Math.pow( value, 4.0 );

} );

gui.add( params, 'bloomThreshold', 0.0, 1.0 ).onChange( function ( value ) {

  bloomPass.threshold = Number( value );

} );

gui.add( params, 'bloomStrength', 0.0, 3.0 ).onChange( function ( value ) {

  bloomPass.strength = Number( value );

} );

gui.add( params, 'bloomRadius', 0.0, 1.0 ).step( 0.01 ).onChange( function ( value ) {

  bloomPass.radius = Number( value );

} );

window.onresize = function () {

  var width = window.innerWidth;
  var height = window.innerHeight;

  camera.aspect = width / height;
  camera.updateProjectionMatrix();

  renderer.setSize( width, height );
  composer.setSize( width, height );

};


function animate() {

  requestAnimationFrame( animate );

  objects.forEach(function(obj){
    obj.rotation.z += obj.vrz;
  });

  stats.update();
  composer.render();

}
animate();

1 ответ

У меня была похожая проблема однажды. Пример из этого комментария помог мне.

Обратите внимание, что в этом примере есть 2 сцены и 2 композитора (конечный композитор получает выходные данные предыдущего композитора в качестве входных данных)

ppoFinal.blendPass.uniforms.tAdd.value = ppoRGB.composer.renderTarget2.texture;

а также render() называется на обоих композиторов.

ppoRGB.composer.render();
ppoFinal.composer.render();

Этот шаблон позволяет вам применять эффекты постобработки выборочно, и он работает хорошо. Проблема заключается в масштабируемости метода и, возможно, производительности. Потому что, когда вы хотите применить другой объект с другим эффектом, вам нужно представить третью сцену и третий композитор. Для моего маленького проекта в прошлом я получил 4 сцены и 4 композитора...

Другие вопросы по тегам