在使用片段着色器应用波浪效果时,防止在WebGL中出现纹理边界扭曲

图加亚克

我在项目中通过Three.js使用GLSL。

我有这样的标志纹理:

当我应用“片段着色器”(Fragment Shader)使其具有波浪效果时,纹理的边界似乎会拉伸。但是,我不想要那个。我希望看到背景颜色(在这种情况下为透明),以便看起来就像挥舞的旗帜。您可以在CodePen看到问题在这里

供参考(如果CodePen由于某种原因将来无法使用),下面是代码:

顶点着色器

varying vec2 vUv;

void main() {
  vUv = uv;

  gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);
}

片段着色器

uniform vec2 u_resolution;
uniform float u_time;
uniform sampler2D image;

uniform vec2 strength;
uniform float speed;

varying vec2 vUv;

vec2 sineWave( vec2 p ) {
  float x = strength.x * sin(-1.0 * p.y + 50.0 * p.x + u_time * speed);
  float y = strength.y * sin(p.y + 10.0 * p.x + u_time * speed);
  return vec2(p.x, p.y + y);
}

void main() {
  vec4 color = texture2D(image, sineWave(vUv));
  gl_FragColor = color;
}

Three.js代码

const CANVAS_WIDTH = 250;
const CANVAS_HEIGHT = 250;

const imageDataUri = '';

const vertexShader = () => {
  return `
    varying vec2 vUv;

    void main() {
      vUv = uv;

      gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);
    }
  `
};

const fragmentShader = () => {
  return `
    uniform vec2 u_resolution;
    uniform float u_time;
    uniform sampler2D image;

    uniform vec2 strength;
    uniform float speed;

    varying vec2 vUv;

    vec2 sineWave( vec2 p ) {
      float x = strength.x * sin(-1.0 * p.y + 50.0 * p.x + u_time * speed);
      float y = strength.y * sin(p.y + 10.0 * p.x + u_time * speed);
      return vec2(p.x, p.y + y);
    }

    void main() {
      vec4 color = texture2D(image, sineWave(vUv));
      gl_FragColor = color;
    }
  `
};

function init() {
    let loader = new THREE.TextureLoader();

    loader.load(
      imageDataUri,
      (texture) => {
        let scene = new THREE.Scene();
        let camera =
          new THREE.PerspectiveCamera(75, CANVAS_WIDTH / CANVAS_HEIGHT, 0.1, 1000);
        let renderer = new THREE.WebGLRenderer({ alpha: true });
        renderer.setClearColor(0x000000, 0);

        renderer.setSize(CANVAS_WIDTH, CANVAS_HEIGHT);

				document.body.appendChild(renderer.domElement);
        
        let uniforms = {
          u_time: { type: 'f', value: 0.1 },
          u_resolution: { type: 'v2', value: new THREE.Vector2() },
          image: { type: 't', value: texture },
          strength: { type: 'v2', value: new THREE.Vector2(0.01, 0.025) },
          speed: { type: 'f', value: 8.0 }
        };
        let material = new THREE.ShaderMaterial({
          uniforms,
          vertexShader: vertexShader(),
          fragmentShader: fragmentShader()
        });
let triangleMesh = new THREE.Mesh(new THREE.PlaneGeometry(CANVAS_WIDTH, CANVAS_HEIGHT, 1, 1), material);

        scene.add(triangleMesh);
        camera.position.z = 250;

        let clock = new THREE.Clock();
        let animate = () => {
          requestAnimationFrame(animate);

          triangleMesh.material.uniforms.u_time.value += clock.getDelta();

          renderer.render(scene, camera);
        };

        animate();
      },
      undefined,
      (err) => { console.error('Could not load texture', err) }
    );

}

init();
body {background: orange;}
<script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/110/three.min.js"></script>

拉比德76

您可以做的是根据纹理v坐标缩放波浪效果如果按比例缩放,1.0-p.y则波效果在顶部边界处为零,在底部处具有最大值。

float y = strength.y * sin(p.y + 10.0 * p.x + u_time * speed) * (1.0-p.y*p.y);

