Значение vTextCoord не работает - c # OPENTK - PullRequest
0 голосов
/ 29 января 2019

Мое приложение отображает изображение в полноэкранном режиме с помощью кода шейдера OpenGL, как показано ниже.Вершинный шейдер, который я здесь использовал, скопирован откуда-то.Может кто-нибудь объяснить, почему здесь используется vTexCoord = (a_position.xy+1)/2;?Когда я пытаюсь с vTexCoord = a_position.xy, мой вывод OpenGL делится на четыре прямоугольника, и только его верхняя правая часть показывает изображение.Другие три стороны кажутся размытыми.Какие изменения я должен сделать, чтобы работать с vTexCoord = a_position.xy?

Некоторые важные функции, используемые в проекте, показаны ниже.Пожалуйста, проверьте и помогите исправить.

float[] vertices = {
            // Left bottom triangle
            -1f, -1f, 0f,
            1f, -1f, 0f,
            1f, 1f, 0f,
            // Right top triangle
            1f, 1f, 0f,
           -1f, 1f, 0f,
           -1f, -1f, 0f
    };
    private void CreateShaders()
    {
        /***********Vert Shader********************/
        vertShader = GL.CreateShader(ShaderType.VertexShader);
        GL.ShaderSource(vertShader, @"attribute vec3 a_position;
                                    varying vec2 vTexCoord;
                                    void main() {
                                    vTexCoord = (a_position.xy+1)/2;
                                    gl_Position = vec4(a_position, 1);
                                    }");
        GL.CompileShader(vertShader);

        /***********Frag Shader ****************/
        fragShader = GL.CreateShader(ShaderType.FragmentShader);
        GL.ShaderSource(fragShader, @"precision highp float;
    uniform sampler2D sTexture;varying vec2 vTexCoord;                      
          void main ()
         {
        vec4 color= texture2D (sTexture, vTexCoord);                           
        gl_FragColor =color;
                             }");
        GL.CompileShader(fragShader);
    }
    private void InitBuffers()
    {
        buffer = GL.GenBuffer();
        positionLocation = GL.GetAttribLocation(program, "a_position");
        positionLocation1 = GL.GetUniformLocation(program, "sTexture");
        GL.EnableVertexAttribArray(positionLocation);
        GL.BindBuffer(BufferTarget.ArrayBuffer, buffer);
        GL.BufferData(BufferTarget.ArrayBuffer, (IntPtr)(vertices.Length * sizeof(float)), vertices, BufferUsageHint.StaticDraw);
        GL.VertexAttribPointer(positionLocation, 3, VertexAttribPointerType.Float, false, 0, 0);
    }
 public void DrawImage(int image)
    {
        GL.Viewport(new Rectangle(0, 0, ScreenWidth, ScreenHeight));

        GL.MatrixMode(MatrixMode.Projection);
        GL.PushMatrix();
        GL.LoadIdentity();

        //GL.Ortho(0, 1920, 0, 1080, 0, 1);

        GL.MatrixMode(MatrixMode.Modelview);
        GL.PushMatrix();
        GL.LoadIdentity();
        GL.Disable(EnableCap.Lighting);
        GL.Enable(EnableCap.Texture2D);
        GL.ActiveTexture(TextureUnit.Texture0);
        GL.BindTexture(TextureTarget.Texture2D, image);
        GL.Uniform1(positionLocation1, 0);
        GL.Begin(PrimitiveType.Quads);
        GL.TexCoord2(0, 1);
        GL.Vertex3(0, 0, 0);
        GL.TexCoord2(0, 0);
        GL.Vertex3(1920, 0, 0);
        GL.TexCoord2(1, 1);
        GL.Vertex3(1920, 1080, 0);
        GL.TexCoord2(1, 0);
        GL.Vertex3(0, 1080, 0);
        GL.End();


        RunShaders();
        GL.Disable(EnableCap.Texture2D);
        GL.PopMatrix();
        GL.MatrixMode(MatrixMode.Projection);
        GL.PopMatrix();
        GL.MatrixMode(MatrixMode.Modelview);
        glControl1.SwapBuffers();
    }
    private void RunShaders()
    {
        GL.UseProgram(program);
        GL.DrawArrays(PrimitiveType.Triangles, 0, vertices.Length / 3);
    }

1 Ответ

0 голосов
/ 29 января 2019

Может кто-нибудь объяснить, почему здесь используется vTexCoord = (a_position.xy+1)/2;?

Координаты вершины в вашем примере находятся в диапазоне [-1, 1] для x и компонента.Это соответствует нормализованному пространству устройства.Нормализованное пространство устройства - это куб от левого нижнего переднего (-1, -1, -1) до правого верхнего правого заднего (1, 1, 1), это область, которая является «видимой».Эта область отображается в окне просмотра.
Это приводит к тому, что координаты вершины, в вашем примере, образуют прямоугольник, который охватывает весь видовой экран.

Если координаты текстуры должны обернуть всю текстуру в квадтогда координаты текстуры (u, v) должны находиться в диапазоне [0, 1].(0, 0) - левый нижний угол текстуры, а (1, 1) - верхний правый.
См. Также Как работают координаты текстуры OpenGL?

Итак, x иy компонент a_position должен отображаться из диапазона [-1, 1] в диапазон [0, 1], чтобы использоваться в качестве ультрафиолетовых координат для поиска текстуры:

u = (a_position.x + 1) / 2
v = (a_position.y + 1) / 2

Какое изменение я должен сделать, чтобы работать с vTexCoord = a_position.xy?

Это невозможно, но вы можете сгенерировать отдельный атрибут координат текстуры, который является общим:

attribute vec3 a_position;
attribute vec2 a_texture;

varying vec2 vTexCoord;

void main() {
    vTexCoord   = a_texture;
    gl_Position = vec4(a_position, 1);
}
float[] vertices = {
  // x    y   z      u   v
    // Left bottom triangle
    -1f, -1f, 0f,    0f, 0f
     1f, -1f, 0f,    1f, 0f
     1f,  1f, 0f,    1f, 1f
    // Right top triangle
     1f,  1f, 0f,    1f, 1f
    -1f,  1f, 0f,    0f, 1f
    -1f, -1f, 0f     0f, 0f
};
buffer = GL.GenBuffer();
GL.BindBuffer(BufferTarget.ArrayBuffer, buffer);
GL.BufferData(BufferTarget.ArrayBuffer, (IntPtr)(vertices.Length * sizeof(float)), vertices, BufferUsageHint.StaticDraw);

positionLocation = GL.GetAttribLocation(program, "a_position");
tetureLocation   = GL.GetAttribLocation(program, "a_texture");

GL.EnableVertexAttribArray(positionLocation);
GL.EnableVertexAttribArray(tetureLocation);

int stride   = sizeof(float) * 5; // 5 because of (x, y, z, u, v)
int offsetUV = sizeof(float) * 3; // 3 because the u and v coordinates are the 4th and 5th coordinate

GL.VertexAttribPointer(positionLocation, 3, VertexAttribPointerType.Float, false, stride, 0);
GL.VertexAttribPointer(tetureLocation,   2, VertexAttribPointerType.Float, false, stride, (IntPtr)(offsetUV));
...