diff examples/tank/tank_main.c @ 131:6a8588df68af

Tank can change direction and navigate on the mud area
author Thinker K.F. Li <thinker@branda.to>
date Wed, 17 Sep 2008 01:30:25 +0800
parents 3a4d6179e6a9
children c65b30e2eda9
line wrap: on
line diff
--- a/examples/tank/tank_main.c	Tue Sep 16 21:23:34 2008 +0800
+++ b/examples/tank/tank_main.c	Wed Sep 17 01:30:25 2008 +0800
@@ -1,5 +1,7 @@
 #include <sys/time.h>
+#include <string.h>
 #include <mb/mb.h>
+#include <mb/tools.h>
 #include "svgs.h"
 
 enum { MUD, ROC, BRI, BSH };
@@ -31,18 +33,183 @@
       MUD, BRI, MUD, BRI, MUD, MUD, MUD, MUD}
 };
 
+#define MAP_W 16
+#define MAP_H 12
+
+/*! \defgroup tank_elf Tank Elf
+ * \brief Tank elf module provides control functions of tanks in game.
+ * @{
+ */
+struct _tank {
+    coord_t *coord_pos;		/*!< \brief coordinate for position */
+    coord_t *coord_rot;		/*!< \brief coordinate for rotation */
+    int map_x, map_y;
+    int direction;
+    mb_progm_t *progm;
+};
+typedef struct _tank tank_t;
+enum { TD_UP = 0, TD_RIGHT, TD_DOWN, TD_LEFT };
+
+static tank_t *tank_new(coord_t *coord_pos,
+			coord_t *coord_rot,
+			int map_x, int map_y,
+			X_MB_runtime_t *mb_rt) {
+    tank_t *tank;
+    redraw_man_t *rdman;
+
+    tank = O_ALLOC(tank_t);
+    if(tank == NULL)
+	return NULL;
+
+    rdman = X_MB_rdman(mb_rt);
+
+    tank->coord_pos = coord_pos;
+    tank->coord_rot = coord_rot;
+    tank->map_x = map_x;
+    tank->map_y = map_y;
+    tank->direction = TD_UP;
+    tank->progm = NULL;
+
+    memset(coord_pos->matrix, 0, sizeof(co_aix[6]));
+    coord_pos->matrix[0] = 1;
+    coord_pos->matrix[2] = map_x * 50;
+    coord_pos->matrix[4] = 1;
+    coord_pos->matrix[5] = map_y * 50;
+    rdman_coord_changed(rdman, coord_pos);
+
+    return tank;
+}
+
+static void tank_free(tank_t *tank, X_MB_runtime_t *xmb_rt) {
+    mb_tman_t *tman;
+
+    if(tank->progm) {
+	tman = X_MB_tman(xmb_rt);
+	mb_progm_abort(tank->progm, tman);
+    }
+    free(tank);
+}
+
+/*! \brief Clean program for a tank.
+ *
+ * It is called when the program is completed.
+ */
+static void clean_tank_progm_handler(event_t *event, void *arg) {
+    tank_t *tank = (tank_t *)arg;
+
+    mb_progm_free(tank->progm);
+    tank->progm = NULL;
+}
+
+#define PI 3.1415926
+
+static void tank_move(tank_t *tank, int direction,
+		    X_MB_runtime_t *xmb_rt) {
+    redraw_man_t *rdman;
+    mb_tman_t *tman;
+    ob_factory_t *factory;
+    /* for the program */
+    mb_progm_t *progm;
+    subject_t *comp_sub;
+    mb_word_t *word;
+    mb_timeval_t start, playing;
+    mb_timeval_t now;
+    /* for position */
+    co_aix sh_x, sh_y;
+    /* for direction */
+    float ang1, ang2;
+    float rot_diff;
+    static co_aix shift_xy[][2] = {{0, -50}, {50, 0}, {0, 50}, {-50, 0}};
+    static int map_shift[4][2] = {{0, -1}, {1, 0}, {0, 1}, {-1, 0}};
+    static float angles[4] = {0, PI / 2, PI , PI * 3 / 2};
+    static float rotations[7] = {PI / 2, PI , -PI / 2,
+				 0, PI / 2, PI , -PI / 2};
+
+    if(tank->progm != NULL)
+	return;
+
+    /*
+     * Keep inside the map.
+     */
+    if(direction == tank->direction) {
+	switch(direction) {
+	case TD_UP:
+	    if(tank->map_y == 0)
+		return;
+	    if(map[tank->map_y - 1][tank->map_x] != MUD)
+		return;
+	    break;
+	case TD_RIGHT:
+	    if(tank->map_x >= (MAP_W - 1))
+		return;
+	    if(map[tank->map_y][tank->map_x + 1] != MUD)
+		return;
+	    break;
+	case TD_DOWN:
+	    if(tank->map_y >= (MAP_H - 1))
+		return;
+	    if(map[tank->map_y + 1][tank->map_x] != MUD)
+		return;
+	    break;
+	case TD_LEFT:
+	    if(tank->map_x == 0)
+		return;
+	    if(map[tank->map_y][tank->map_x - 1] != MUD)
+		return;
+	    break;
+	}
+    }
+
+    rdman = X_MB_rdman(xmb_rt);
+    tman = X_MB_tman(xmb_rt);
+    factory = X_MB_ob_factory(xmb_rt);
+
+    progm = mb_progm_new(1, rdman);
+    tank->progm = progm;
+
+    MB_TIMEVAL_SET(&start, 0, 0);
+    MB_TIMEVAL_SET(&playing, 0, 500000);
+    word = mb_progm_next_word(progm, &start, &playing);
+
+    if(direction == tank->direction) {
+	/* Shift/move */
+	sh_x = shift_xy[direction][0];
+	sh_y = shift_xy[direction][1];
+	mb_shift_new(sh_x, sh_y, tank->coord_pos, word);
+	tank->map_x += map_shift[direction][0];
+	tank->map_y += map_shift[direction][1];
+    } else {
+	/* Change direction */
+	rot_diff = rotations[3 - tank->direction + direction];
+	ang1 = angles[tank->direction];
+	ang2 = ang1 + rot_diff;
+	mb_rotate_new(ang1, ang2, tank->coord_rot, word);
+	tank->direction = direction;
+    }
+
+    /* Clean program when it is completed. */
+    comp_sub = mb_progm_get_complete(progm);
+    subject_add_observer(factory, comp_sub,
+			 clean_tank_progm_handler, tank);
+
+    get_now(&now);
+    mb_progm_start(progm, tman, &now);
+}
+
+/* @} */
+
 typedef struct _tank_rt tank_rt_t;
 
 struct _tank_rt {
-    tank1_t *tank1;
-    tank2_t *tank2;
+    tank_t *tank1;
+    tank1_t *tank1_o;
+    tank_t *tank2;
+    tank2_t *tank2_o;
     int n_enemy;
-    tank_en_t *tank_enemies[10];
+    tank_t *tank_enemies[10];
+    tank_en_t *tank_enemies_o[10];
     void *map[12][16];
     X_MB_runtime_t *mb_rt;
-
-    mb_progm_t *tank1_progm;
-
     observer_t *kb_observer;
 };
 
