|
@@ -0,0 +1,425 @@
|
|
|
+#include <string.h>
|
|
|
+#include <rtthread.h>
|
|
|
+
|
|
|
+#include <rtgui/rtgui.h>
|
|
|
+#include <rtgui/rtgui_app.h>
|
|
|
+#include <rtgui/widgets/container.h>
|
|
|
+#include <rtgui/widgets/window.h>
|
|
|
+#include <rtgui/widgets/button.h>
|
|
|
+
|
|
|
+#include "snake.h"
|
|
|
+
|
|
|
+#define LATTICE_SIZE (20)
|
|
|
+#define FOOD_MAX (8)
|
|
|
+
|
|
|
+#define WALL_COLOR RTGUI_RGB(255, 0, 0)
|
|
|
+#define SNAKE_COLOR RTGUI_RGB(111, 0, 255)
|
|
|
+#define BACKGROUND_COLOR RTGUI_RGB(153, 153, 0)
|
|
|
+#define FOOD_COLOR RTGUI_RGB(0, 111, 111)
|
|
|
+#define min(a, b) ((a) < (b) ? (a) : (b))
|
|
|
+
|
|
|
+static rtgui_timer_t *timer;
|
|
|
+static rt_size_t room_size_x, room_size_y;
|
|
|
+static rt_size_t lattice_size_x, lattice_size_y;
|
|
|
+static struct rtgui_rect room_rect, lattice_rect;
|
|
|
+
|
|
|
+map_t* map;
|
|
|
+SNAKE_DIR run_state;
|
|
|
+rt_int32_t snake_len;
|
|
|
+rt_int32_t food_num;
|
|
|
+
|
|
|
+static void snake_fill_lattice(struct rtgui_dc *dc,
|
|
|
+ rt_uint32_t x,
|
|
|
+ rt_uint32_t y,
|
|
|
+ rtgui_color_t color)
|
|
|
+{
|
|
|
+ struct rtgui_rect rect;
|
|
|
+
|
|
|
+ // coordinate conversion
|
|
|
+ y = (lattice_size_y - 1) - y;
|
|
|
+
|
|
|
+ RTGUI_DC_BC(dc) = color;
|
|
|
+
|
|
|
+ rect.x1 = lattice_rect.x1 + (LATTICE_SIZE * x);
|
|
|
+ rect.x2 = rect.x1 + LATTICE_SIZE;
|
|
|
+ rect.x1 += 2;
|
|
|
+
|
|
|
+ rect.y1 = lattice_rect.y1 + (LATTICE_SIZE * y);
|
|
|
+ rect.y2 = rect.y1 + LATTICE_SIZE;
|
|
|
+ rect.y1 += 2;
|
|
|
+
|
|
|
+ rtgui_dc_fill_rect(dc, &rect);
|
|
|
+}
|
|
|
+
|
|
|
+static void snake_draw(struct rtgui_widget *widget)
|
|
|
+{
|
|
|
+ struct rtgui_dc *dc;
|
|
|
+ struct rtgui_rect rect;
|
|
|
+ rt_uint32_t i;
|
|
|
+
|
|
|
+// rt_kprintf("snake_draw\r\n");
|
|
|
+
|
|
|
+ dc = rtgui_dc_begin_drawing(widget);
|
|
|
+ if (dc == RT_NULL)
|
|
|
+ {
|
|
|
+ rt_kprintf("dc == RT_NULL\r\n");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* get room size, run once frist. */
|
|
|
+ if((room_size_x == 0) || (room_size_y == 0))
|
|
|
+ {
|
|
|
+ rt_size_t tmp;
|
|
|
+
|
|
|
+ rtgui_widget_get_rect(widget, &rect);
|
|
|
+ rt_kprintf("rect => x1:%d x2:%d, y1:%d y2:%d\r\n", rect.x1, rect.x2, rect.y1, rect.y2);
|
|
|
+
|
|
|
+ room_size_x = rect.x2 - rect.x1;
|
|
|
+ room_size_y = rect.y2 - rect.y1;
|
|
|
+ memcpy(&room_rect, &rect, sizeof(struct rtgui_rect));
|
|
|
+ rt_kprintf("room_rect => x1:%d x2:%d, y1:%d y2:%d\r\n",
|
|
|
+ room_rect.x1, room_rect.x2,
|
|
|
+ room_rect.y1, room_rect.y2);
|
|
|
+
|
|
|
+ lattice_size_x = (room_rect.x2 - room_rect.x1) / LATTICE_SIZE;
|
|
|
+ lattice_size_y = (room_rect.y2 - room_rect.y1) / LATTICE_SIZE;
|
|
|
+ lattice_size_x -= 2;
|
|
|
+ lattice_size_y -= 2;
|
|
|
+ rt_kprintf("lattice_size_x:%d lattice_size_y:%d\r\n",
|
|
|
+ lattice_size_x,
|
|
|
+ lattice_size_y);
|
|
|
+
|
|
|
+ tmp = (room_rect.x2 - room_rect.x1) - (LATTICE_SIZE * lattice_size_x);
|
|
|
+ lattice_rect.x1 = room_rect.x1 + (tmp / 2);
|
|
|
+ lattice_rect.x2 = lattice_rect.x1 + (LATTICE_SIZE * lattice_size_x);
|
|
|
+
|
|
|
+ tmp = (room_rect.y2 - room_rect.y1) - (LATTICE_SIZE * lattice_size_y);
|
|
|
+ lattice_rect.y1 = room_rect.y1 + (tmp / 2);
|
|
|
+ lattice_rect.y2 = lattice_rect.y1 + (LATTICE_SIZE * lattice_size_y);
|
|
|
+ rt_kprintf("lattice_rect => x1:%d x2:%d, y1:%d y2:%d\r\n",
|
|
|
+ lattice_rect.x1, lattice_rect.x2,
|
|
|
+ lattice_rect.y1, lattice_rect.y2);
|
|
|
+
|
|
|
+ /* create snake. */
|
|
|
+ {
|
|
|
+ point_t start;
|
|
|
+ map = map_init(lattice_size_x, lattice_size_y);
|
|
|
+ if (map != RT_NULL)
|
|
|
+ {
|
|
|
+ start.x = snake_init_pointx;
|
|
|
+ start.y = snake_init_pointy;
|
|
|
+ run_state = SNAKE_DIR_DOWN;
|
|
|
+
|
|
|
+ if (snake_init(&start, snake_length_init, run_state, map))
|
|
|
+ {
|
|
|
+ food_num = 1;
|
|
|
+ food_init(map, food_num);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ map_deinit(map);
|
|
|
+ map = RT_NULL;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ RTGUI_DC_BC(dc) = BACKGROUND_COLOR;
|
|
|
+ rtgui_dc_fill_rect(dc, &room_rect);
|
|
|
+
|
|
|
+ memcpy(&rect, &lattice_rect, sizeof(struct rtgui_rect));
|
|
|
+ rect.x2 += 1;
|
|
|
+ rect.y2 += 1;
|
|
|
+ RTGUI_DC_FC(dc) = WALL_COLOR;
|
|
|
+ rtgui_dc_draw_rect(dc, &rect);
|
|
|
+
|
|
|
+ for(i=1; i<lattice_size_y; i++)
|
|
|
+ {
|
|
|
+// rtgui_dc_draw_horizontal_line(struct rtgui_dc* dc, int x1, int x2, int y);
|
|
|
+ memcpy(&rect, &lattice_rect, sizeof(struct rtgui_rect));
|
|
|
+ rect.x1 += 1;
|
|
|
+ rect.x2 -= 1;
|
|
|
+ rtgui_dc_draw_horizontal_line(dc, rect.x1, rect.x2,
|
|
|
+ rect.y1 + (LATTICE_SIZE * i));
|
|
|
+ }
|
|
|
+
|
|
|
+ for(i=1; i<lattice_size_x; i++)
|
|
|
+ {
|
|
|
+// rtgui_dc_draw_vertical_line(struct rtgui_dc* dc, int x, int y1, int y2);
|
|
|
+ memcpy(&rect, &lattice_rect, sizeof(struct rtgui_rect));
|
|
|
+ rect.y1 += 1;
|
|
|
+ rect.y2 -= 1;
|
|
|
+ rtgui_dc_draw_vertical_line(dc, rect.x1 + (LATTICE_SIZE * i),
|
|
|
+ rect.y1, rect.y2);
|
|
|
+ }
|
|
|
+
|
|
|
+ /* draw snake. */
|
|
|
+ {
|
|
|
+ rt_uint32_t x, y;
|
|
|
+ for (y=0; y<map->height; y++)
|
|
|
+ {
|
|
|
+ for (x=0; x<map->width; x++)
|
|
|
+ {
|
|
|
+ switch (map->range[y * map->width + x])
|
|
|
+ {
|
|
|
+ case NORMAL:
|
|
|
+ break;
|
|
|
+ case FOOD:
|
|
|
+ snake_fill_lattice(dc, x, y, FOOD_COLOR);
|
|
|
+ break;
|
|
|
+ case OVER:
|
|
|
+ snake_fill_lattice(dc, x, y, SNAKE_COLOR);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ rtgui_dc_end_drawing(dc);
|
|
|
+
|
|
|
+ return;
|
|
|
+}
|
|
|
+
|
|
|
+static void snake_update(struct rtgui_widget *widget)
|
|
|
+{
|
|
|
+ struct rtgui_dc *dc;
|
|
|
+ rt_int32_t x, y;
|
|
|
+ rt_uint32_t i;
|
|
|
+
|
|
|
+// rt_kprintf("snake_update\r\n");
|
|
|
+
|
|
|
+ dc = rtgui_dc_begin_drawing(widget);
|
|
|
+ if (dc == RT_NULL)
|
|
|
+ {
|
|
|
+ rt_kprintf("dc == RT_NULL\r\n");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ for(i=0; i<3; i++)
|
|
|
+ {
|
|
|
+ if(i < 2)
|
|
|
+ {
|
|
|
+ x = map->snake_flush[i].x;
|
|
|
+ y = map->snake_flush[i].y;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ x = map->food_flush[0].x;
|
|
|
+ y = map->food_flush[0].y;
|
|
|
+ }
|
|
|
+
|
|
|
+ if((x >= 0) && (y >= 0))
|
|
|
+ {
|
|
|
+// rt_kprintf("snake_flush[%d].x:%d, snake_flush[%d].y:%d\r\n",
|
|
|
+// i, x,
|
|
|
+// i, y);
|
|
|
+
|
|
|
+ switch (map->range[(map->width * y) + x])
|
|
|
+ {
|
|
|
+ case NORMAL:
|
|
|
+ snake_fill_lattice(dc, x, y, BACKGROUND_COLOR);
|
|
|
+ break;
|
|
|
+ case FOOD:
|
|
|
+ snake_fill_lattice(dc, x, y, FOOD_COLOR);
|
|
|
+ break;
|
|
|
+ case OVER:
|
|
|
+ snake_fill_lattice(dc, x, y, SNAKE_COLOR);
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+// rt_kprintf("\r\n");
|
|
|
+
|
|
|
+
|
|
|
+// if((map->snake_flush[1].x >= 0) && (map->snake_flush[1].y >= 0))
|
|
|
+// {
|
|
|
+// rt_kprintf("snake_flush[1].x:%d, snake_flush[1].y:%d\r\n",
|
|
|
+// map->snake_flush[1].x, map->snake_flush[1].y);
|
|
|
+// snake_fill_lattice(dc,
|
|
|
+// map->snake_flush[1].x,
|
|
|
+// map->snake_flush[1].y,
|
|
|
+// SNAKE_COLOR);
|
|
|
+// }
|
|
|
+
|
|
|
+// x = map->food_flush[0].x;
|
|
|
+// y = map->food_flush[0].y;
|
|
|
+// if((map->food_flush[0].x >= 0) && (map->food_flush[0].y >= 0))
|
|
|
+// {
|
|
|
+// rt_kprintf("food_flush[0].x:%d, food_flush[0].y:%d\r\n",
|
|
|
+// map->food_flush[0].x, map->food_flush[0].y);
|
|
|
+// snake_fill_lattice(dc,
|
|
|
+// map->food_flush[0].x,
|
|
|
+// map->food_flush[0].y,
|
|
|
+// FOOD_COLOR);
|
|
|
+// }
|
|
|
+
|
|
|
+ rtgui_dc_end_drawing(dc);
|
|
|
+ return;
|
|
|
+}
|
|
|
+
|
|
|
+static void snake_handler(struct rtgui_widget *widget, rtgui_event_t *event)
|
|
|
+{
|
|
|
+ struct rtgui_event_kbd* ekbd;
|
|
|
+
|
|
|
+ ekbd = (struct rtgui_event_kbd*) event;
|
|
|
+ if (ekbd->type == RTGUI_KEYDOWN)
|
|
|
+ {
|
|
|
+ switch (ekbd->key)
|
|
|
+ {
|
|
|
+ case RTGUIK_UP:
|
|
|
+ rt_kprintf("RTGUIK_UP\r\n");
|
|
|
+ //snake_key(1);
|
|
|
+ //snake_step(map, UP);
|
|
|
+ run_state = SNAKE_DIR_UP;
|
|
|
+ break;
|
|
|
+ case RTGUIK_DOWN:
|
|
|
+ rt_kprintf("RTGUIK_DOWN\r\n");
|
|
|
+ //snake_key(2);
|
|
|
+ //snake_step(map, DOWN);
|
|
|
+ run_state = SNAKE_DIR_DOWN;
|
|
|
+ break;
|
|
|
+ case RTGUIK_LEFT:
|
|
|
+ rt_kprintf("RTGUIK_LEFT\r\n");
|
|
|
+ //snake_key(3);
|
|
|
+ //snake_step(map, LEFT);
|
|
|
+ run_state = SNAKE_DIR_LEFT;
|
|
|
+ break;
|
|
|
+ case RTGUIK_RIGHT:
|
|
|
+ rt_kprintf("RTGUIK_RIGHT\r\n");
|
|
|
+ //snake_key(4);
|
|
|
+ //snake_step(map, RIGHT);
|
|
|
+ run_state = SNAKE_DIR_RIGHT;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static rt_bool_t event_handler(struct rtgui_object *object, rtgui_event_t *event)
|
|
|
+{
|
|
|
+ struct rtgui_widget *widget = RTGUI_WIDGET(object);
|
|
|
+
|
|
|
+ rt_kprintf("event_handler\r\n");
|
|
|
+
|
|
|
+ if (event->type == RTGUI_EVENT_PAINT)
|
|
|
+ {
|
|
|
+ rt_kprintf("RTGUI_EVENT_PAINT\r\n");
|
|
|
+// rtgui_container_event_handler(object, event);
|
|
|
+ rtgui_win_event_handler((struct rtgui_object*)object, event);
|
|
|
+ snake_draw(widget);
|
|
|
+ rtgui_timer_start(timer);
|
|
|
+ }
|
|
|
+ else if (event->type == RTGUI_EVENT_SHOW)
|
|
|
+ {
|
|
|
+ rt_kprintf("RTGUI_EVENT_SHOW\r\n");
|
|
|
+// rtgui_container_event_handler(object, event);
|
|
|
+ rtgui_win_event_handler((struct rtgui_object*)object, event);
|
|
|
+ snake_draw(widget);
|
|
|
+ rtgui_timer_start(timer);
|
|
|
+ }
|
|
|
+ else if (event->type == RTGUI_EVENT_HIDE)
|
|
|
+ {
|
|
|
+ rt_kprintf("RTGUI_EVENT_HIDE\r\n");
|
|
|
+// rtgui_container_event_handler(object, event);
|
|
|
+ rtgui_win_event_handler((struct rtgui_object*)object, event);
|
|
|
+ rtgui_timer_stop(timer);
|
|
|
+ }
|
|
|
+ else if (event->type == RTGUI_EVENT_WIN_DEACTIVATE)
|
|
|
+ {
|
|
|
+ rt_kprintf("RTGUI_EVENT_WIN_DEACTIVATE\r\n");
|
|
|
+// rtgui_container_event_handler(object, event);
|
|
|
+ rtgui_win_event_handler((struct rtgui_object*)object, event);
|
|
|
+ rtgui_timer_stop(timer);
|
|
|
+ }
|
|
|
+ else if (event->type == RTGUI_EVENT_KBD)
|
|
|
+ {
|
|
|
+// rt_kprintf("RTGUI_EVENT_KBD\r\n");
|
|
|
+// rtgui_container_event_handler(object, event);
|
|
|
+ rtgui_win_event_handler((struct rtgui_object*)object, event);
|
|
|
+ snake_handler(widget, event);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ rt_kprintf("event->type:%d\r\n", event->type);
|
|
|
+// return rtgui_container_event_handler(object, event);
|
|
|
+ return rtgui_win_event_handler((struct rtgui_object*)object, event);
|
|
|
+ }
|
|
|
+
|
|
|
+ return RT_FALSE;
|
|
|
+}
|
|
|
+
|
|
|
+static void timeout(struct rtgui_timer *timer, void *parameter)
|
|
|
+{
|
|
|
+ struct rtgui_widget *widget;
|
|
|
+
|
|
|
+ if (!map)
|
|
|
+ return;
|
|
|
+
|
|
|
+ if (snake_step(run_state, map) == FOOD)
|
|
|
+ {
|
|
|
+ // food--;
|
|
|
+ snake_len++;
|
|
|
+ if (snake_len >= (map->width * map->height) / 3)
|
|
|
+ {
|
|
|
+ point_t start;
|
|
|
+
|
|
|
+ start.x = snake_init_pointx;
|
|
|
+ start.y = snake_init_pointy;
|
|
|
+ run_state = SNAKE_DIR_DOWN;
|
|
|
+ snake_len = snake_length_init;
|
|
|
+
|
|
|
+ if (!snake_restart(&start, snake_len, run_state, map))
|
|
|
+ {
|
|
|
+ map_deinit(map);
|
|
|
+ map = RT_NULL;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ food_init(map, 1);
|
|
|
+ }
|
|
|
+
|
|
|
+ widget = RTGUI_WIDGET(parameter);
|
|
|
+// snake_draw(widget);
|
|
|
+ snake_update(widget);
|
|
|
+
|
|
|
+ return;
|
|
|
+}
|
|
|
+
|
|
|
+void snake_main(void)
|
|
|
+{
|
|
|
+ struct rtgui_app* application;
|
|
|
+ struct rtgui_win* win;
|
|
|
+
|
|
|
+ application = rtgui_app_create(rt_thread_self(), "sanke_app");
|
|
|
+ if (application != RT_NULL)
|
|
|
+ {
|
|
|
+ win = rtgui_mainwin_create(RT_NULL,
|
|
|
+ "sanke_win",
|
|
|
+ RTGUI_WIN_STYLE_MAINWIN | RTGUI_WIN_STYLE_DESTROY_ON_CLOSE);
|
|
|
+ if(win == RT_NULL)
|
|
|
+ {
|
|
|
+ rt_kprintf("sanke_win create fail!\r\n");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ rtgui_object_set_event_handler(RTGUI_OBJECT(win), event_handler);
|
|
|
+
|
|
|
+ timer = rtgui_timer_create(RT_TICK_PER_SECOND / 2,
|
|
|
+ RT_TIMER_FLAG_PERIODIC,
|
|
|
+ timeout,
|
|
|
+ (void *)win);
|
|
|
+
|
|
|
+ rtgui_win_show(win, RT_TRUE);
|
|
|
+
|
|
|
+ //Í˳öºó²Å·µ»Ø
|
|
|
+ map_deinit(map);
|
|
|
+ snake_deinit();
|
|
|
+ food_deinit();
|
|
|
+ rtgui_app_destroy(application);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+#include <finsh.h>
|
|
|
+FINSH_FUNCTION_EXPORT(snake_main, snake run)
|