【问题标题】:How to flip Y-axis in GLSL shader before gl_FragColor = texture2D(*,*);如何在 gl_FragColor = texture2D(*,*) 之前在 GLSL 着色器中翻转 Y 轴;
【发布时间】:2015-10-22 14:21:25
【问题描述】:

在应用透视变换之前,我需要在着色器中颠倒我的纹理。我在 vert.glsl 中修改了 vertTexCoord,但不知道在 swap.glsl 的什么地方使用它。这样做的方法就像

gl_FragColor = texture2D(texture, vertTexCoord );

不起作用,因为我还需要透视修改纹理。

vert.glsl:

#define PROCESSING_COLOR_SHADER

uniform mat4 transform;
uniform mat4 texMatrix;

attribute vec4 vertex;
attribute vec4 color;
attribute vec2 texCoord;

varying vec4 vertColor;
varying vec4 vertTexCoord;

void main() {
  gl_Position = transform * vertex;

  vertColor = color;
  vertTexCoord = texMatrix * vec4(texCoord, 1.0, 1.0);
}

swap.glsl:

#ifdef GL_ES
precision highp float;
#endif

// General parameters
uniform sampler2D from;
uniform sampler2D to;
uniform float progress;
uniform vec2 resolution;

uniform float reflection;
uniform float perspective;
uniform float depth;

varying vec4 vertColor;
varying vec4 vertTexCoord;

const vec4 black = vec4(0.0, 0.0, 0.0, 1.0);
const vec2 boundMin = vec2(0.0, 0.0);
const vec2 boundMax = vec2(1.0, 1.0);

bool inBounds (vec2 p) {
  return all(lessThan(boundMin, p)) && all(lessThan(p, boundMax));
}

vec2 project (vec2 p) {
  return p * vec2(1.0, -1.2) + vec2(0.0, -0.02);
}

vec4 bgColor (vec2 p, vec2 pfr, vec2 pto) {
  vec4 c = black;
  pfr = project(pfr);
  if (inBounds(pfr)) {
    c += mix(black, texture2D(from, pfr), reflection * mix(1.0, 0.0, pfr.y));
  }
  pto = project(pto);
  if (inBounds(pto)) {
    c += mix(black, texture2D(to, pto), reflection * mix(1.0, 0.0, pto.y));
  }
  return c;
}

void main() {
  vec2 p = gl_FragCoord.xy / resolution.xy;  
  vec2 pfr, pto = vec2(-1.);

  float size = mix(1.0, depth, progress);
  float persp = perspective * progress;
  pfr = (p + vec2(-0.0, -0.5)) * vec2(size/(1.0-perspective*progress), size/(1.0-size*persp*p.x)) + vec2(0.0, 0.5);

  size = mix(1.0, depth, 1.-progress);
  persp = perspective * (1.-progress);
  pto = (p + vec2(-1.0, -0.5)) * vec2(size/(1.0-perspective*(1.0-progress)), size/(1.0-size*persp*(0.5-p.x))) + vec2(1.0, 0.5);

  bool fromOver = progress < 0.5;

  if (fromOver) {
    if (inBounds(pfr)) {
      gl_FragColor = texture2D(from, pfr);
    }
    else if (inBounds(pto)) {
      gl_FragColor = texture2D(to, pto);
    }
    else {
      gl_FragColor = bgColor(p, pfr, pto);
    }
  }
  else {
    if (inBounds(pto)) {
      gl_FragColor = texture2D(to, pto);
    }
    else if (inBounds(pfr)) {
      gl_FragColor = texture2D(from, pfr);
    }
    else {
      gl_FragColor = bgColor(p, pfr, pto);
    }
  }
}

【问题讨论】:

  • texMatrix 是做什么的?
  • "在顶点着色器中有一个名为 texMatrix 的新统一,它重新调整每个顶点的纹理坐标(在附加属性 texCoord 中传递),以考虑纹理沿 Y 轴倒置(因为处理的垂直轴相对于 OpenGL 倒置)”,如 processing.org/tutorials/pshader 所写

标签: opengl glsl processing


【解决方案1】:

您对纹理进行采样

(u,v)

如果要翻转 Y 轴,只需在 处采样

(u, 1.0f -v)

所以您更新后的main 将如下所示:

void main() {
  gl_Position = transform * vertex;

  vertColor = color;
  newTCoord = texCoord;
  newTCoord.y = 1.0 - newTCoord.y;
  vertTexCoord = vec4(newTCoord, 1.0, 1.0);
}

【讨论】:

  • 我的 vertTexCoord 已经在 texMatrix 的帮助下颠倒过来了(可以使用 Processing)。问题是如何在swap.glsl中的透视变换之后或之前将其应用于gl_FragColor
  • gl_Position 是 OpenGL 用来确定要插值和绘制的像素的方法。像您的 vertexTexCoord 这样的每个顶点属性也会被插值。我不知道您为什么要对纹理坐标应用透视变换。
  • 我想还不清楚。 Swap.glsl 像这样进行照片转换:transitions.glsl.io/transition/2a3f2e907e1c0a152e60 问题是我的照片是颠倒的。而且我不能修改 gl_FragColor,因为它被那些转换使用。我需要一种在转换修改之前翻转纹理的方法。
猜你喜欢
  • 2011-08-26
  • 1970-01-01
  • 1970-01-01
  • 2014-06-23
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-05-09
  • 1970-01-01
相关资源
最近更新 更多