@@ -54,64 +221,38 @@
 	rdman_coord_changed(rdman, (g)->root_coord);	\
     } while(0)
 
-static void free_progm_handler(event_t *event, void *arg) {
-    tank_rt_t *tank_rt = (tank_rt_t *)arg;
-
-    mb_progm_free(tank_rt->tank1_progm);
-    tank_rt->tank1_progm = NULL;
-}
-
 static void keyboard_handler(event_t *event, void *arg) {
     X_kb_event_t *xkey = (X_kb_event_t *)event;
     tank_rt_t *tank_rt = (tank_rt_t *)arg;
-    redraw_man_t *rdman;
-    mb_tman_t *tman;
-    mb_word_t *word;
-    ob_factory_t *factory;
-    subject_t *comp_sub;
-    mb_timeval_t start_tm, playing, now;
-
-    if(tank_rt->tank1_progm != NULL)
-	return;
+    int direction;
 
-    rdman = X_MB_rdman(tank_rt->mb_rt);
-    tman = X_MB_tman(tank_rt->mb_rt);
-
-    tank_rt->tank1_progm = mb_progm_new(2, rdman);
-    comp_sub = mb_progm_get_complete(tank_rt->tank1_progm);
-    factory = rdman_get_ob_factory(rdman);
-    subject_add_observer(factory, comp_sub,
-			 free_progm_handler,
-			 tank_rt);
-
-    MB_TIMEVAL_SET(&start_tm, 0, 0);
-    MB_TIMEVAL_SET(&playing, 0, 500000);
-    word = mb_progm_next_word(tank_rt->tank1_progm,
-			      &start_tm, &playing);
+    if(xkey->event.type != EVT_KB_PRESS)
+	return;
 
     switch(xkey->sym) {
     case 0xff51:		/* left */
-	mb_shift_new(-50, 0, tank_rt->tank1->root_coord, word);
+	direction = TD_LEFT;
 	break;
 
     case 0xff52:		/* up */
-	mb_shift_new(0, -50, tank_rt->tank1->root_coord, word);
+	direction = TD_UP;
 	break;
 
     case 0xff53:		/* right */
-	mb_shift_new(50, 0, tank_rt->tank1->root_coord, word);
+	direction = TD_RIGHT;
 	break;
 
     case 0xff54:		/* down */
-	mb_shift_new(0, 50, tank_rt->tank1->root_coord, word);
+	direction = TD_DOWN;
 	break;
 
     case 0x20:			/* space */
     case 0xff0d:		/* enter */
-	break;
+    default:
+	return;
     }
-    get_now(&now);
-    mb_progm_start(tank_rt->tank1_progm, tman, &now);
+
+    tank_move(tank_rt->tank1, direction, tank_rt->mb_rt);
 }
 
 static void init_keyboard(tank_rt_t *tank_rt) {
@@ -130,19 +271,39 @@
 	subject_add_observer(factory, kbevents, keyboard_handler, tank_rt);
 }
 
