ТРИ. JS | GLSL Установить цвет частиц по текстуре сцены - PullRequest
2 голосов
/ 01 мая 2020

У меня есть матрица частиц GLSL (THREE.Points), которые должны быть окрашены общей текстурой сцены.

enter image description here

Итак, результат должно быть так:

enter image description here

Да, я могу создать холст и установить buffer.color с помощью ctx.getImageData (left, top, width, height) , но возможно ли установить его внутри шейдера?

Я пытался получить глобальную позицию частицы, но безуспешно.

Вывод:

enter image description here

    
var blob = "";
      
var pattern = "";
    
var width = 512,
    height = 512;

inits();
    
function inits(){

renderer = new THREE.WebGLRenderer();
renderer.setSize(width, height);
document.body.appendChild(renderer.domElement);
    
scene = new THREE.Scene();
camera = new THREE.OrthographicCamera( - 1, 1, 1, - 1, 0, 1 );
scene.add(camera);

var N = 8.0;
    
var points = [];
var colors = [];
    
for(var y = 16; y <= height - 16; y += N){
    for(var x = 16; x <= width - 16; x += N){
        
        var dx = -1.0 + x / width * 2;
        var dy = -1.0 + y / height * 2;

        points.push(new THREE.Vector2(dx, dy));
         
    }
}
    
buffer = {

    position: new Float32Array(points.length * 3),
    color: new Float32Array(points.length * 3),
    radius: new Float32Array(points.length * 1),

};
    
for(var i = 0; i < points.length; i++){
    
    buffer.position[i * 3] = points[i].x;
    buffer.position[i * 3 + 1] = points[i].y;
    buffer.position[i * 3 + 2] = 0;
    
    buffer.radius[i] = N * 0.5;
    
    buffer.color[i * 3] = 1.0;
    buffer.color[i * 3 + 1] = 0.0;
    buffer.color[i * 3 + 2] = 1.0;
}
    
uniforms = { 
    
    blob: { value: new THREE.TextureLoader().load(blob) },
    pattern: { value: new THREE.TextureLoader().load(pattern) }
    
};

var material = new THREE.ShaderMaterial( {

    uniforms: uniforms,
    vertexShader: document.getElementById( 'vertexShader' ).textContent,
    fragmentShader: document.getElementById( 'fragmentShader' ).textContent,
    transparent: true
    
} );
    
material.extensions.fragDepth = true;
material.extensions.drawBuffers = true;

var cloud = new THREE.BufferGeometry();

cloud.addAttribute('position', new THREE.BufferAttribute(buffer.position, 3));
cloud.addAttribute('radius', new THREE.BufferAttribute(buffer.radius, 1));
cloud.addAttribute('color', new THREE.BufferAttribute(buffer.color, 3));
    
var points = new THREE.Points(cloud, material);
points.dynamic = true;
points.frustumCulled = false;

scene.add(points);
    
animate();
    
}

function animate() {

  renderer.render( scene, camera );
  requestAnimationFrame( animate );
    
}
<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <title>PixelWordMap Demo</title>
  <meta name="viewport" content="initial-scale=1,maximum-scale=1,user-scalable=no" />
  <meta name="author" content="Vladimir V. KUCHINOV">
  <link rel="stylesheet" href="styles.css" charset="utf-8">

  <script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/94/three.js"></script>

    
</head>

<body>
    
<script type='x-shader/x-vertex' id='vertexShader'>

      uniform sampler2D blob;
      uniform sampler2D pattern;
      attribute float radius;
      attribute vec3 color;
      
      varying vec3 vColor;
      varying vec2 pos;
        
      void main() {
      
        vColor = color;
        
        vec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );
        gl_PointSize = radius;
        pos = vec2(1.0 + position.x / 2.0, 1.0 + position.y / 2.0);
        gl_Position = projectionMatrix * mvPosition;
        
      }
      
</script>
    
<script type='x-shader/x-fragment' id='fragmentShader'>

      uniform sampler2D blob;
      uniform sampler2D pattern;
      varying vec3 vColor;
      varying vec2 pos;
      
      void main() {
      
          vec4 color = texture2D(pattern, vec2(pos.x, pos.y));
          gl_FragColor = color * texture2D(blob, gl_PointCoord);
          if(gl_FragColor.a < 0.1) discard; 
          
      }

