Проблемы WebGL texImage2D: Как получить текстуру на объект / объект?

Поэтому я ищу некоторую помощь по WebGL текстурам /3D рендерингу.

Смысл моего кода (что я уже сделал с этим) заключается в создании 3D-лабиринта из текстурированных блоков / кубов, по которому пользователь может перемещаться с помощью клавиш со стрелками или WASD. Проблема, которую я имею, только с texImage2d(), я верю.

Я пробовал такие вещи, как

 var stonePic = document.getElementById("stoneTex");
 gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, stonePic));

который не работал и привел к этому.

Я тоже пробовал

var stonePic = new Image();
stonePic.src = "stoneWall2.png";

что привело к той же ошибке.

Что я делаю неправильно? Могу ли я это исправить?

<HTML>
    <HEAD>
        <TITLE>Brick Wall Maze</TITLE>
    <SCRIPT>

    //VERTEX SHADER TEXT
    var vertexShaderText = 
    [
        'precision mediump float;',
        '',
        'attribute vec3 vertPosition;',
        'attribute vec2 vertTexCoord;',
        'varying vec2 fragTexCoord;',
        'uniform vec3 theta;',
        'uniform vec3 trans;',
        'uniform float thetaC;',
        'uniform vec3 camLoc;',
        'void main()',
        '{',
        'fragTexCoord = vertTexCoord;',
        'vec3 c = cos(theta);',
        'vec3 s = sin(theta);',
        '',
        'mat4 ry = mat4(c.y,0.0,-1.0*s.y,0.0, 0.0,1.0,0.0,0.0, s.y,0.0,c.y,0.0, 0.0,0.0,0.0,1.0);',
        'mat4 translate = mat4(1,0,0,0, 0,1,0,0, 0,0,1,0, trans.x,trans.y,trans.z,1);',
        'vec4 tempLoc = vec4(vertPosition, 1.0);',

        'float l = -1.0;',
        'float r = 1.0;',
        'float t = 1.0;',
        'float b = -1.0;',
        'float f = 100.0;',
        'float n = 1.0;',
        'mat4 perspective  = mat4(2.0*n/(r-l),0,0,0,  0,2.0*n/(t-b),0,0, (r+l)/(r-l),(t+b)/(t-b),-1.0*(f+n)/(f-n),-1.0,   0,0,-2.0*f*n/(f-n),0);',

        'float tempc = cos(thetaC);',
        'float temps = sin(thetaC);',
        'mat4 camRY = mat4(tempc,0,-1.0*temps,0, 0,1,0,0, temps,0,tempc,0, 0,0,0,1);',
        'mat4 viewM = mat4(1.0,0,0,0, 0,1.0,0,0, 0,0,1.0,0, camLoc.x,camLoc.y,camLoc.z,1.0);',
        'gl_Position = perspective * camRY * viewM * translate * ry * tempLoc;',
        '}'
    ].join("\n");

    // FRAGMENT SHADER TEXT
    var fragmentShaderText = 
    [
        'precision mediump float;',
        'varying vec2 fragTexCoord;',
        'uniform sampler2D sampler;',
        'void main()',
        '{',
        'gl_FragColor = texture2D(sampler, fragTexCoord);',
        '}'
    ].join('\n');

    // ~ ~ ~ ~ ~ ~ ~ ~ ~ ~

    function initializeGL()
    {   
        var canvas = document.getElementById("screen"); 
        var gl =  canvas.getContext("webgl") || canvas.getContext("experimental-webgl");
        if (!gl) { alert("WEBGL IS NOT AVAILABLE"); }

        gl.viewport(0, 0, canvas.width, canvas.height);
        gl.clearColor(0.5, 0.7, 0.6 ,1.0);
        gl.clear(gl.COLOR_BUFFER_BIT|gl.DEPTH_BUFFER_BIT);
        gl.enable(gl.DEPTH_TEST);

        return gl;
    }

    function initializeShaders(gl)
    {
        // VERTEX SHADER
        var vertexShader = gl.createShader(gl.VERTEX_SHADER);
        gl.shaderSource(vertexShader, vertexShaderText);
        gl.compileShader(vertexShader);
        if (!gl.getShaderParameter(vertexShader, gl.COMPILE_STATUS)) {
            console.log("ERROR: ",gl.getShaderInfoLog(vertexShader));
        }

        // FRAGMENT SHADER
        var fragmentShader = gl.createShader(gl.FRAGMENT_SHADER);
        gl.shaderSource(fragmentShader,fragmentShaderText);
        gl.compileShader(fragmentShader);
        if (!gl.getShaderParameter(fragmentShader, gl.COMPILE_STATUS)) {
            console.log("ERROR: ",gl.getShaderInfoLog(fragmentShader));
        }

        // PROGRAM
        var program = gl.createProgram();
        gl.attachShader(program, vertexShader);
        gl.attachShader(program, fragmentShader);

        gl.linkProgram(program);
        if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {
            console.error("ERROR", gl.getShaderInfoLog(program));
        }

        gl.validateProgram(program);
        if (!gl.getProgramParameter(program, gl.VALIDATE_STATUS)) {
            console.error("ERROR", gl.getShaderInfoLog(program));
        }

        return program;
    }

    // var brickTexture;
    var stoneTexture;

    function setupIndBuffers(gl, program, buffer)
    {
        gl.bindBuffer(gl.ARRAY_BUFFER, buffer);     
        positionAttributeLocation = gl.getAttribLocation(program, "vertPosition");
        texCoordAttributeLocation = gl.getAttribLocation(program, "vertTexCoord");
        gl.vertexAttribPointer(
        positionAttributeLocation, //ATTRIBUTE LOCATION
        3, //NUMBER of elements per attribute
        gl.FLOAT, //TYPES OF ELEMENTS
        gl.FALSE,
        5*Float32Array.BYTES_PER_ELEMENT, //SIZE OF AN INDIVIDUAL VERTEX
        0 //OFFSET
        );


        gl.vertexAttribPointer(
        texCoordAttributeLocation, //ATTRIBUTE LOCATION
        2, //NUMBER of elements per attribute
        gl.FLOAT, //TYPES OF ELEMENTS
        gl.FALSE,
        5*Float32Array.BYTES_PER_ELEMENT, //SIZE OF AN INDIVIDUAL VERTEX
        3*Float32Array.BYTES_PER_ELEMENT //OFFSET
        );

        gl.enableVertexAttribArray(positionAttributeLocation);
        gl.enableVertexAttribArray(texCoordAttributeLocation);
    }

    function setupVertices(gl, program)
    {
        /*
        // Brick Texture
        brickTexture = gl.createTexture();
        gl.bindTexture(gl.TEXTURE_2D, brickTexture);

        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.MIRRORED_REPEAT);
        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.MIRRORED_REPEAT);

        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);

        var brickArray = [];

        for (i = 0; i < 16; i++) {
            for (j = 0; j < 16; j++) {
                if (i == 0 || j == 0) {
                    // Push Black
                    brickArray.push(0, 0, 255);
                }
                else {
                    // Push Red
                    brickArray.push(220, 30, 30, 255);
                }
            }
        }

        gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, 16, 16, 0, gl.RGBA, gl.UNSIGNED_BYTE, new Uint8Array(brickArray));
        gl.bindTexture(gl.TEXTURE_2D, null);
        */

        // ~ ~ ~ ~ ~
        // STONE TEXTURE
        stoneTexture = gl.createTexture();
        gl.bindTexture(gl.TEXTURE_2D, stoneTexture);

        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);

        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);

        gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, stonePic);

        gl.bindTexture(gl.TEXTURE_2D, null);
    }

    /*
    brickVertices = 
    [
        // X, Y, Z     U, V,               

        // Top
        -1.0, 1.0, -1.0,   0.0, 0.0,
        -1.0, 1.0, 1.0,    0.0, 10,
        1.0, 1.0, 1.0,     10, 10,
        1.0, 1.0, -1.0,    10, 0.0,

        // Left
        -1.0, 1.0, 1.0,    10,10,
        -1.0, -1.0, 1.0,   0,10,
        -1.0, -1.0, -1.0,  0,0,
        -1.0, 1.0, -1.0,   10,0,

        // Right
        1.0, 1.0, 1.0,    10,10,
        1.0, -1.0, 1.0,   0,10,
        1.0, -1.0, -1.0,  0,0,
        1.0, 1.0, -1.0,   10,0,

        // Front
        1.0, 1.0, 1.0,     10,10,
        1.0, -1.0, 1.0,    10,0,
        -1.0, -1.0, 1.0,   0,0,
        -1.0, 1.0, 1.0,    0,10,

        // Back
        1.0, 1.0, -1.0,     10,10,
        1.0, -1.0, -1.0,    10,0,
        -1.0, -1.0, -1.0,   0,0,
        -1.0, 1.0, -1.0,    0,10,

        // Bottom
        -1.0, -1.0, -1.0,   0,0,
        -1.0, -1.0, 1.0,    0,10,
        1.0, -1.0, 1.0,     10,10,
        1.0, -1.0, -1.0,    10,0,
    ];
    */

    stoneVertices =
    [
        // X, Y, Z     U, V,

        // Top
        -1.0, 1.0, -1.0,   0.0, 0.0,
        -1.0, 1.0, 1.0,    0.0, 1.0,
        1.0, 1.0, 1.0,     1.0, 1.0,
        1.0, 1.0, -1.0,    1.0, 0.0,

        // Left
        -1.0, 1.0, 1.0,    1,1,
        -1.0, -1.0, 1.0,   0,1,
        -1.0, -1.0, -1.0,  0,0,
        -1.0, 1.0, -1.0,   1,0,

        // Right
        1.0, 1.0, 1.0,    1,1,
        1.0, -1.0, 1.0,   0,1,
        1.0, -1.0, -1.0,  0,0,
        1.0, 1.0, -1.0,   1,0,

        // Front
        1.0, 1.0, 1.0,     1,1,
        1.0, -1.0, 1.0,    1,0,
        -1.0, -1.0, 1.0,   0,0,
        -1.0, 1.0, 1.0,    0,1,

        // Back
        1.0, 1.0, -1.0,     1,1,
        1.0, -1.0, -1.0,    1,0,
        -1.0, -1.0, -1.0,   0,0,
        -1.0, 1.0, -1.0,    0,1,

        // Bottom
        -1.0, -1.0, -1.0,   0,0,
        -1.0, -1.0, 1.0,    0,1,
        1.0, -1.0, 1.0,     1,1,
        1.0, -1.0, -1.0,    1,0,
        ];

    /*
    stoneVertices =
    [
        // X, Y, Z     U, V,               

        // Top
        -1.0, 1.0, -1.0,   0.0, 0.0,
        -1.0, 1.0, 1.0,    0.0, 1.0,
        1.0, 1.0, 1.0,     1.0, 1.0,
        1.0, 1.0, -1.0,    1.0, 0.0,

        // Left
        -1.0, 1.0, 1.0,    1,1,
        -1.0, -1.0, 1.0,   0,1,
        -1.0, -1.0, -1.0,  0,0,
        -1.0, 1.0, -1.0,   1,0,

        // Right
        1.0, 1.0, 1.0,    1,1,
        1.0, -1.0, 1.0,   0,1,
        1.0, -1.0, -1.0,  0,0,
        1.0, 1.0, -1.0,   1,0,

        // Front
        1.0, 1.0, 1.0,     1,1,
        1.0, -1.0, 1.0,    1,0,
        -1.0, -1.0, 1.0,   0,0,
        -1.0, 1.0, 1.0,    0,1,

        // Back
        1.0, 1.0, -1.0,     1,1,
        1.0, -1.0, -1.0,    1,0,
        -1.0, -1.0, -1.0,   0,0,
        -1.0, 1.0, -1.0,    0,1,

        // Bottom
        -1.0, -1.0, -1.0,   0,0,
        -1.0, -1.0, 1.0,    0,1,
        1.0, -1.0, 1.0,     1,1,
        1.0, -1.0, -1.0,    1,0,
        ];
        */

        class Cube
        {
            constructor(test)
            {
                this.tranLoc = gl.getUniformLocation(program, "trans");
                this.thetaLoc = gl.getUniformLocation(program, "theta");
                this.loc = [0, 0, 0];

                if(test) {
                    this.verts = stoneBuffer;
                    this.texture = stoneTexture;
                }
                /*else {
                    this.verts = stoneBuffer;
                    this.texture = stoneTexture;
                }*/

                this.boxIndices =
                [
                    // Top
                    0, 1, 2,
                    0, 2, 3,
                    // Left
                    5, 4, 6,
                    6, 4, 7,
                    // Right
                    8, 9, 10,
                    8, 10, 11,
                    // Front
                    13, 12, 14,
                    15, 14, 12,
                    // Back
                    16, 17, 18,
                    16, 18, 19,
                    // Bottom
                    21, 20, 22,
                    22, 20, 23
                ];

                this.iBuffer = gl.createBuffer();

                gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.iBuffer);
                gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint8Array(this.boxIndices), gl.STATIC_DRAW);
                gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);   
            }

            render()
            {
                if (this.texture == stoneTexture) {
                    //gl.bindBuffer(gl.ARRAY_BUFFER,brickBuffer);
                    setupIndBuffers(gl, program, stoneBuffer);
                }
                /*else {
                    //gl.bindBuffer(gl.ARRAY_BUFFER,this.crateBuffer);
                    setupIndBuffers(gl, program, brickBuffer);
                }*/

                gl.bindTexture(gl.TEXTURE_2D, this.texture);
                gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.iBuffer);
                var thetaV = [0, 0, 0];
                gl.uniform3fv(this.tranLoc, new Float32Array(this.loc));
                gl.uniform3fv(this.thetaLoc, new Float32Array(thetaV));
                gl.drawElements(gl.TRIANGLES, this.boxIndices.length, gl.UNSIGNED_BYTE, 0);
                gl.bindTexture(gl.TEXTURE_2D, null);    
                gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, null);
                gl.bindBuffer(gl.ARRAY_BUFFER, null);
            }
        }

    </SCRIPT>
    </HEAD>

    <BODY>
        <CANVAS ID = "screen" WIDTH = "800" HEIGHT = "600" ALT = "Your browser does not support canvas."/>
        <IMG ID = "stoneTex" SRC = "stoneWall2.png" WIDTH = "50" HEIGHT = "50" ALT = "tex"/>

    <SCRIPT>
    //Init GL System
    var gl = initializeGL();
    var program = initializeShaders(gl);
    setupVertices(gl, program);
    gl.useProgram(program);
    //var stonePic = document.getElementById("stoneTex");

    // SETUP BRICK BUFFER
    /*var brickBuffer = gl.createBuffer();
    setupIndBuffers(gl, program, brickBuffer);
    gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(brickVertices), gl.STATIC_DRAW);
    gl.bindBuffer(gl.ARRAY_BUFFER, null);
    */

    // SETUP STONE BUFFER
    var stoneBuffer = gl.createBuffer();
    setupIndBuffers(gl, program, stoneBuffer);
    gl.bufferData(gl.ARRAY_BUFFER, new Float32Array(stoneVertices), gl.STATIC_DRAW);
    gl.bindBuffer(gl.ARRAY_BUFFER, null);

    // Initialize and render actual objects.
    var x = new Cube(true);
    x.loc = [-5, 0, -10];
    //var y = new Cube(false);
    //y.loc = [5, 0, -10];

    var loop = function()
    {
        gl.clear(gl.COLOR_BUFFER_BIT|gl.DEPTH_BUFFER_BIT);
        x.render();
        requestAnimationFrame(loop);
    }

    requestAnimationFrame(loop);
    </SCRIPT>
    </BODY>
    </HTML>