为了在标志的顶部区域产生较小的波动影响,请按v坐标((1.0-p.y*p.y)的平方进行缩放

float y = strength.y * sin(p.y + 10.0 * p.x + u_time * speed) * (1.0-p.y*p.y);

const CANVAS_WIDTH = 250;
const CANVAS_HEIGHT = 250;

const imageDataUri = '';

const vertexShader = () => {
  return `
    varying vec2 vUv;

    void main() {
      vUv = uv;

      gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);
    }
  `
};

const fragmentShader = () => {
  return `
    uniform vec2 u_resolution;
    uniform float u_time;
    uniform sampler2D image;

    uniform vec2 strength;
    uniform float speed;

    varying vec2 vUv;

    vec2 sineWave( vec2 p ) {
      float x = strength.x * sin(-1.0 * p.y + 50.0 * p.x + u_time * speed);
      float y = strength.y * sin(p.y + 10.0 * p.x + u_time * speed) * (1.0-p.y*p.y);
      return vec2(p.x, p.y + y);
    }

    void main() {
      vec4 color = texture2D(image, sineWave(vUv));
      gl_FragColor = color;
    }
  `
};

function init() {
    let loader = new THREE.TextureLoader();

    loader.load(
      imageDataUri,
      (texture) => {
        let scene = new THREE.Scene();
        let camera =
          new THREE.PerspectiveCamera(75, CANVAS_WIDTH / CANVAS_HEIGHT, 0.1, 1000);
        let renderer = new THREE.WebGLRenderer({ alpha: true });
        renderer.setClearColor(0x000000, 0);

        renderer.setSize(CANVAS_WIDTH, CANVAS_HEIGHT);

				document.body.appendChild(renderer.domElement);
        
        let uniforms = {
          u_time: { type: 'f', value: 0.1 },
          u_resolution: { type: 'v2', value: new THREE.Vector2() },
          image: { type: 't', value: texture },
          strength: { type: 'v2', value: new THREE.Vector2(0.01, 0.025) },
          speed: { type: 'f', value: 8.0 }
        };
        let material = new THREE.ShaderMaterial({
          uniforms,
          vertexShader: vertexShader(),
          fragmentShader: fragmentShader()
        });
let triangleMesh = new THREE.Mesh(new THREE.PlaneGeometry(CANVAS_WIDTH, CANVAS_HEIGHT, 1, 1), material);

        scene.add(triangleMesh);
        camera.position.z = 250;

        let clock = new THREE.Clock();
        let animate = () => {
          requestAnimationFrame(animate);

          triangleMesh.material.uniforms.u_time.value += clock.getDelta();

          renderer.render(scene, camera);
        };

        animate();
      },
      undefined,
      (err) => { console.error('Could not load texture', err) }
    );

}

init();
body {background: orange;}
<script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/110/three.min.js"></script>

本文收集自互联网,转载请注明来源。

如有侵权,请联系[email protected] 删除。

编辑于
0

我来说两句

0条评论
登录后参与评论

相关文章

来自分类Dev

使用Vulkan在片段着色器中缓慢获取纹理

来自分类Dev

使用着色器在画布上镜像效果(WebGL)

来自分类Dev

片段着色器忽略纹理坐标

来自分类Dev

片段着色器中的纹理读取

来自分类Dev

在片段着色器中的OpenGL中更改纹理的颜色效果不佳

来自分类Dev

WebGL / three.js片段着色器-用alpha透明色覆盖alpha透明纹理

来自分类Dev

WebGL / three.js片段着色器-用alpha透明色覆盖alpha透明纹理

来自分类Dev

使用着色器时纹理无法渲染?

来自分类Dev

如何在着色器中制作波形扭曲效果?

来自分类Dev

鱼眼扭曲鼠标位置-片段着色器

来自分类Dev

GLSL着色器可提供纹理“烟熏”效果

来自分类Dev

GLSL着色器可提供纹理“烟熏”效果

来自分类Dev

webgl:与其他程序/着色器共享纹理

来自分类Dev

GLSL片段着色器可智能地平铺纹理

来自分类Dev

glsl片段着色器计算纹理位置

来自分类Dev

OpenGL片段着色器返回纹理的固定颜色

来自分类Dev

片段着色器中的OpenGL纹理为黑色

来自分类Dev

在GLSL着色器中使用多个纹理时,纹理被覆盖

来自分类Dev

为什么用片段着色器进行纹理映射时会在纹理坐标跳跃时生成线条?

来自分类Dev

为什么在使用着色器时出现黑屏?

来自分类Dev

将WebGL片段着色器转换为GLES

来自分类Dev

WebGL-丢弃顶点着色器标记的片段

来自分类Dev

在WebGL片段着色器中绘制网格

来自分类Dev

WebGL片段着色器可用于多个光源?

来自分类Dev

WebGL片段着色器不透明度

来自分类Dev

在WebGL片段着色器中绘制网格

来自分类Dev

WebGL片段着色器可用于多个光源吗?

来自分类Dev

webgl使用帧缓冲对象在图像上的多个片段着色器产生黑色输出

来自分类Dev

使用变化的Webgl将属性传递给片段着色器

Related 相关文章

  1. 1

    使用Vulkan在片段着色器中缓慢获取纹理

  2. 2

    使用着色器在画布上镜像效果(WebGL)

  3. 3

    片段着色器忽略纹理坐标

  4. 4

    片段着色器中的纹理读取

  5. 5

    在片段着色器中的OpenGL中更改纹理的颜色效果不佳

  6. 6

    WebGL / three.js片段着色器-用alpha透明色覆盖alpha透明纹理

  7. 7

    WebGL / three.js片段着色器-用alpha透明色覆盖alpha透明纹理

  8. 8

    使用着色器时纹理无法渲染?

  9. 9

    如何在着色器中制作波形扭曲效果?

  10. 10

    鱼眼扭曲鼠标位置-片段着色器

  11. 11

    GLSL着色器可提供纹理“烟熏”效果

  12. 12

    GLSL着色器可提供纹理“烟熏”效果

  13. 13

    webgl:与其他程序/着色器共享纹理

  14. 14

    GLSL片段着色器可智能地平铺纹理

  15. 15

    glsl片段着色器计算纹理位置

  16. 16

    OpenGL片段着色器返回纹理的固定颜色

  17. 17

    片段着色器中的OpenGL纹理为黑色

  18. 18

    在GLSL着色器中使用多个纹理时,纹理被覆盖

  19. 19

    为什么用片段着色器进行纹理映射时会在纹理坐标跳跃时生成线条?

  20. 20

    为什么在使用着色器时出现黑屏?

  21. 21

    将WebGL片段着色器转换为GLES

  22. 22

    WebGL-丢弃顶点着色器标记的片段

  23. 23

    在WebGL片段着色器中绘制网格

  24. 24

    WebGL片段着色器可用于多个光源?

  25. 25

    WebGL片段着色器不透明度

  26. 26

    在WebGL片段着色器中绘制网格

  27. 27

    WebGL片段着色器可用于多个光源吗?

  28. 28

    webgl使用帧缓冲对象在图像上的多个片段着色器产生黑色输出

  29. 29

    使用变化的Webgl将属性传递给片段着色器

热门标签

归档