2010-09-04 18:18:14 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
IMPORTANT - READ ME!
|
|
|
|
|
|
|
|
This file sets up a simple interface that the individual demos can use to get
|
|
|
|
a Chipmunk space running and draw what's in it. In order to keep the Chipmunk
|
|
|
|
examples clean and simple, they contain no graphics code. All drawing is done
|
|
|
|
by accessing the Chipmunk structures at a very low level. It is NOT
|
|
|
|
recommended to write a game or application this way as it does not scale
|
|
|
|
beyond simple shape drawing and is very dependent on implementation details
|
|
|
|
about Chipmunk which may change with little to no warning.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
|
|
#include "cocos2d.h"
|
|
|
|
#include "cocos2dChipmunkDemo.h"
|
|
|
|
|
|
|
|
#define SLEEP_TICKS 16
|
|
|
|
|
|
|
|
//extern chipmunkDemo Test;
|
|
|
|
extern chipmunkDemo LogoSmash;
|
|
|
|
extern chipmunkDemo Simple;
|
|
|
|
extern chipmunkDemo PyramidStack;
|
|
|
|
extern chipmunkDemo Plink;
|
|
|
|
extern chipmunkDemo Tumble;
|
|
|
|
extern chipmunkDemo PyramidTopple;
|
|
|
|
extern chipmunkDemo Bounce;
|
|
|
|
extern chipmunkDemo Planet;
|
|
|
|
extern chipmunkDemo Springies;
|
|
|
|
extern chipmunkDemo Pump;
|
|
|
|
extern chipmunkDemo TheoJansen;
|
|
|
|
extern chipmunkDemo MagnetsElectric;
|
|
|
|
extern chipmunkDemo UnsafeOps;
|
|
|
|
extern chipmunkDemo Query;
|
|
|
|
extern chipmunkDemo OneWay;
|
|
|
|
extern chipmunkDemo Player;
|
|
|
|
extern chipmunkDemo Sensors;
|
|
|
|
extern chipmunkDemo Joints;
|
|
|
|
|
|
|
|
//extern chipmunkDemo Test;
|
|
|
|
|
|
|
|
static chipmunkDemo *demos[] = {
|
|
|
|
&LogoSmash,
|
|
|
|
&Simple,
|
|
|
|
&PyramidStack,
|
|
|
|
&Plink,
|
|
|
|
&Tumble,
|
|
|
|
&PyramidTopple,
|
|
|
|
&Bounce,
|
|
|
|
&Planet,
|
|
|
|
&Springies,
|
|
|
|
&Pump,
|
|
|
|
&TheoJansen,
|
|
|
|
&MagnetsElectric,
|
|
|
|
&UnsafeOps,
|
|
|
|
// &Query,
|
|
|
|
&OneWay,
|
|
|
|
&Player,
|
|
|
|
&Sensors,
|
|
|
|
&Joints,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int maxDemos = sizeof(demos) / sizeof(demos[0]);
|
|
|
|
|
|
|
|
static const int demoCount = sizeof(demos)/sizeof(chipmunkDemo *);
|
|
|
|
static chipmunkDemo *currDemo = NULL;
|
|
|
|
static const int firstDemoIndex = 'a' - 'a';
|
|
|
|
|
|
|
|
static int ticks = 0;
|
|
|
|
static cpSpace *space;
|
|
|
|
|
|
|
|
cpVect mousePoint;
|
|
|
|
cpVect mousePoint_last;
|
|
|
|
cpBody *mouseBody = NULL;
|
|
|
|
cpConstraint *mouseJoint = NULL;
|
|
|
|
|
|
|
|
char messageString[1024] = {};
|
|
|
|
|
|
|
|
int key_up = 0;
|
|
|
|
int key_down = 0;
|
|
|
|
int key_left = 0;
|
|
|
|
int key_right = 0;
|
|
|
|
|
|
|
|
cpVect arrowDirection = {};
|
|
|
|
|
|
|
|
drawSpaceOptions options = {
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
1,
|
|
|
|
4.0f,
|
|
|
|
0.0f,
|
|
|
|
1.5f,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
drawString(int x, int y, char *str)
|
|
|
|
{
|
|
|
|
// implement me
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
drawInstructions()
|
|
|
|
{
|
|
|
|
drawString(-300, 220,
|
|
|
|
"Controls:\n"
|
|
|
|
"A - * Switch demos. (return restarts)\n"
|
|
|
|
"Use the mouse to grab objects.\n"
|
|
|
|
"Arrow keys control some demos.\n"
|
|
|
|
"\\ enables anti-aliasing.\n"
|
|
|
|
"- toggles spatial hash visualization.\n"
|
|
|
|
"= toggles bounding boxes."
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int maxArbiters = 0;
|
|
|
|
static int maxPoints = 0;
|
|
|
|
static int maxConstraints = 0;
|
|
|
|
|
|
|
|
static void
|
|
|
|
drawInfo()
|
|
|
|
{
|
|
|
|
int arbiters = space->arbiters->num;
|
|
|
|
int points = 0;
|
|
|
|
|
|
|
|
for(int i=0; i<arbiters; i++)
|
|
|
|
points += ((cpArbiter *)(space->arbiters->arr[i]))->numContacts;
|
|
|
|
|
|
|
|
int constraints = (space->constraints->num + points)*(space->iterations + space->elasticIterations);
|
|
|
|
|
|
|
|
maxArbiters = arbiters > maxArbiters ? arbiters : maxArbiters;
|
|
|
|
maxPoints = points > maxPoints ? points : maxPoints;
|
|
|
|
maxConstraints = constraints > maxConstraints ? constraints : maxConstraints;
|
|
|
|
|
|
|
|
char buffer[1000];
|
|
|
|
char *format =
|
|
|
|
"Arbiters: %d (%d) - "
|
|
|
|
"Contact Points: %d (%d)\n"
|
|
|
|
"Other Constraints: %d, Iterations: %d\n"
|
|
|
|
"Constraints x Iterations: %d (%d)";
|
|
|
|
|
|
|
|
sprintf(buffer, format,
|
|
|
|
arbiters, maxArbiters,
|
|
|
|
points, maxPoints,
|
|
|
|
space->constraints->num, space->iterations + space->elasticIterations,
|
|
|
|
constraints, maxConstraints
|
|
|
|
);
|
|
|
|
|
|
|
|
drawString(0, 220, buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
display(void)
|
|
|
|
{
|
|
|
|
// glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
//
|
|
|
|
// drawSpace(space, currDemo->drawOptions ? currDemo->drawOptions : &options);
|
|
|
|
// drawInstructions();
|
|
|
|
// drawInfo();
|
|
|
|
// drawString(-300, -210, messageString);
|
|
|
|
//
|
|
|
|
// glutSwapBuffers();
|
|
|
|
ticks++;
|
|
|
|
|
|
|
|
cpVect newPoint = cpvlerp(mousePoint_last, mousePoint, 0.25f);
|
|
|
|
mouseBody->p = newPoint;
|
|
|
|
mouseBody->v = cpvmult(cpvsub(newPoint, mousePoint_last), 60.0f);
|
|
|
|
mousePoint_last = newPoint;
|
|
|
|
currDemo->updateFunc(ticks);
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *
|
|
|
|
demoTitle(chipmunkDemo *demo)
|
|
|
|
{
|
|
|
|
static char title[1024];
|
|
|
|
sprintf(title, "Demo: %s", demo->name);
|
|
|
|
|
|
|
|
return title;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
runDemo(chipmunkDemo *demo)
|
|
|
|
{
|
|
|
|
if(currDemo)
|
|
|
|
currDemo->destroyFunc();
|
|
|
|
|
|
|
|
currDemo = demo;
|
|
|
|
ticks = 0;
|
|
|
|
mouseJoint = NULL;
|
|
|
|
messageString[0] = '\0';
|
|
|
|
maxArbiters = 0;
|
|
|
|
maxPoints = 0;
|
|
|
|
maxConstraints = 0;
|
|
|
|
space = currDemo->initFunc();
|
|
|
|
|
|
|
|
// glutSetWindowTitle(demoTitle(currDemo));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
keyboard(unsigned char key, int x, int y)
|
|
|
|
{
|
|
|
|
int index = key - 'a';
|
|
|
|
|
|
|
|
if(0 <= index && index < demoCount){
|
|
|
|
runDemo(demos[index]);
|
|
|
|
} else if(key == '\r'){
|
|
|
|
runDemo(currDemo);
|
|
|
|
} else if(key == '-'){
|
|
|
|
options.drawHash = !options.drawHash;
|
|
|
|
} else if(key == '='){
|
|
|
|
options.drawBBs = !options.drawBBs;
|
|
|
|
} else if(key == '\\'){
|
|
|
|
glEnable(GL_LINE_SMOOTH);
|
|
|
|
glEnable(GL_POINT_SMOOTH);
|
|
|
|
glEnable(GL_BLEND);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
glHint(GL_LINE_SMOOTH_HINT, GL_DONT_CARE);
|
|
|
|
glHint(GL_POINT_SMOOTH_HINT, GL_DONT_CARE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static cpVect
|
|
|
|
mouseToSpace(int x, int y)
|
|
|
|
{
|
|
|
|
// GLdouble model[16];
|
|
|
|
// glGetDoublev(GL_MODELVIEW_MATRIX, model);
|
|
|
|
//
|
|
|
|
// GLdouble proj[16];
|
|
|
|
// glGetDoublev(GL_PROJECTION_MATRIX, proj);
|
|
|
|
//
|
|
|
|
// GLint view[4];
|
|
|
|
// glGetIntegerv(GL_VIEWPORT, view);
|
|
|
|
//
|
|
|
|
// GLdouble mx, my, mz;
|
|
|
|
// gluUnProject(x, glutGet(GLUT_WINDOW_HEIGHT) - y, 0.0f, model, proj, view, &mx, &my, &mz);
|
|
|
|
|
|
|
|
// return cpv(mx, my);
|
|
|
|
return cpv(x,y);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
mouse(int x, int y)
|
|
|
|
{
|
|
|
|
mousePoint = mouseToSpace(x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
click(int button, int state, int x, int y)
|
|
|
|
{
|
|
|
|
// if(button == GLUT_LEFT_BUTTON){
|
|
|
|
// if(state == GLUT_DOWN){
|
|
|
|
// cpVect point = mouseToSpace(x, y);
|
|
|
|
//
|
|
|
|
// cpShape *shape = cpSpacePointQueryFirst(space, point, GRABABLE_MASK_BIT, 0);
|
|
|
|
// if(shape){
|
|
|
|
// cpBody *body = shape->body;
|
|
|
|
// mouseJoint = cpPivotJointNew2(mouseBody, body, cpvzero, cpBodyWorld2Local(body, point));
|
|
|
|
// mouseJoint->maxForce = 50000.0f;
|
|
|
|
// mouseJoint->biasCoef = 0.15f;
|
|
|
|
// cpSpaceAddConstraint(space, mouseJoint);
|
|
|
|
// }
|
|
|
|
// } else if(mouseJoint){
|
|
|
|
// cpSpaceRemoveConstraint(space, mouseJoint);
|
|
|
|
// cpConstraintFree(mouseJoint);
|
|
|
|
// mouseJoint = NULL;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
timercall(int value)
|
|
|
|
{
|
|
|
|
// glutTimerFunc(SLEEP_TICKS, timercall, 0);
|
|
|
|
//
|
|
|
|
// glutPostRedisplay();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_arrowDirection()
|
|
|
|
{
|
|
|
|
int x = 0, y = 0;
|
|
|
|
|
|
|
|
if(key_up) y += 1;
|
|
|
|
if(key_down) y -= 1;
|
|
|
|
if(key_right) x += 1;
|
|
|
|
if(key_left) x -= 1;
|
|
|
|
|
|
|
|
arrowDirection = cpv(x, y);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
arrowKeyDownFunc(int key, int x, int y)
|
|
|
|
{
|
|
|
|
// if(key == GLUT_KEY_UP) key_up = 1;
|
|
|
|
// else if(key == GLUT_KEY_DOWN) key_down = 1;
|
|
|
|
// else if(key == GLUT_KEY_LEFT) key_left = 1;
|
|
|
|
// else if(key == GLUT_KEY_RIGHT) key_right = 1;
|
|
|
|
|
|
|
|
set_arrowDirection();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
arrowKeyUpFunc(int key, int x, int y)
|
|
|
|
{
|
|
|
|
// if(key == GLUT_KEY_UP) key_up = 0;
|
|
|
|
// else if(key == GLUT_KEY_DOWN) key_down = 0;
|
|
|
|
// else if(key == GLUT_KEY_LEFT) key_left = 0;
|
|
|
|
// else if(key == GLUT_KEY_RIGHT) key_right = 0;
|
|
|
|
|
|
|
|
set_arrowDirection();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
idle(void)
|
|
|
|
{
|
|
|
|
// glutPostRedisplay();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
initGL(void)
|
|
|
|
{
|
|
|
|
// glClearColor(1.0f, 1.0f, 1.0f, 0.0f);
|
|
|
|
//
|
|
|
|
// glMatrixMode(GL_PROJECTION);
|
|
|
|
// glLoadIdentity();
|
|
|
|
// glOrtho(-320.0f, 320.0f, -240.0f, 240.0f, -1.0f, 1.0f);
|
|
|
|
// glTranslatef(0.5f, 0.5f, 0.0f);
|
|
|
|
//
|
|
|
|
// glEnableClientState(GL_VERTEX_ARRAY);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
glutStuff(int argc, const char *argv[])
|
|
|
|
{
|
|
|
|
// glutInit(&argc, (char**)argv);
|
|
|
|
//
|
|
|
|
// glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGBA);
|
|
|
|
//
|
|
|
|
// glutInitWindowSize(640, 480);
|
|
|
|
// glutCreateWindow(demoTitle(demos[firstDemoIndex]));
|
|
|
|
//
|
|
|
|
// initGL();
|
|
|
|
//
|
|
|
|
// glutDisplayFunc(display);
|
|
|
|
//// glutIdleFunc(idle);
|
|
|
|
// glutTimerFunc(SLEEP_TICKS, timercall, 0);
|
|
|
|
//
|
|
|
|
// glutIgnoreKeyRepeat(1);
|
|
|
|
// glutKeyboardFunc(keyboard);
|
|
|
|
//
|
|
|
|
// glutSpecialFunc(arrowKeyDownFunc);
|
|
|
|
// glutSpecialUpFunc(arrowKeyUpFunc);
|
|
|
|
//
|
|
|
|
// glutMotionFunc(mouse);
|
|
|
|
// glutPassiveMotionFunc(mouse);
|
|
|
|
// glutMouseFunc(click);
|
|
|
|
}
|
|
|
|
|
|
|
|
//#include <sys/time.h>
|
|
|
|
//void time_trial(char index, int count)
|
|
|
|
//{
|
|
|
|
// currDemo = demos[index];
|
|
|
|
// space = currDemo->initFunc();
|
|
|
|
//
|
|
|
|
// struct timeval start_time, end_time;
|
|
|
|
// gettimeofday(&start_time, NULL);
|
|
|
|
//
|
|
|
|
// for(int i=0; i<count; i++)
|
|
|
|
// currDemo->updateFunc(i);
|
|
|
|
//
|
|
|
|
// gettimeofday(&end_time, NULL);
|
|
|
|
// long millisecs = (end_time.tv_sec - start_time.tv_sec)*1000;
|
|
|
|
// millisecs += (end_time.tv_usec - start_time.tv_usec)/1000;
|
|
|
|
//
|
|
|
|
// currDemo->destroyFunc();
|
|
|
|
//
|
|
|
|
// printf("Time(%c) = %ldms\n", index + 'a', millisecs);
|
|
|
|
//}
|
|
|
|
|
|
|
|
void ChipmunkTestScene::runThisTest()
|
|
|
|
{
|
|
|
|
// create layer
|
|
|
|
ChipmunkTestLayer* pLayer = new ChipmunkTestLayer();
|
|
|
|
pLayer->init();
|
|
|
|
pLayer->autorelease();
|
|
|
|
|
|
|
|
addChild(pLayer);
|
|
|
|
|
|
|
|
CCDirector::getSharedDirector()->replaceScene(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ChipmunkTestLayer::init()
|
|
|
|
{
|
|
|
|
CCLayer::init();
|
|
|
|
CCLayer::setIsTouchEnabled(TRUE);
|
|
|
|
|
|
|
|
demoIndex = 0;
|
|
|
|
|
|
|
|
cpInitChipmunk();
|
|
|
|
mouseBody = cpBodyNew(INFINITY, INFINITY);
|
|
|
|
|
|
|
|
runDemo(demos[firstDemoIndex]);
|
|
|
|
|
|
|
|
label = CCLabel::labelWithString(demos[firstDemoIndex]->name, "Arial", 32);
|
2010-09-06 15:23:00 +08:00
|
|
|
label->setPosition( ccp(0, -300) );
|
2010-09-04 18:18:14 +08:00
|
|
|
label->setColor(ccBLACK);
|
|
|
|
addChild(label);
|
|
|
|
|
|
|
|
// [self schedule: @selector(step:)];
|
|
|
|
schedule( schedule_selector(ChipmunkTestLayer::step));
|
|
|
|
}
|
|
|
|
|
|
|
|
void ChipmunkTestLayer::onEnter()
|
|
|
|
{
|
|
|
|
CCLayer::onEnter();
|
|
|
|
|
|
|
|
glClearColor(1,1,1,1);
|
|
|
|
float factor = 1.0f;
|
|
|
|
|
|
|
|
glMatrixMode(GL_PROJECTION);
|
|
|
|
glLoadIdentity();
|
|
|
|
// portraint
|
|
|
|
glOrthof(-320/factor, 320/factor, -480/factor, 480/factor, -1.0f, 1.0f);
|
|
|
|
// landscape
|
|
|
|
// glOrthof(-480/factor, 480/factor, -320/factor, 320/factor, 1.0f, -1.0f);
|
|
|
|
|
|
|
|
glMatrixMode(GL_MODELVIEW);
|
|
|
|
glLoadIdentity();
|
|
|
|
|
|
|
|
glPointSize(3.0f);
|
|
|
|
glEnable(GL_LINE_SMOOTH);
|
|
|
|
glEnable(GL_POINT_SMOOTH);
|
|
|
|
glEnable(GL_BLEND);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
glHint(GL_LINE_SMOOTH_HINT, GL_DONT_CARE);
|
|
|
|
glHint(GL_POINT_SMOOTH_HINT, GL_DONT_CARE);
|
|
|
|
glLineWidth(1.5f);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ChipmunkTestLayer::step(ccTime dt)
|
|
|
|
{
|
|
|
|
// call chipmunk demo c function
|
|
|
|
display();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ChipmunkTestLayer::draw()
|
|
|
|
{
|
|
|
|
drawSpace(space, currDemo->drawOptions ? currDemo->drawOptions : &options);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ChipmunkTestLayer::ccTouchesEnded(NSSet* touches, UIEvent *event)
|
|
|
|
{
|
|
|
|
demoIndex++;
|
|
|
|
if( demoIndex >= maxDemos )
|
|
|
|
demoIndex = 0;
|
|
|
|
|
|
|
|
runDemo(demos[demoIndex]);
|
|
|
|
|
|
|
|
label->setString( demos[demoIndex]->name );
|
|
|
|
}
|
|
|
|
|