1 ответ

То, что вы делаете неправильно, не ждет загрузки изображений. Изображения в веб-браузерах загружаются асинхронно. Это означает, что вы должны ждать их загрузки, прежде чем звонить gl.texImage2D

Обычно я бы сделал что-то подобное

// Create a texture.
var texture = gl.createTexture();
gl.bindTexture(gl.TEXTURE_2D, texture);
 
// Fill the texture with a 1x1 blue pixel.
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, 1, 1, 0, gl.RGBA, gl.UNSIGNED_BYTE,
              new Uint8Array([0, 0, 255, 255]));
 
// Asynchronously load an image
var image = new Image();
image.src = "resources/f-texture.png";
image.addEventListener('load', function() {
  // Now that the image has loaded make copy it to the texture.
  gl.bindTexture(gl.TEXTURE_2D, texture);
  gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA,gl.UNSIGNED_BYTE, image);
  // if it's a power of 2 in both dimensions then
  // we can generate mips, otherwise we'd have to do other things
  gl.generateMipmap(gl.TEXTURE_2D);
});

texture теперь текстура, и вы можете начать рендеринг с ней немедленно. После завершения загрузки изображения его содержимое будет скопировано в текстуру. В этот момент вам нужно либо снова выполнить рендеринг, либо, если, как и в большинстве приложений WebGL (и в вопросе, о котором идет речь), вы выполняете рендеринг непрерывно, он будет отображаться автоматически при следующем рендеринге вашего приложения

Смотрите эти статьи

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