2019-03-18 14:12:40 +00:00
|
|
|
/**
|
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
|
|
|
*/
|
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
var BlockCheckY = require('./BlockCheckY');
|
2019-05-03 16:28:06 +00:00
|
|
|
var GetOverlapY = require('./GetOverlapY');
|
2020-09-25 17:01:40 +00:00
|
|
|
var ProcessY = require('./ProcessY');
|
2017-11-09 13:02:55 +00:00
|
|
|
|
2018-02-09 03:44:23 +00:00
|
|
|
/**
|
2018-10-19 11:32:43 +00:00
|
|
|
* Separates two overlapping bodies on the Y-axis (vertically).
|
|
|
|
*
|
2019-05-03 16:28:06 +00:00
|
|
|
* 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 11:32:43 +00:00
|
|
|
*
|
2019-05-03 16:28:06 +00:00
|
|
|
* The bodies won't be separated if there is no vertical 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.SeparateY
|
|
|
|
* @since 3.0.0
|
|
|
|
*
|
2019-05-03 16:28:06 +00:00
|
|
|
* @param {Phaser.Physics.Arcade.Body} body1 - The first Body to separate.
|
2018-10-19 11:32:43 +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
|
|
|
*
|
2018-10-19 16:45:05 +00:00
|
|
|
* @return {boolean} `true` if the two bodies overlap vertically, otherwise `false`.
|
2018-02-09 03:44:23 +00:00
|
|
|
*/
|
2019-05-03 16:28:06 +00:00
|
|
|
var SeparateY = function (body1, body2, overlapOnly, bias)
|
2017-11-09 13:02:55 +00:00
|
|
|
{
|
2019-05-03 16:28:06 +00:00
|
|
|
var overlap = GetOverlapY(body1, body2, overlapOnly, bias);
|
2019-03-21 01:02:38 +00:00
|
|
|
|
2020-09-24 17:10:11 +00:00
|
|
|
var body1Pushable = body1.pushable;
|
|
|
|
var body2Pushable = body2.pushable;
|
|
|
|
var body1Immovable = body1.immovable;
|
|
|
|
var body2Immovable = body2.immovable;
|
|
|
|
|
2017-11-09 13:02:55 +00:00
|
|
|
// Can't separate two immovable bodies, or a body with its own custom separation logic
|
2020-09-24 17:10:11 +00:00
|
|
|
if (overlapOnly || overlap === 0 || (body1Immovable && body2Immovable) || body1.customSeparateY || body2.customSeparateY)
|
2019-03-13 17:27:11 +00:00
|
|
|
{
|
2019-05-03 16:28:06 +00:00
|
|
|
// return true if there was some overlap, otherwise false
|
|
|
|
return (overlap !== 0) || (body1.embedded && body2.embedded);
|
2019-03-21 01:02:38 +00:00
|
|
|
}
|
2019-03-22 19:08:08 +00:00
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
var blockedState = BlockCheckY(body1, body2, Math.abs(overlap));
|
|
|
|
|
2019-05-03 16:28:06 +00:00
|
|
|
// Adjust their positions and velocities accordingly (if there was any overlap)
|
|
|
|
var v1 = body1.velocity.y;
|
|
|
|
var v2 = body2.velocity.y;
|
2019-03-22 19:08:08 +00:00
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
var body1FullImpact = v2 - v1 * body1.bounce.y;
|
|
|
|
var body2FullImpact = v1 - v2 * body2.bounce.y;
|
|
|
|
|
2020-09-24 17:10:11 +00:00
|
|
|
if (!body1Immovable && !body2Immovable)
|
2019-03-20 04:07:58 +00:00
|
|
|
{
|
2020-09-25 17:01:40 +00:00
|
|
|
if (blockedState > 0)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
2019-03-20 15:00:17 +00:00
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
// negative delta = up, positive delta = down (inc. gravity)
|
2020-09-24 17:10:11 +00:00
|
|
|
overlap = Math.abs(overlap);
|
|
|
|
|
|
|
|
var body1MovingUp = body1._dy < 0;
|
2020-09-25 17:01:40 +00:00
|
|
|
var body1MovingDown = body1._dy > 0;
|
|
|
|
var body1Stationary = body1._dy === 0;
|
2020-09-24 17:10:11 +00:00
|
|
|
|
|
|
|
var body2MovingUp = body2._dy < 0;
|
2020-09-25 17:01:40 +00:00
|
|
|
var body2MovingDown = body2._dy > 0;
|
|
|
|
var body2Stationary = body2._dy === 0;
|
2020-09-24 17:10:11 +00:00
|
|
|
|
|
|
|
var body1OnTop = Math.abs(body1.bottom - body2.y) <= Math.abs(body2.bottom - body1.y);
|
|
|
|
var body2OnTop = !body1OnTop;
|
2019-03-20 15:00:17 +00:00
|
|
|
|
2019-05-03 16:28:06 +00:00
|
|
|
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;
|
2019-03-11 17:27:29 +00:00
|
|
|
|
2019-05-03 16:28:06 +00:00
|
|
|
nv1 -= avg;
|
|
|
|
nv2 -= avg;
|
2019-03-15 19:22:51 +00:00
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
var body1MassImpact = avg + nv1 * body1.bounce.y;
|
|
|
|
var body2MassImpact = avg + nv2 * body2.bounce.y;
|
|
|
|
|
|
|
|
ProcessY.SetProcessY(
|
|
|
|
body1Pushable,
|
|
|
|
body2Pushable,
|
|
|
|
body1MassImpact,
|
|
|
|
body2MassImpact,
|
|
|
|
body1FullImpact,
|
|
|
|
body2FullImpact
|
|
|
|
);
|
|
|
|
|
2020-09-24 17:10:11 +00:00
|
|
|
// -----------------------------------------------------------------------
|
2020-09-25 17:01:40 +00:00
|
|
|
// Pushable Checks
|
2020-09-24 17:10:11 +00:00
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
// Body1 hits Body2 from below
|
|
|
|
if (body1MovingUp && body2OnTop)
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-28 07:07:19 +00:00
|
|
|
// return ProcessY.RunProcessY(body1, body2, overlap, -overlap, body2Stationary, body2MovingDown, body1, 'PushY1');
|
|
|
|
return ProcessY.RunProcessY(body1, body2, overlap, -overlap, body2Stationary, body2MovingDown, body1);
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
// Body2 hits Body1 from below
|
|
|
|
if (body2MovingUp && body1OnTop)
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-28 07:07:19 +00:00
|
|
|
// return ProcessY.RunProcessY(body1, body2, -overlap, overlap, body1Stationary, body1MovingDown, body2, 'PushY2');
|
|
|
|
return ProcessY.RunProcessY(body1, body2, -overlap, overlap, body1Stationary, body1MovingDown, body2);
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
// Body1 hits Body2 from above
|
|
|
|
if (body1MovingDown && body1OnTop)
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-28 07:07:19 +00:00
|
|
|
// return ProcessY.RunProcessY(body1, body2, overlap, -overlap, body2Stationary, body2MovingUp, body1, 'PushY3');
|
|
|
|
return ProcessY.RunProcessY(body1, body2, overlap, -overlap, body2Stationary, body2MovingUp, body1);
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
// Body2 hits Body1 from above
|
|
|
|
if (body2MovingDown && body2OnTop)
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-28 07:07:19 +00:00
|
|
|
// return ProcessY.RunProcessY(body1, body2, -overlap, overlap, body1Stationary, body1MovingUp, body2, 'PushY4');
|
|
|
|
return ProcessY.RunProcessY(body1, body2, -overlap, overlap, body1Stationary, body1MovingUp, body2);
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
console.log('uh oh');
|
|
|
|
}
|
|
|
|
else if (body1Immovable)
|
|
|
|
{
|
2020-09-28 07:07:19 +00:00
|
|
|
// console.log('SepY 1');
|
2020-09-24 17:10:11 +00:00
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
// Body1 is immovable
|
|
|
|
if (blockedState === 1 || blockedState === 3)
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-25 17:01:40 +00:00
|
|
|
// But Body2 cannot go anywhere either, so we cancel out velocity
|
|
|
|
body2.velocity.y = 0;
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
2020-09-25 17:01:40 +00:00
|
|
|
else
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-25 17:01:40 +00:00
|
|
|
body2.y += overlap;
|
|
|
|
body2.velocity.y = body2FullImpact;
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
// This is special case code that handles things like horizontally moving platforms you can ride
|
|
|
|
if (body1.moves)
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-25 17:01:40 +00:00
|
|
|
body2.x += (body1.x - body1.prev.x) * body1.friction.x;
|
|
|
|
body2._dx = body2.x - body2.prev.x;
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
2020-09-25 17:01:40 +00:00
|
|
|
}
|
|
|
|
else if (body2Immovable)
|
|
|
|
{
|
2020-09-28 07:07:19 +00:00
|
|
|
// console.log('SepY 2');
|
2020-09-24 17:10:11 +00:00
|
|
|
|
2020-09-25 17:01:40 +00:00
|
|
|
// Body2 is immovable
|
|
|
|
if (blockedState === 2 || blockedState === 4)
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-25 17:01:40 +00:00
|
|
|
// But Body1 cannot go anywhere either, so we cancel out velocity
|
|
|
|
body1.velocity.y = 0;
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
2020-09-25 17:01:40 +00:00
|
|
|
else
|
2020-09-24 17:10:11 +00:00
|
|
|
{
|
2020-09-25 17:01:40 +00:00
|
|
|
body1.y -= overlap;
|
|
|
|
body1.velocity.y = body1FullImpact;
|
2020-09-24 17:10:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This is special case code that handles things like horizontally moving platforms you can ride
|
2019-05-03 16:28:06 +00:00
|
|
|
if (body2.moves)
|
2019-03-13 17:27:11 +00:00
|
|
|
{
|
2019-05-03 16:28:06 +00:00
|
|
|
body1.x += (body2.x - body2.prev.x) * body2.friction.x;
|
2020-07-07 18:31:54 +00:00
|
|
|
body1._dx = body1.x - body1.prev.x;
|
2019-03-18 14:12:40 +00:00
|
|
|
}
|
2019-03-13 17:27:11 +00:00
|
|
|
}
|
2019-03-11 17:27:29 +00:00
|
|
|
|
2017-11-09 13:02:55 +00:00
|
|
|
// If we got this far then there WAS overlap, and separation is complete, so return true
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = SeparateY;
|