В предыдущих двух постах (раз, два) я реализовал 2д-коллайдер, который применяет правило "угол падения равен углу отражения", которое визуально достаточно неплохо работает (особенно когда много частиц), но все же слишком упрощенно показывает отражения после столкновений. Например, частицы всегда имеют одну и ту же скорость и меняется только направление, а хотелось бы, чтобы частицы все же обменивались импульсами более реалистично. Поэтому я решил усложнить задачу и сделать коллайдер, который реализует более сложный закон, а именно закон абсолютно упругого столкновения и закон сохранения импульса. Все формулы и пояснения можно найти здесь. Вот ключевые моменты:
Собственно, это вся теория, которая нам нужна. Кто-то может заметить, что все эти иллюстрации и формулы работают для одномерного пространства, но нам же нужен двумерный случай! В википедии специально для этого указано следующее:
В случае столкновения двух тел в трёхмерном пространстве векторы импульсов тел до и после столкновения лежат в одной плоскости. Вектор скорости каждого тела может быть разложен на две компоненты: одна по общей нормали поверхности сталкивающихся тел в точке контакта, а другая параллельная поверхности столкновения. Поскольку сила удара действует только по линии столкновения, компоненты скорости, векторы которых проходят по касательной к точке столкновения, не изменяются. Скорости, направленные вдоль линии столкновения, могут быть вычислены с помощью тех же уравнений, что и столкновения в одном измерении. Окончательные скорости могут быть вычислены из двух новых компонентов скоростей и будут зависеть от точки столкновения.
То есть все, что нам в итоге надо сделать, это взять ось столкновения, которая в случае двух кругов всегда проходит через их центры и спроецировать вектор движения каждого круга на эту ось - это у нас будет изменяемая компонента, которая меняется после удара в соотв. с вышеуказанными формулами. Вторая компонтента - это проекция вектора движения каждого из кругов на тангециальную составляющую, которая вычисляется путем поворота оси столкновения на 90 градусов. Вторая компонтена у нас не меняется после удара. После того, как мы вычислили изменяемую компоненту, мы умножили этот скаляр на единичный вектор вдоль оси столкновения и сложили его с тангенциальной составляющей, получив результирующий вектор движения после удара кругов. Вот картинка, которая это поясняет:
В коде это выглядит вот так:
- void pulseColl(const Ball& b2)
- {
- // direction to other ball
- Point2f toB2 = b2.pos - pos;
- Point2f toB2Unit = toB2.unitVector();
- // calculate dot product
- float dotProdB = f.dotProduct(toB2Unit);
- // calculate speed v for both balls
- // that is projection to hit axis
- float v1 = dotProdB;
- float dotProdB2 = b2.f.dotProduct(toB2Unit);
- float v2 = dotProdB2;
- // if move projection to hit axis
- // for both balls are directed to move away
- // one from other then do nothing
- if (v1 <= 0 && v2 >= 0)
- {
- return;
- }
- // mass is equal to square of 2d ball
- float m1 = r * r;
- float m2 = b2.r * b2.r;
- // speed of this ball after collision
- float v1new = (2 * m2 * v2 + v1 * (m1 - m2)) / (m1 + m2);
- // "to" move component
- Point2f tmpTo = toB2Unit * v1new;
- // "tangent" move component
- Point2f tanUnit = Mat2x2f().rot(k_PI / 2) * toB2Unit;
- float dotProdTan1 = f.dotProduct(tanUnit);
- Point2f tmpTan = tanUnit * dotProdTan1;
- // new move vector
- Point2f newF = tmpTo + tmpTan;
- f = newF;
- }
Код ядра выглядит так:
- Ball pulseColl(Ball b1, const Ball b2)
- {
- // direction to other ball
- float2 p1 = (float2)(b1.pos.x, b1.pos.y);
- float2 p2 = (float2)(b2.pos.x, b2.pos.y);
- float2 toB2 = p2 - p1;
- float2 toB2Unit = getUnitVector(toB2);
- // calculate dot product
- float2 f1 = (float2)(b1.f.x, b1.f.y);
- float dotProdB1 = getDotProduct(f1, toB2Unit);
- // calculate speed v for both balls
- // that is projection to hit axis
- float v1 = dotProdB1;
- float2 f2 = (float2)(b2.f.x, b2.f.y);
- float dotProdB2 = getDotProduct(f2, toB2Unit);
- float v2 = dotProdB2;
- // if move projection to hit axis
- // for both balls are directed to move away
- // one from other then do nothing
- if (v1 <= 0 && v2 >= 0)
- {
- return b1;
- }
- // mass is equal to square of 2d ball
- float m1 = b1.r * b1.r;
- float m2 = b2.r * b2.r;
- // speed of this ball after collision
- float v1new = (2 * m2 * v2 + v1 * (m1 - m2)) / (m1 + m2);
- // "to" move component
- float2 tmpTo = toB2Unit * v1new;
- // "tangent" move component
- float2 tanUnit = rotVect(toB2Unit, M_PI_2_F);
- float dotProdTan1 = getDotProduct(f1, tanUnit);
- float2 tmpTan = tanUnit * dotProdTan1;
- // new move vector
- float2 newF = tmpTo + tmpTan;
- b1.f.x = newF[0];
- b1.f.y = newF[1];
- return b1;
- }
Исходный код находится здесь.
Комментариев нет:
Отправить комментарий