Mercurial > hg > old > magoroku_racing
view col.cc @ 146:d1ddd095ab03
draw dot hit cource -rainbow-
author | e085768 |
---|---|
date | Fri, 29 Jul 2011 15:18:48 +0900 |
parents | be127b7d1812 |
children |
line wrap: on
line source
#include <stdio.h> #include"libps2.h" #include"ps2util.h" #include"col.h" #include"mytype.h" #include <stdbool.h> #include <string.h> #include "gSprite.h" extern Viewer *droot; /** * v0, v1が描く三角形と点pの内外判定(approx detection) * 点pはv0, v1が描く面のどこかに位置している。 * pが面の三角形の外側にあるときTRUEを戻し、 * そうでなければFALSEを戻す。境界は含む。 */ bool col_detect_approx(FVECTOR p, FVECTOR v0, FVECTOR v1) { float v_r1, v_r2, r1_r2, r1_r1, r2_r2; float a, b, r; v_r1 = ps2_vu0_inner_product( p, v0 ); v_r2 = ps2_vu0_inner_product( p, v1 ); r1_r2 = ps2_vu0_inner_product( v0, v1 ); r1_r1 = ps2_vu0_inner_product( v0, v0 ); r2_r2 = ps2_vu0_inner_product( v1, v1 ); a = (v_r1 * r2_r2 - v_r2 * r1_r2); b = (v_r2 * r1_r1 - v_r1 * r1_r2); r = (r1_r1 * r2_r2 - r1_r2 * r1_r2); return (a>=0 && b>=0 && a+b-r<=0) ? true : false; } /** * 垂心oの算出. 点pからv0, v1が描く面へ垂線を引いたときの交点oを * 垂心(orthocenter)と呼ぶ。 */ static void col_orthocenter(FVECTOR o, FVECTOR p, FVECTOR p0, FVECTOR normal) { FVECTOR o0, d; ps2_vu0_sub_vector(o0, p, p0); o0[W_AXIS] = 0; ps2_vu0_scale_vector( d, normal, ps2_vu0_inner_product(o0, normal)); o[X_AXIS] = p[X_AXIS] - d[X_AXIS]; o[Y_AXIS] = p[Y_AXIS] - d[Y_AXIS]; o[Z_AXIS] = p[Z_AXIS] - d[Z_AXIS]; o[W_AXIS] = 1; } /** * 点pと面との衝突(点pとdetection face面との垂心oがdetection face領域の * 中にあるかを)判定衝突時(oがdetection face領域の外)はFALSEを戻し、そう * でなければTRUE、 垂心(orthocenter)、垂心のある面の法線(normal * vector)を戻す. */ bool col_detect(ColFaceHd* hd, FVECTOR pose, FVECTOR p) { FVECTOR o, o0; int i; ColFacePtr col_face=hd->colface; for (i=0; i<hd->facenum; i++) { // * 垂心oを求める col_orthocenter(o, p, col_face[i].p0, col_face[i].normal); // * detection face上のある点p0から垂心oへのベクトルo0を求める ps2_vu0_sub_vector(o0, o, col_face[i].p0); // * ベクトルv0, v1の描く三角形とo0とで内外判定(range approx detection)を行う if (col_detect_approx(o0, col_face[i].v0, col_face[i].v1)==true) { goto NOCOLLISION; } } return false; NOCOLLISION: // "壁に"ぶつかっていない = course上にいる printf("---on hd->facenum = %d/%d ----- ",i, hd->facenum); printf("P x,y,z = %f,%f,%f ",col_face[i].p0[0],col_face[i].p0[1],col_face[i].p0[2]); printf("N x,y,z = %f,%f,%f\n",col_face[i].normal[0],col_face[i].normal[1],col_face[i].normal[2]); ps2_vu0_copy_vector(p, o); ps2_vu0_copy_vector(pose, col_face[i].normal); return true; } /* * detection face作成 * Collision XML から Polygon を読み込んで、 * Normal Vector(Polygon に直交した Vector) を生成する。 * face の中に、既に Normal Vector が計算されているので、 * それを利用してもよい。 */ void col_init(ColFaceHd* hd, char *colImg) { static int col_num = 0; col_num ++; // xmlから読み込み call_createFromXMLfile(colImg); char str[100]; sprintf(str,"hoge%d",col_num); printf("%s\n",str); printf("%d\n",col_num); // SceneGraphPtr face = droot->createSceneGraph("hoge"); SceneGraphPtr face = droot->createSceneGraph(str); int i; FVECTOR p1, p2; ColFacePtr colface; hd->facenum = face->size/3; malloc_align16(&hd->free_addr, &colface, sizeof(ColFace)*hd->facenum); for (i=0; i < hd->facenum; i++) { memcpy(colface[i].p0, &face->pp->tri[i].ver1, sizeof(FVECTOR)); memcpy(p1, &face->pp->tri[i].ver3, sizeof(FVECTOR)); memcpy(p2, &face->pp->tri[i].ver2, sizeof(FVECTOR)); ps2_vu0_sub_vector(colface[i].v0, p1, colface[i].p0); ps2_vu0_sub_vector(colface[i].v1, p2, colface[i].p0); ps2_vu0_outer_product(colface[i].normal, colface[i].v0, colface[i].v1); ps2_vu0_normalize(colface[i].normal, colface[i].normal); if (colface[i].normal[1] > 0) { colface[i].normal[0] = -colface[i].normal[0]; colface[i].normal[1] = -colface[i].normal[1]; colface[i].normal[2] = -colface[i].normal[2]; } } hd->colface = colface; }