Verlet 3D Ball Breaking

Я пытаюсь использовать интеграцию верлетов для создания сферы мягкого тела.

Я в основном создаю сетку и прикрепляю каждую вершину пружинами. Я использую three.js для рендеринга сферы. Я считаю, что все работает правильно, но я не уверен, как получить желаемые результаты. Мой шар разрушается, если соединения слишком слабые, или взрывается, если слишком жесткие. Что-то не так. Проблема возникает в тот момент, когда объект падает и падает на землю. Я опубликую основные части своего кода, однако я не уверен, что мой подход верен и может быть использован для получения желаемых результатов.

function verletStick3D(p1, p2, distance, rigid){


    this.type = "stick";
    this.p1 = p1;
    this.p2 = p2; 
    this.distance = distance;
    this.rigid = rigid;
    this.update = function(){

        var dx = this.p1.x - this.p2.x;
        var dy = this.p1.y - this.p2.y;
        var dz = this.p1.z - this.p2.z;
        var distance = Math.sqrt(dx * dx + dy * dy + dz * dz);
        var difference = this.distance - distance;
        var percent = difference / distance / 2;
        var offsetX = dx * percent * this.rigid;
        var offsetY = dy * percent * this.rigid;
        var offsetZ = dz * percent * this.rigid;
        this.p1.x += offsetX;
        this.p1.y += offsetY;
        this.p1.z += offsetZ;
        this.p2.x -= offsetX;
        this.p2.y -= offsetY;
        this.p2.z -= offsetZ;

    }
}

function verletPoint3D(x, y,z, maxX, maxZ, bounce = 0.9, friction = 

1.0, gravity = {x:0, y:-0.03, z:0}){
    y = y + 10;
    this.type = "point";
    this.x=x; 
    this.y=y;
    this.z=z;
    this.xOld=x;
    this.yOld=y;
    this.zOld=z;
    this.bounce = bounce;
    this.friction = friction;
    this.gravity = gravity;
    this.update = function(){
        var vx = (this.x - this.xOld) * this.friction;
        var vy = (this.y - this.yOld) * this.friction;
        var vz = (this.z - this.zOld) * this.friction;
        this.xOld = this.x;
        this.yOld = this.y;
        this.zOld = this.z;
        this.x += vx;
        this.y += vy;
        this.z += vz;
        this.y += this.gravity.y;
        this.x += this.gravity.x;
        this.z += this.gravity.z;



    }
    this.setGravity = function(gravity){
        this.gravity = gravity;
    }
    this.constrain = function(){
        var vx = (this.x - this.xOld) ;
        var vy = (this.y - this.yOld) ;

        if(this.x  > this.maxX){
            this.x = width;
            this.xOld = this.x + vx * this.bounce;
        }
        if(this.x < this.minX){
            this.x ;
            this.xOld = this.x + vx * this.bounce;
        }
        if(this.z > this.maxZ ){
            this.z = height ;
            this.zOld = this.z + vz * this.bounce;

        }
        if(this.z  < this.minZ){
            this.z = 0;
            this.zOld = this.z + vz * this.bounce;
        }
        if(this.y  < 0){
            this.y = 0;
            this.yOld = this.y + vy * this.bounce;
        }

    }
}

function verletFromMesh(vertices, faces, maxX, maxY){
    var points  = [];
    for(var i = 0; i < vertices.length; i++){

        var p = new verletPoint3D(vertices[i].x, vertices[i].y, vertices[i].z, maxX, maxY, 0.9, 0.9);
        points.push(p);

    }

    var springs = []
    /*
    for(var i = 0; i < faces.length; i++){
        var face = faces[i];
        var p1 = face.a;
        var p2 = face.b;
        var p3 = face.c;
        springs.push(new verletStick( p1,p2, distance3D(p1, p2), 1.0) )
        springs.push(new verletStick( p2,p3, distance3D(p2, p3), 1.0) )
        springs.push(new verletStick( p3,p1, distance3D(p3, p1), 1.0) )
    }*/
    for(var i = 0; i < points.length; i++){
        var p1 = points[i];
        for(var j= i+1; j < points.length; j++){
            var p2 = points[j];
            if(p1.x == p2.x && p1.y == p2.y && p1.z == p2.z)
                continue;
            var dist = distance3D(p1, p2)

            springs.push(new verletStick3D( p1,p2, dist, 1.0) )


        }


    }


    return new verletObj(points, springs);
}

1 ответ

Хм, трудно понять, где именно вы идете не так, но мне кажется, что ваши пружины не смогут изменить движение. Это может привести к взрывам / разрушениям в зависимости от их силы. Я настоятельно рекомендую попытаться явно ограничить их влияние на сферу, и, надеюсь, это поможет вам в дальнейшей отладке этой проблемы.

Как работает ваша математика, особенно при настройке смещений dx * percent * this.rigid; Создается впечатление, что эти динамические силы ведут себя статически, и поэтому заходят слишком далеко в игру.

Ps: вы проверили этот пост? Речь идет о трении, но я думаю, что ответ здесь довольно цитируемый:

Просто введите небольшое постоянное ускорение на движущиеся объекты, которые указывают в направлении, противоположном движению. И убедитесь, что он не может полностью изменить движение; если вы обнаружите это на этапе интеграции, просто установите скорость на ноль.

Надеюсь, это как-то направит вас в правильном направлении.

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