2011-05-21 14:58:54 +08:00
|
|
|
/* Copyright (c) 2007 Scott Lembcke
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include "chipmunk_private.h"
|
|
|
|
|
|
|
|
#pragma mark Point Query Functions
|
|
|
|
|
|
|
|
typedef struct pointQueryContext {
|
|
|
|
cpLayers layers;
|
|
|
|
cpGroup group;
|
|
|
|
cpSpacePointQueryFunc func;
|
|
|
|
void *data;
|
|
|
|
} pointQueryContext;
|
|
|
|
|
|
|
|
static void
|
|
|
|
pointQueryHelper(cpVect *point, cpShape *shape, pointQueryContext *context)
|
|
|
|
{
|
|
|
|
if(
|
|
|
|
!(shape->group && context->group == shape->group) && (context->layers&shape->layers) &&
|
|
|
|
cpShapePointQuery(shape, *point)
|
|
|
|
){
|
|
|
|
context->func(shape, context->data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cpSpacePointQuery(cpSpace *space, cpVect point, cpLayers layers, cpGroup group, cpSpacePointQueryFunc func, void *data)
|
|
|
|
{
|
|
|
|
pointQueryContext context = {layers, group, func, data};
|
|
|
|
|
|
|
|
cpSpaceLock(space); {
|
|
|
|
cpSpaceHashPointQuery(space->activeShapes, point, (cpSpaceHashQueryFunc)pointQueryHelper, &context);
|
|
|
|
cpSpaceHashPointQuery(space->staticShapes, point, (cpSpaceHashQueryFunc)pointQueryHelper, &context);
|
|
|
|
} cpSpaceUnlock(space);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
rememberLastPointQuery(cpShape *shape, cpShape **outShape)
|
|
|
|
{
|
|
|
|
if(!shape->sensor) *outShape = shape;
|
|
|
|
}
|
|
|
|
|
|
|
|
cpShape *
|
|
|
|
cpSpacePointQueryFirst(cpSpace *space, cpVect point, cpLayers layers, cpGroup group)
|
|
|
|
{
|
|
|
|
cpShape *shape = NULL;
|
|
|
|
cpSpacePointQuery(space, point, layers, group, (cpSpacePointQueryFunc)rememberLastPointQuery, &shape);
|
|
|
|
|
|
|
|
return shape;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark Segment Query Functions
|
|
|
|
|
|
|
|
typedef struct segQueryContext {
|
|
|
|
cpVect start, end;
|
|
|
|
cpLayers layers;
|
|
|
|
cpGroup group;
|
|
|
|
cpSpaceSegmentQueryFunc func;
|
|
|
|
} segQueryContext;
|
|
|
|
|
|
|
|
static cpFloat
|
|
|
|
segQueryFunc(segQueryContext *context, cpShape *shape, void *data)
|
|
|
|
{
|
|
|
|
cpSegmentQueryInfo info;
|
|
|
|
|
|
|
|
if(
|
|
|
|
!(shape->group && context->group == shape->group) && (context->layers&shape->layers) &&
|
|
|
|
cpShapeSegmentQuery(shape, context->start, context->end, &info)
|
|
|
|
){
|
|
|
|
context->func(shape, info.t, info.n, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cpSpaceSegmentQuery(cpSpace *space, cpVect start, cpVect end, cpLayers layers, cpGroup group, cpSpaceSegmentQueryFunc func, void *data)
|
|
|
|
{
|
|
|
|
segQueryContext context = {
|
|
|
|
start, end,
|
|
|
|
layers, group,
|
|
|
|
func,
|
|
|
|
};
|
|
|
|
|
|
|
|
cpSpaceLock(space); {
|
|
|
|
cpSpaceHashSegmentQuery(space->staticShapes, &context, start, end, 1.0f, (cpSpaceHashSegmentQueryFunc)segQueryFunc, data);
|
|
|
|
cpSpaceHashSegmentQuery(space->activeShapes, &context, start, end, 1.0f, (cpSpaceHashSegmentQueryFunc)segQueryFunc, data);
|
|
|
|
} cpSpaceUnlock(space);
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct segQueryFirstContext {
|
|
|
|
cpVect start, end;
|
|
|
|
cpLayers layers;
|
|
|
|
cpGroup group;
|
|
|
|
} segQueryFirstContext;
|
|
|
|
|
|
|
|
static cpFloat
|
|
|
|
segQueryFirst(segQueryFirstContext *context, cpShape *shape, cpSegmentQueryInfo *out)
|
|
|
|
{
|
|
|
|
cpSegmentQueryInfo info;
|
|
|
|
|
|
|
|
if(
|
|
|
|
!(shape->group && context->group == shape->group) &&
|
|
|
|
(context->layers&shape->layers) &&
|
|
|
|
!shape->sensor &&
|
|
|
|
cpShapeSegmentQuery(shape, context->start, context->end, &info) &&
|
|
|
|
info.t < out->t
|
|
|
|
){
|
|
|
|
*out = info;
|
|
|
|
}
|
|
|
|
|
|
|
|
return out->t;
|
|
|
|
}
|
|
|
|
|
|
|
|
cpShape *
|
|
|
|
cpSpaceSegmentQueryFirst(cpSpace *space, cpVect start, cpVect end, cpLayers layers, cpGroup group, cpSegmentQueryInfo *out)
|
|
|
|
{
|
|
|
|
cpSegmentQueryInfo info = {NULL, 1.0f, cpvzero};
|
|
|
|
if(out){
|
|
|
|
(*out) = info;
|
|
|
|
} else {
|
|
|
|
out = &info;
|
|
|
|
}
|
|
|
|
|
|
|
|
segQueryFirstContext context = {
|
|
|
|
start, end,
|
|
|
|
layers, group
|
|
|
|
};
|
|
|
|
|
|
|
|
cpSpaceHashSegmentQuery(space->staticShapes, &context, start, end, 1.0f, (cpSpaceHashSegmentQueryFunc)segQueryFirst, out);
|
|
|
|
cpSpaceHashSegmentQuery(space->activeShapes, &context, start, end, out->t, (cpSpaceHashSegmentQueryFunc)segQueryFirst, out);
|
|
|
|
|
|
|
|
return out->shape;
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark BB Query Functions
|
|
|
|
|
|
|
|
typedef struct bbQueryContext {
|
|
|
|
cpLayers layers;
|
|
|
|
cpGroup group;
|
|
|
|
cpSpaceBBQueryFunc func;
|
|
|
|
void *data;
|
|
|
|
} bbQueryContext;
|
|
|
|
|
|
|
|
static void
|
|
|
|
bbQueryHelper(cpBB *bb, cpShape *shape, bbQueryContext *context)
|
|
|
|
{
|
|
|
|
if(
|
|
|
|
!(shape->group && context->group == shape->group) && (context->layers&shape->layers) &&
|
|
|
|
cpBBintersects(*bb, shape->bb)
|
|
|
|
){
|
|
|
|
context->func(shape, context->data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cpSpaceBBQuery(cpSpace *space, cpBB bb, cpLayers layers, cpGroup group, cpSpaceBBQueryFunc func, void *data)
|
|
|
|
{
|
|
|
|
bbQueryContext context = {layers, group, func, data};
|
|
|
|
|
|
|
|
cpSpaceLock(space); {
|
|
|
|
cpSpaceHashQuery(space->activeShapes, &bb, bb, (cpSpaceHashQueryFunc)bbQueryHelper, &context);
|
|
|
|
cpSpaceHashQuery(space->staticShapes, &bb, bb, (cpSpaceHashQueryFunc)bbQueryHelper, &context);
|
|
|
|
} cpSpaceUnlock(space);
|
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark Shape Query Functions
|
|
|
|
|
|
|
|
typedef struct shapeQueryContext {
|
|
|
|
cpSpaceShapeQueryFunc func;
|
|
|
|
void *data;
|
|
|
|
cpBool anyCollision;
|
|
|
|
} shapeQueryContext;
|
|
|
|
|
|
|
|
// Callback from the spatial hash.
|
|
|
|
static void
|
|
|
|
shapeQueryHelper(cpShape *a, cpShape *b, shapeQueryContext *context)
|
|
|
|
{
|
|
|
|
// Reject any of the simple cases
|
|
|
|
if(
|
|
|
|
(a->group && a->group == b->group) ||
|
|
|
|
!(a->layers & b->layers) ||
|
2011-07-11 10:34:03 +08:00
|
|
|
a == b
|
2011-05-21 14:58:54 +08:00
|
|
|
) return;
|
|
|
|
|
|
|
|
cpContact contacts[CP_MAX_CONTACTS_PER_ARBITER];
|
|
|
|
int numContacts = 0;
|
|
|
|
|
|
|
|
// Shape 'a' should have the lower shape type. (required by cpCollideShapes() )
|
|
|
|
if(a->klass->type <= b->klass->type){
|
|
|
|
numContacts = cpCollideShapes(a, b, contacts);
|
|
|
|
} else {
|
|
|
|
numContacts = cpCollideShapes(b, a, contacts);
|
|
|
|
for(int i=0; i<numContacts; i++) contacts[i].n = cpvneg(contacts[i].n);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(numContacts){
|
2011-07-11 10:34:03 +08:00
|
|
|
context->anyCollision = !(a->sensor || b->sensor);
|
2011-05-21 14:58:54 +08:00
|
|
|
|
|
|
|
if(context->func){
|
|
|
|
cpContactPointSet set = {numContacts, {}};
|
|
|
|
for(int i=0; i<set.count; i++){
|
|
|
|
set.points[i].point = contacts[i].p;
|
|
|
|
set.points[i].normal = contacts[i].p;
|
|
|
|
set.points[i].dist = contacts[i].dist;
|
|
|
|
}
|
|
|
|
|
|
|
|
context->func(b, &set, context->data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cpBool
|
|
|
|
cpSpaceShapeQuery(cpSpace *space, cpShape *shape, cpSpaceShapeQueryFunc func, void *data)
|
|
|
|
{
|
|
|
|
cpBB bb = cpShapeCacheBB(shape);
|
|
|
|
shapeQueryContext context = {func, data, cpFalse};
|
|
|
|
|
|
|
|
cpSpaceLock(space); {
|
|
|
|
cpSpaceHashQuery(space->activeShapes, shape, bb, (cpSpaceHashQueryFunc)shapeQueryHelper, &context);
|
|
|
|
cpSpaceHashQuery(space->staticShapes, shape, bb, (cpSpaceHashQueryFunc)shapeQueryHelper, &context);
|
|
|
|
} cpSpaceUnlock(space);
|
|
|
|
|
|
|
|
return context.anyCollision;
|
|
|
|
}
|