Update geometry with image [Solved]

Hi!
I have been working on changing an image and that it becomes particles and it works but when I change the image, it does not update and also starts to slow down the browser.

Neither with verticesNeedUpdate works

The part important is the Drawmap function

(In the html file i call changeState to update the image)

The code is this

var renderer, scene, camera, ww, wh, particles, mw, mh, mz, numState;

numState = 0;

mz = 6; // Matrerial size

ww = document.getElementById('map-container').offsetWidth,

wh = 450;

mw = ww * 2;

mh = wh * 2;

renderer = new THREE.WebGLRenderer({

    canvas: document.getElementById("map"),

    antialias: false

});

camera = new THREE.OrthographicCamera( ww / - 2, ww / 2, wh / 2, wh / - 2, 1, 1000 );

scene = new THREE.Scene();

particles = new THREE.Points();

var centerVector = new THREE.Vector3(0, 0, 0);

var previousTime = 0;

    speed = 10;

    isMouseDown = false;

var getImageData = function(image) {

    var canvas = document.createElement("canvas");

    canvas.width = image.width;

    canvas.height = image.height;

    var ctx = canvas.getContext("2d");

    ctx.drawImage(image, 0, 0);

    return ctx.getImageData(0, 0, image.width, image.height);

}

var material = new THREE.PointsMaterial({

    size: 3,

    color: 0xFFFFFF,

    sizeAttenuation: false

});

particles.material = material

scene.add(camera);

scene.add(particles);

var drawTheMap = function() {

    let vertices = particles.geometry; // this acts as a REFERENCE!

    vertices.length = 0; // clears the vertices array

    for (var y = 0, y2 = imagedata.height; y < y2; y += 2) {

        for (var x = 0, x2 = imagedata.width; x < x2; x += 2) {

            if (imagedata.data[(x * 4 + y * 4 * imagedata.width)] < 128) {

                var vertex = new THREE.Vector3();

                vertex.x = x - imagedata.width / 2;

                vertex.y = -y + imagedata.height / 2;

                vertex.z = -Math.random()*500;

                vertex.speed = Math.random() / speed + 0.015;

                vertices.vertices.push(vertex);

                

                vertex = null

            }

        }

    }

    

    particles.geometry.verticesNeedUpdate = true;

    requestAnimationFrame(render);

};

var init = function() {

    renderer.setSize(ww, wh);

    renderer.setClearColor(0x12347C);

    

    camera.position.set(7, 0, 4);

    camera.lookAt(centerVector);

    

    camera.zoom = 4;

    camera.updateProjectionMatrix();

    imagedata = getImageData(image);

    drawTheMap();

    

    onResize();

    window.addEventListener('mousemove', onMousemove, false);

    window.addEventListener('mousedown', onMousedown, false);

    window.addEventListener('mouseup', onMouseup, false);

    window.addEventListener('resize', onResize, false);

};

var onResize = function(){

    var mov1, mov2;

    ww = document.getElementById('map-container').offsetWidth;

    wh = 450;

    

    if (window.innerWidth > 850) {

        mw = ww * 2;

        mh = wh * 2;

        mz = 6;

        mov1 = 2.2;

        mov2 = 1.9;

        particles.material.size = mz;

    } else {

        mw = ww;

        mh = wh;

        mz = 3;

        mov1 = 2;

        mov2 = 2;

        particles.material.size = mz;

    }

    renderer.setSize(mw, mh);

    camera.left    = ww / - mov1;

    camera.right   = ww / 2;

    camera.top     = wh / mov2;

    camera.bottom  = wh / - 2;

    camera.updateProjectionMatrix();

};

var onMouseup = function(){

    isMouseDown = false;

}

var onMousedown = function(e){

    isMouseDown = true;

    lastMousePos = {x:e.clientX, y:e.clientY};

};

var onMousemove = function(e){

    if(isMouseDown){

        camera.position.x += (e.clientX-lastMousePos.x)/100;

        camera.position.y -= (e.clientY-lastMousePos.y)/100;

        camera.lookAt(centerVector);

        lastMousePos = {x:e.clientX, y:e.clientY};

    }

};

var render = function(a) {

    requestAnimationFrame(render);

    particles.geometry.verticesNeedUpdate = true;

    if(!isMouseDown){

        camera.position.x += (0-camera.position.x)*0.06;

        camera.position.y += (0-camera.position.y)*0.06;

        camera.lookAt(centerVector);

    }

    renderer.render(scene, camera);

};

var imgData;

var image;

imgData ="";

const changeState = function(state, num) {

    document.getElementById('dropbox-choose').innerHTML = state;

    numState = num;

    switch (numState)

    {

        case 0:

            imgData ="";

        break;

        case 1:

            imgData = ""

        break;

    }

    image.src = imgData;

}

var image = document.createElement("img");

image.onload = init;

image.src = imgData;

Thank for your support.

A typical cause of performance issue is the fact to update an instance of Geometry.

It is more performant to use BufferGeometry instead since all instances of Geometry have to be converted to BufferGeometry before rendering.

So I suggest you use BufferGeometry right from the beginning. Keep in mind that almost all official three.js example use BufferGeometry so it should be easy to find out how to work with this class. The official documentation is also a good starting point:

https://threejs.org/docs/index.html#api/en/core/BufferGeometry

1 Like

Thank you very much, I have changed to BufferGeometry and it works perfectly.

1 Like