+/*! \brief Make coord objects for elfs (tanks). */
+static void make_elf_coords(redraw_man_t *rdman, coord_t **coord_pos,
+			    coord_t **coord_rot, coord_t **coord_center) {
+    coord_t *coord_back;
+
+    *coord_pos = rdman_coord_new(rdman, rdman->root_coord);
+
+    coord_back = rdman_coord_new(rdman, *coord_pos);
+    coord_back->matrix[2] = 25;
+    coord_back->matrix[5] = 25;
+    rdman_coord_changed(rdman, coord_back);
+
+    *coord_rot = rdman_coord_new(rdman, coord_back);
+
+    *coord_center = rdman_coord_new(rdman, *coord_rot);
+    (*coord_center)->matrix[2] = -25;
+    (*coord_center)->matrix[5] = -25;
+    rdman_coord_changed(rdman, *coord_center);
+}
+
 void
 initial_tank(tank_rt_t *tank_rt, X_MB_runtime_t *mb_rt) {
     redraw_man_t *rdman;
-    mb_tman_t *tman;
+    /* for map areas */
     mud_t *mud;
     brick_t *brick;
     rock_t *rock;
     bush_t *bush;
-    mb_word_t *word;
-    mb_timeval_t start, playing;
-    mb_timeval_t mbtv;
-    subject_t *comp_sub;
-    ob_factory_t *factory;
+    /* for tanks */
+    coord_t *coord_center, *coord_pos, *coord_rot;
+    tank1_t *tank1_o;
+    tank2_t *tank2_o;
+    tank_en_t *tank_en_o;
     int i, j;
 
     rdman = X_MB_rdman(mb_rt);
@@ -175,41 +336,25 @@
 	}
     }
 
-    tank_rt->tank1 = tank1_new(rdman, rdman->root_coord);
-    CHANGE_POS(tank_rt->tank1, 5 * 50, 11 * 50);
-    tank_rt->tank2 = tank2_new(rdman, rdman->root_coord);
-    CHANGE_POS(tank_rt->tank2, 10 * 50, 11 * 50);
+    make_elf_coords(rdman, &coord_pos, &coord_rot, &coord_center);
+    tank1_o = tank1_new(rdman, coord_center);
+    tank_rt->tank1 = tank_new(coord_pos, coord_rot, 5, 11, mb_rt);
+    tank_rt->tank1_o = tank1_o;
+
+    make_elf_coords(rdman, &coord_pos, &coord_rot, &coord_center);
+    tank2_o = tank2_new(rdman, coord_center);
+    tank_rt->tank2 = tank_new(coord_pos, coord_rot, 10, 11, mb_rt);
+    tank_rt->tank2_o = tank2_o;
+
     for(i = 0; i < 3; i++) {
-	tank_rt->tank_enemies[i] = tank_en_new(rdman, rdman->root_coord);
-	CHANGE_POS(tank_rt->tank_enemies[i], (2 + i * 3) * 50, 0);
+	make_elf_coords(rdman, &coord_pos, &coord_rot, &coord_center);
+	tank_en_o = tank_en_new(rdman, coord_center);
+	tank_rt->tank_enemies[i] = tank_new(coord_pos, coord_rot,
+					    i * 3 + 3, 0, mb_rt);
+	tank_rt->tank_enemies_o[i] = tank_en_o;
     }
     tank_rt->n_enemy = i;
 
-    tank_rt->tank1_progm = mb_progm_new(4, rdman);
-
-    MB_TIMEVAL_SET(&start, 1, 0);
-    MB_TIMEVAL_SET(&playing, 3, 0);
-    word = mb_progm_next_word(tank_rt->tank1_progm, &start, &playing);
-
-    mb_shift_new(0, -150, tank_rt->tank1->root_coord, word);
-    mb_shift_new(0, -150, tank_rt->tank2->root_coord, word);
-
-    MB_TIMEVAL_SET(&start, 5, 0);
-    MB_TIMEVAL_SET(&playing, 3, 0);
-    word = mb_progm_next_word(tank_rt->tank1_progm, &start, &playing);
-
-    mb_shift_new(0, 150, tank_rt->tank1->root_coord, word);
-    mb_shift_new(0, 150, tank_rt->tank2->root_coord, word);
-
-    /* Free program after program completed. */
-    comp_sub = mb_progm_get_complete(tank_rt->tank1_progm);
-    factory = rdman_get_ob_factory(rdman);
-    subject_add_observer(factory, comp_sub, free_progm_handler, tank_rt);
-
-    tman = X_MB_tman(mb_rt);
-    get_now(&mbtv);
-    mb_progm_start(tank_rt->tank1_progm, tman, &mbtv);
-
     init_keyboard(tank_rt);
 }