</script>

</body>
</html>

1 Ответ

2 голосов
/ 01 мая 2020

Выходным данным вершинного шейдера, варьирующимся vec2 pos;, никогда не присваивается значение, поскольку позиция назначается локальной переменной pos с тем же именем: vec2 pos = vec2(...);.
Кроме того, вы пропустили фигурные скобки, чтобы расставить приоритеты + операция:

pos = (position.xy + 1.0) / 2.0;

Выберите другое число для N to get a different количества точек:

var N = 32;  
for(var y = N/2; y <= height; y += N){
    for(var x = N/2; x <= width; x += N){
        var dx = -1.0 + x / width * 2;
        var dy = -1.0 + y / height * 2;
        points.push(new THREE.Vector2(dx, dy));
    }
}

Обратите внимание, что размер точки ограничен. См. Обход ограничений gl_PointSize в трех. js / webGL .

var blob = "";
      
var pattern = "";
    
var width = 512,
    height = 512;

inits();
    
function inits(){

renderer = new THREE.WebGLRenderer();
renderer.setSize(width, height);
document.body.appendChild(renderer.domElement);
    
scene = new THREE.Scene();
camera = new THREE.OrthographicCamera( - 1, 1, 1, - 1, 0, 1 );
scene.add(camera);

  
var points = [];
var colors = [];
    
var N = 32;  
for(var y = N/2; y <= height; y += N){
    for(var x = N/2; x <= width; x += N){
        var dx = -1.0 + x / width * 2;
        var dy = -1.0 + y / height * 2;
        points.push(new THREE.Vector2(dx, dy));
    }
}
    
buffer = {

    position: new Float32Array(points.length * 3),
    color: new Float32Array(points.length * 3),
    radius: new Float32Array(points.length * 1),

};
    
for(var i = 0; i < points.length; i++){
    
    buffer.position[i * 3] = points[i].x;
    buffer.position[i * 3 + 1] = points[i].y;
    buffer.position[i * 3 + 2] = 0;
    
    buffer.radius[i] = N * 0.5;
    
    buffer.color[i * 3] = 1.0;
    buffer.color[i * 3 + 1] = 0.0;
    buffer.color[i * 3 + 2] = 1.0;
}
    
uniforms = { 
    
    blob: { value: new THREE.TextureLoader().load(blob) },
    pattern: { value: new THREE.TextureLoader().load(pattern) }
    
};

var material = new THREE.ShaderMaterial( {

    uniforms: uniforms,
    vertexShader: document.getElementById( 'vertexShader' ).textContent,
    fragmentShader: document.getElementById( 'fragmentShader' ).textContent,
    transparent: true
    
} );
    
material.extensions.fragDepth = true;
material.extensions.drawBuffers = true;

var cloud = new THREE.BufferGeometry();

cloud.addAttribute('position', new THREE.BufferAttribute(buffer.position, 3));
cloud.addAttribute('radius', new THREE.BufferAttribute(buffer.radius, 1));
cloud.addAttribute('color', new THREE.BufferAttribute(buffer.color, 3));
    
var points = new THREE.Points(cloud, material);
points.dynamic = true;
points.frustumCulled = false;

scene.add(points);
    
animate();
    
}

function animate() {

  renderer.render( scene, camera );
  requestAnimationFrame( animate );
    
}
<script src="https://cdn.jsdelivr.net/npm/three@0.115/build/three.js"></script>

<script type='x-shader/x-vertex' id='vertexShader'>
uniform sampler2D blob;
uniform sampler2D pattern;
attribute float radius;
attribute vec3 color;

varying vec3 vColor;
varying vec2 pos;

void main() {

    vColor = color;

    vec4 mvPosition = modelViewMatrix * vec4( position, 1.0 );
    gl_PointSize = radius;
    pos = (position.xy + 1.0) / 2.0;
    gl_Position = projectionMatrix * mvPosition;
}
</script>

<script type='x-shader/x-fragment' id='fragmentShader'>
uniform sampler2D blob;
uniform sampler2D pattern;
varying vec3 vColor;
varying vec2 pos;

void main() {

    vec4 color = texture2D(pattern, vec2(pos.x, pos.y));
    gl_FragColor = color * texture2D(blob, gl_PointCoord);
    if(gl_FragColor.a < 0.1) discard; 
}
</script>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...