phaser/src/physics/arcade/SeparateX.js

180 lines
6.1 KiB
JavaScript
Raw Normal View History

2018-02-12 16:01:20 +00:00
/**
* @author Richard Davey <rich@photonstorm.com>
2020-01-15 12:07:09 +00:00
* @copyright 2020 Photon Storm Ltd.
2019-05-10 15:15:04 +00:00
* @license {@link https://opensource.org/licenses/MIT|MIT License}
2018-02-12 16:01:20 +00:00
*/
var BlockCheckX = require('./BlockCheckX');
var GetOverlapX = require('./GetOverlapX');
var ProcessX = require('./ProcessX');
2018-02-09 03:44:23 +00:00
/**
2018-10-19 16:45:05 +00:00
* Separates two overlapping bodies on the X-axis (horizontally).
*
* Separation involves moving two overlapping bodies so they don't overlap anymore and adjusting their velocities based on their mass. This is a core part of collision detection.
2018-10-19 16:45:05 +00:00
*
* The bodies won't be separated if there is no horizontal overlap between them, if they are static, or if either one uses custom logic for its separation.
2018-02-09 03:44:23 +00:00
*
* @function Phaser.Physics.Arcade.SeparateX
* @since 3.0.0
*
* @param {Phaser.Physics.Arcade.Body} body1 - The first Body to separate.
2018-10-19 16:45:05 +00:00
* @param {Phaser.Physics.Arcade.Body} body2 - The second Body to separate.
* @param {boolean} overlapOnly - If `true`, the bodies will only have their overlap data set and no separation will take place.
* @param {number} bias - A value to add to the delta value during overlap checking. Used to prevent sprite tunneling.
2018-02-09 03:44:23 +00:00
*
* @return {boolean} `true` if the two bodies overlap vertically, otherwise `false`.
2018-02-09 03:44:23 +00:00
*/
var SeparateX = function (body1, body2, overlapOnly, bias)
{
var overlap = GetOverlapX(body1, body2, overlapOnly, bias);
var body1Pushable = body1.pushable;
var body2Pushable = body2.pushable;
var body1Immovable = body1.immovable;
var body2Immovable = body2.immovable;
// Can't separate two immovable bodies, or a body with its own custom separation logic
if (overlapOnly || overlap === 0 || (body1Immovable && body2Immovable) || body1.customSeparateX || body2.customSeparateX)
2019-03-24 23:30:57 +00:00
{
// return true if there was some overlap, otherwise false
return (overlap !== 0) || (body1.embedded && body2.embedded);
2019-03-24 23:30:57 +00:00
}
var blockedState = BlockCheckX(body1, body2, Math.abs(overlap));
// Adjust their positions and velocities accordingly (if there was any overlap)
var v1 = body1.velocity.x;
var v2 = body2.velocity.x;
2019-03-24 23:30:57 +00:00
var body1FullImpact = v2 - v1 * body1.bounce.x;
var body2FullImpact = v1 - v2 * body2.bounce.x;
if (!body1Immovable && !body2Immovable)
2019-03-24 23:30:57 +00:00
{
if (blockedState > 0)
{
return true;
}
// negative delta = up, positive delta = down (inc. gravity)
overlap = Math.abs(overlap);
var body1MovingLeft = body1._dx < 0;
var body1MovingRight = body1._dx > 0;
var body1Stationary = body1._dx === 0;
var body2MovingLeft = body2._dx < 0;
var body2MovingRight = body2._dx > 0;
var body2Stationary = body2._dx === 0;
var body1OnLeft = Math.abs(body1.right - body2.x) <= Math.abs(body2.right - body1.x);
var body2OnLeft = !body1OnLeft;
var nv1 = Math.sqrt((v2 * v2 * body2.mass) / body1.mass) * ((v2 > 0) ? 1 : -1);
var nv2 = Math.sqrt((v1 * v1 * body1.mass) / body2.mass) * ((v1 > 0) ? 1 : -1);
var avg = (nv1 + nv2) * 0.5;
nv1 -= avg;
nv2 -= avg;
var body1MassImpact = avg + nv1 * body1.bounce.y;
var body2MassImpact = avg + nv2 * body2.bounce.y;
ProcessX.SetProcessX(
body1Pushable,
body2Pushable,
body1MassImpact,
body2MassImpact,
body1FullImpact,
body2FullImpact
);
// -----------------------------------------------------------------------
// Pushable Checks
// -----------------------------------------------------------------------
// Body1 hits Body2 on its right side
if (body1MovingLeft && body2OnLeft)
{
return ProcessX.RunProcessX(body1, body2, overlap, -overlap, body2Stationary, body2MovingRight, 'PushX1');
}
// Body2 hits Body1 on its right side
if (body2MovingLeft && body1OnLeft)
{
return ProcessX.RunProcessX(body1, body2, -overlap, overlap, body1Stationary, body1MovingRight, 'PushX2');
}
// Body1 hits Body2 from above
/*
if (body1MovingDown && body1OnTop)
{
return ProcessX.RunProcessX(body1, body2, overlap, -overlap, body2Stationary, body2MovingUp, 'PushX3');
}
2019-03-24 23:30:57 +00:00
// Body2 hits Body1 from above
if (body2MovingDown && body2OnTop)
2019-03-24 23:30:57 +00:00
{
return ProcessX.RunProcessX(body1, body2, -overlap, overlap, body1Stationary, body1MovingUp, 'PushX4');
2019-03-24 23:30:57 +00:00
}
*/
console.log('uh oh');
// console.log('body1MovingUp', body1MovingUp, 'body2MovingUp', body2MovingUp, 'body1OnTop', body1OnTop, 'body2OnTop', body2OnTop);
2019-03-24 23:30:57 +00:00
}
else if (body1Immovable)
2019-03-24 23:30:57 +00:00
{
console.log('SepX 1');
// Body1 is immovable
if (blockedState === 1 || blockedState === 3)
{
// But Body2 cannot go anywhere either, so we cancel out velocity
body2.velocity.x = 0;
}
else
{
body2.x += overlap;
body2.velocity.x = body2FullImpact;
}
2019-03-24 23:30:57 +00:00
// This is special case code that handles things like vertically moving platforms you can ride
if (body1.moves)
2019-03-24 23:30:57 +00:00
{
body2.y += (body1.y - body1.prev.y) * body1.friction.y;
body2._dy = body2.y - body2.prev.y;
}
}
else if (body2Immovable)
{
console.log('SepX 2');
// Body2 is immovable
if (blockedState === 2 || blockedState === 4)
{
// But Body1 cannot go anywhere either, so we cancel out velocity
body1.velocity.x = 0;
}
else
{
body1.x -= overlap;
body1.velocity.x = body1FullImpact;
}
// This is special case code that handles things like vertically moving platforms you can ride
if (body2.moves)
{
body1.y += (body2.y - body2.prev.y) * body2.friction.y;
body1._dy = body1.y - body1.prev.y;
}
}
// If we got this far then there WAS overlap, and separation is complete, so return true
return true;
};
module.exports = SeparateX;