mirror of
https://github.com/liabru/matter-js.git
synced 2025-01-13 16:18:50 -05:00
composites now support removals (implemented isModified flag)
This commit is contained in:
parent
4514f2d598
commit
b253683cb5
5 changed files with 187 additions and 35 deletions
|
@ -11,6 +11,8 @@ var Composite = {};
|
||||||
|
|
||||||
(function() {
|
(function() {
|
||||||
|
|
||||||
|
var _nextId = 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Description
|
* Description
|
||||||
* @method create
|
* @method create
|
||||||
|
@ -19,12 +21,49 @@ var Composite = {};
|
||||||
*/
|
*/
|
||||||
Composite.create = function(options) {
|
Composite.create = function(options) {
|
||||||
return Common.extend({
|
return Common.extend({
|
||||||
|
id: Composite.nextId(),
|
||||||
|
parent: null,
|
||||||
|
isModified: false,
|
||||||
bodies: [],
|
bodies: [],
|
||||||
constraints: [],
|
constraints: [],
|
||||||
composites: []
|
composites: []
|
||||||
}, options);
|
}, options);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the next unique compositeID
|
||||||
|
* @method nextId
|
||||||
|
* @return {Number} Unique compositeID
|
||||||
|
*/
|
||||||
|
Composite.nextId = function() {
|
||||||
|
return _nextId++;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the composite's `isModified` flag.
|
||||||
|
* If `updateParents` is true, all parents will be set (default: false).
|
||||||
|
* If `updateChildren` is true, all children will be set (default: false).
|
||||||
|
* @method setModified
|
||||||
|
* @param {composite} composite
|
||||||
|
* @param {boolean} isModified
|
||||||
|
* @param {boolean} updateParents
|
||||||
|
* @param {boolean} updateChildren
|
||||||
|
*/
|
||||||
|
Composite.setModified = function(composite, isModified, updateParents, updateChildren) {
|
||||||
|
composite.isModified = isModified;
|
||||||
|
|
||||||
|
if (updateParents && composite.parent) {
|
||||||
|
Composite.setModified(composite.parent, isModified, updateParents, updateChildren);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (updateChildren) {
|
||||||
|
for(var i = 0; i < composite.composites.length; i++) {
|
||||||
|
var childComposite = composite.composites[i];
|
||||||
|
Composite.setModified(childComposite, isModified, updateParents, updateChildren);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Description
|
* Description
|
||||||
* @method addComposite
|
* @method addComposite
|
||||||
|
@ -34,6 +73,8 @@ var Composite = {};
|
||||||
*/
|
*/
|
||||||
Composite.addComposite = function(compositeA, compositeB) {
|
Composite.addComposite = function(compositeA, compositeB) {
|
||||||
compositeA.composites.push(compositeB);
|
compositeA.composites.push(compositeB);
|
||||||
|
compositeB.parent = compositeA;
|
||||||
|
Composite.setModified(compositeA, true, true, false);
|
||||||
return compositeA;
|
return compositeA;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -46,6 +87,44 @@ var Composite = {};
|
||||||
*/
|
*/
|
||||||
Composite.addBody = function(composite, body) {
|
Composite.addBody = function(composite, body) {
|
||||||
composite.bodies.push(body);
|
composite.bodies.push(body);
|
||||||
|
Composite.setModified(composite, true, true, false);
|
||||||
|
return composite;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes a body from the given composite, and optionally searching its children recursively
|
||||||
|
* @method removeBody
|
||||||
|
* @param {composite} composite
|
||||||
|
* @param {body} body
|
||||||
|
* @param {boolean} deep
|
||||||
|
* @return {composite} The original composite with the body removed
|
||||||
|
*/
|
||||||
|
Composite.removeBody = function(composite, body, deep) {
|
||||||
|
var position = composite.bodies.indexOf(body);
|
||||||
|
if (position !== -1) {
|
||||||
|
Composite.removeBodyAt(composite, position);
|
||||||
|
Composite.setModified(composite, true, true, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deep) {
|
||||||
|
for (var i = 0; i < composite.composites.length; i++){
|
||||||
|
Composite.removeBody(composite.composites[i], body, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return composite;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes a body from the given composite
|
||||||
|
* @method removeBodyAt
|
||||||
|
* @param {composite} composite
|
||||||
|
* @param {number} position
|
||||||
|
* @return {composite} The original composite with the body removed
|
||||||
|
*/
|
||||||
|
Composite.removeBodyAt = function(composite, position) {
|
||||||
|
composite.bodies.splice(position, 1);
|
||||||
|
Composite.setModified(composite, true, true, false);
|
||||||
return composite;
|
return composite;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -58,6 +137,71 @@ var Composite = {};
|
||||||
*/
|
*/
|
||||||
Composite.addConstraint = function(composite, constraint) {
|
Composite.addConstraint = function(composite, constraint) {
|
||||||
composite.constraints.push(constraint);
|
composite.constraints.push(constraint);
|
||||||
|
Composite.setModified(composite, true, true, false);
|
||||||
|
return composite;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes a constraint from the given composite, and optionally searching its children recursively
|
||||||
|
* @method removeConstraint
|
||||||
|
* @param {composite} composite
|
||||||
|
* @param {constraint} constraint
|
||||||
|
* @param {boolean} deep
|
||||||
|
* @return {composite} The original composite with the constraint removed
|
||||||
|
*/
|
||||||
|
Composite.removeConstraint = function(composite, constraint, deep) {
|
||||||
|
var position = composite.constraints.indexOf(constraint);
|
||||||
|
if (position !== -1) {
|
||||||
|
Composite.removeConstraintAt(composite, position);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deep) {
|
||||||
|
for (var i = 0; i < composite.composites.length; i++){
|
||||||
|
Composite.removeConstraint(composite.composites[i], constraint, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return composite;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes a body from the given composite
|
||||||
|
* @method removeConstraintAt
|
||||||
|
* @param {composite} composite
|
||||||
|
* @param {number} position
|
||||||
|
* @return {composite} The original composite with the constraint removed
|
||||||
|
*/
|
||||||
|
Composite.removeConstraintAt = function(composite, position) {
|
||||||
|
composite.constraints.splice(position, 1);
|
||||||
|
Composite.setModified(composite, true, true, false);
|
||||||
|
return composite;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Removes all bodies, constraints and composites from the given composite
|
||||||
|
* Optionally clearing its children recursively
|
||||||
|
* @method clear
|
||||||
|
* @param {world} world
|
||||||
|
* @param {boolean} keepStatic
|
||||||
|
* @param {boolean} deep
|
||||||
|
*/
|
||||||
|
Composite.clear = function(composite, keepStatic, deep) {
|
||||||
|
if (deep) {
|
||||||
|
for (var i = 0; i < composite.composites.length; i++){
|
||||||
|
Composite.clear(composite.composites[i], keepStatic, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (keepStatic) {
|
||||||
|
composite.bodies = composite.bodies.filter(function(body) { return body.isStatic; });
|
||||||
|
} else {
|
||||||
|
composite.bodies.length = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
composite.constraints.length = 0;
|
||||||
|
composite.composites.length = 0;
|
||||||
|
Composite.setModified(composite, true, true, false);
|
||||||
|
|
||||||
return composite;
|
return composite;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -91,23 +235,4 @@ var Composite = {};
|
||||||
return constraints;
|
return constraints;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Removes all bodies, constraints and composites from the given composite (non-recursive)
|
|
||||||
* @method clear
|
|
||||||
* @param {world} world
|
|
||||||
* @param {boolean} keepStatic
|
|
||||||
*/
|
|
||||||
Composite.clear = function(composite, keepStatic) {
|
|
||||||
if (keepStatic) {
|
|
||||||
composite.bodies = composite.bodies.filter(function(body) { return body.isStatic; });
|
|
||||||
} else {
|
|
||||||
composite.bodies.length = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
composite.constraints.length = 0;
|
|
||||||
composite.composites.length = 0;
|
|
||||||
|
|
||||||
return composite;
|
|
||||||
};
|
|
||||||
|
|
||||||
})();
|
})();
|
|
@ -48,7 +48,7 @@ var Grid = {};
|
||||||
for (i = 0; i < bodies.length; i++) {
|
for (i = 0; i < bodies.length; i++) {
|
||||||
var body = bodies[i];
|
var body = bodies[i];
|
||||||
|
|
||||||
if (body.isSleeping)
|
if (body.isSleeping && !forceUpdate)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
// don't update out of world bodies
|
// don't update out of world bodies
|
||||||
|
|
|
@ -167,6 +167,10 @@ var Engine = {};
|
||||||
*/
|
*/
|
||||||
Events.trigger(engine, 'tick beforeUpdate', event);
|
Events.trigger(engine, 'tick beforeUpdate', event);
|
||||||
|
|
||||||
|
// if world has been modified, clear the render scene graph
|
||||||
|
if (engine.world.isModified)
|
||||||
|
engine.render.controller.clear(engine.render);
|
||||||
|
|
||||||
// update
|
// update
|
||||||
Engine.update(engine, delta, correction);
|
Engine.update(engine, delta, correction);
|
||||||
|
|
||||||
|
@ -234,18 +238,24 @@ var Engine = {};
|
||||||
broadphasePairs = [],
|
broadphasePairs = [],
|
||||||
i;
|
i;
|
||||||
|
|
||||||
|
// get lists of all bodies and constraints, no matter what composites they are in
|
||||||
var allBodies = Composite.allBodies(world),
|
var allBodies = Composite.allBodies(world),
|
||||||
allConstraints = Composite.allConstraints(world);
|
allConstraints = Composite.allConstraints(world);
|
||||||
|
|
||||||
|
// reset metrics logging
|
||||||
Metrics.reset(engine.metrics);
|
Metrics.reset(engine.metrics);
|
||||||
|
|
||||||
|
// if sleeping enabled, call the sleeping controller
|
||||||
if (engine.enableSleeping)
|
if (engine.enableSleeping)
|
||||||
Sleeping.update(allBodies);
|
Sleeping.update(allBodies);
|
||||||
|
|
||||||
|
// applies gravity to all bodies
|
||||||
Body.applyGravityAll(allBodies, world.gravity);
|
Body.applyGravityAll(allBodies, world.gravity);
|
||||||
|
|
||||||
|
// update the mouse constraint
|
||||||
MouseConstraint.update(engine.mouseConstraint, allBodies, engine.input);
|
MouseConstraint.update(engine.mouseConstraint, allBodies, engine.input);
|
||||||
|
|
||||||
|
// update all body position and rotation by integration
|
||||||
Body.updateAll(allBodies, delta * engine.timeScale, correction, world.bounds);
|
Body.updateAll(allBodies, delta * engine.timeScale, correction, world.bounds);
|
||||||
|
|
||||||
// update all constraints
|
// update all constraints
|
||||||
|
@ -255,16 +265,24 @@ var Engine = {};
|
||||||
|
|
||||||
// broadphase pass: find potential collision pairs
|
// broadphase pass: find potential collision pairs
|
||||||
if (broadphase.controller) {
|
if (broadphase.controller) {
|
||||||
broadphase.controller.update(broadphase.instance, allBodies, engine);
|
|
||||||
|
// if world is dirty, we must flush the whole grid
|
||||||
|
if (world.isModified)
|
||||||
|
broadphase.controller.clear(broadphase.instance);
|
||||||
|
|
||||||
|
// update the grid buckets based on current bodies
|
||||||
|
broadphase.controller.update(broadphase.instance, allBodies, engine, world.isModified);
|
||||||
broadphasePairs = broadphase.instance.pairsList;
|
broadphasePairs = broadphase.instance.pairsList;
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
|
// if no broadphase set, we just pass all bodies
|
||||||
broadphasePairs = allBodies;
|
broadphasePairs = allBodies;
|
||||||
}
|
}
|
||||||
|
|
||||||
// narrowphase pass: find actual collisions, then create or update collision pairs
|
// narrowphase pass: find actual collisions, then create or update collision pairs
|
||||||
var collisions = broadphase.detector(broadphasePairs, engine);
|
var collisions = broadphase.detector(broadphasePairs, engine);
|
||||||
|
|
||||||
// update pairs
|
// update collision pairs
|
||||||
var pairs = engine.pairs,
|
var pairs = engine.pairs,
|
||||||
timestamp = engine.timing.timestamp;
|
timestamp = engine.timing.timestamp;
|
||||||
Pairs.update(pairs, collisions, timestamp);
|
Pairs.update(pairs, collisions, timestamp);
|
||||||
|
@ -286,11 +304,16 @@ var Engine = {};
|
||||||
}
|
}
|
||||||
Resolver.postSolvePosition(allBodies);
|
Resolver.postSolvePosition(allBodies);
|
||||||
|
|
||||||
|
// update metrics log
|
||||||
Metrics.update(engine.metrics, engine);
|
Metrics.update(engine.metrics, engine);
|
||||||
|
|
||||||
// clear force buffers
|
// clear force buffers
|
||||||
Body.resetForcesAll(allBodies);
|
Body.resetForcesAll(allBodies);
|
||||||
|
|
||||||
|
// clear all composite modified flags
|
||||||
|
if (world.isModified)
|
||||||
|
Composite.setModified(world, false, false, true);
|
||||||
|
|
||||||
return engine;
|
return engine;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -315,14 +338,6 @@ var Engine = {};
|
||||||
Sleeping.set(body, false);
|
Sleeping.set(body, false);
|
||||||
body.id = Body.nextId();
|
body.id = Body.nextId();
|
||||||
}
|
}
|
||||||
|
|
||||||
var broadphase = engineA.broadphase[engineA.broadphase.current];
|
|
||||||
|
|
||||||
if (broadphase.controller === Grid) {
|
|
||||||
var grid = broadphase.instance;
|
|
||||||
Grid.clear(grid);
|
|
||||||
Grid.update(grid, bodies, engineA, true);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -339,12 +354,9 @@ var Engine = {};
|
||||||
World.addConstraint(engine.world, engine.mouseConstraint.constraint);
|
World.addConstraint(engine.world, engine.mouseConstraint.constraint);
|
||||||
|
|
||||||
var broadphase = engine.broadphase[engine.broadphase.current];
|
var broadphase = engine.broadphase[engine.broadphase.current];
|
||||||
|
|
||||||
if (broadphase.controller === Grid)
|
|
||||||
Grid.clear(broadphase.instance);
|
|
||||||
|
|
||||||
if (broadphase.controller) {
|
if (broadphase.controller) {
|
||||||
var bodies = Composite.allBodies(world);
|
var bodies = Composite.allBodies(world);
|
||||||
|
broadphase.controller.clear(broadphase.instance);
|
||||||
broadphase.controller.update(broadphase.instance, bodies, engine, true);
|
broadphase.controller.update(broadphase.instance, bodies, engine, true);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -132,7 +132,12 @@ var Gui = {};
|
||||||
|
|
||||||
var physics = datGui.addFolder('Engine');
|
var physics = datGui.addFolder('Engine');
|
||||||
physics.add(engine, 'enableSleeping');
|
physics.add(engine, 'enableSleeping');
|
||||||
physics.add(engine.broadphase, 'current', ['grid', 'bruteForce']);
|
|
||||||
|
physics.add(engine.broadphase, 'current', ['grid', 'bruteForce'])
|
||||||
|
.onFinishChange(function(value) {
|
||||||
|
Composite.setModified(engine.world, true, false, false);
|
||||||
|
});
|
||||||
|
|
||||||
physics.add(engine, 'timeScale', 0.1, 2).step(0.1);
|
physics.add(engine, 'timeScale', 0.1, 2).step(0.1);
|
||||||
physics.add(engine, 'velocityIterations', 1, 10).step(1);
|
physics.add(engine, 'velocityIterations', 1, 10).step(1);
|
||||||
physics.add(engine, 'positionIterations', 1, 10).step(1);
|
physics.add(engine, 'positionIterations', 1, 10).step(1);
|
||||||
|
|
|
@ -61,6 +61,16 @@ var Render = {};
|
||||||
return render;
|
return render;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clears the renderer. In this implementation, this is a noop.
|
||||||
|
* @method clear
|
||||||
|
* @param {RenderPixi} render
|
||||||
|
*/
|
||||||
|
Render.clear = function(render) {
|
||||||
|
// nothing required to clear this renderer implentation
|
||||||
|
// if a scene graph is required, clear it here (see RenderPixi.js)
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the background CSS property of the canvas
|
* Sets the background CSS property of the canvas
|
||||||
* @method setBackground
|
* @method setBackground
|
||||||
|
|
Loading…
Reference in a new issue