forked from len0rd/rockbox
		
	1) this patch replace all keymaps with PLA ones. It also clean some optionnal compiling that are not needed anymore througt PLA. 2) the patch also made required change to the manual in order to match code's change. 3) it also add an alternative exit button (PLA_EXIT or PLA_CANCEL). Change-Id: If6e78711eaab1dd2c907b418ba72c8789ceaa72b Reviewed-on: http://gerrit.rockbox.org/105 Reviewed-by: Thomas Martitz <kugel@rockbox.org> Tested-by: Thomas Martitz <kugel@rockbox.org>
		
			
				
	
	
		
			226 lines
		
	
	
	
		
			7.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			226 lines
		
	
	
	
		
			7.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*****************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _// __ \_/ ___\|  |/ /| __ \ / __ \  \/  /
 | |
|  *   Jukebox    |    |   ( (__) )  \___|    ( | \_\ ( (__) )    (
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2006 - 2008 Alexander Papst
 | |
|  * Idea from http://www.tetris1d.org
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU General Public License
 | |
|  * as published by the Free Software Foundation; either version 2
 | |
|  * of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/  
 | |
| 
 | |
| #include "plugin.h"
 | |
| #include "lib/pluginlib_actions.h"
 | |
| 
 | |
| /* this set the context to use with PLA */
 | |
| static const struct button_mapping *plugin_contexts[] = { pla_main_ctx };
 | |
| #define ONEDROCKBLOX_DOWN              PLA_DOWN
 | |
| #define ONEDROCKBLOX_DOWN_REPEAT       PLA_DOWN_REPEAT
 | |
| #define ONEDROCKBLOX_QUIT              PLA_EXIT
 | |
| #define ONEDROCKBLOX_QUIT2             PLA_CANCEL
 | |
| 
 | |
| #define mrand(max) (short)(rb->rand()%max)
 | |
| 
 | |
| #define TILES 11
 | |
| 
 | |
| /**********
 | |
| ** Lots of defines for the drawing stuff :-)
 | |
| ** The most ugly way i could think of...
 | |
| */
 | |
| 
 | |
| #if (LCD_WIDTH > LCD_HEIGHT)
 | |
|    /* Any screens larger than the minis LCD */
 | |
| #if (LCD_WIDTH > 132)
 | |
|      /* Max size of one block */
 | |
| #    define WIDTH (int)((LCD_HEIGHT * 0.85) / TILES)
 | |
|      /* Align the playing filed centered */
 | |
| #    define CENTER_X (int)(LCD_WIDTH/2-(WIDTH/2))
 | |
| #    define CENTER_Y (int)(LCD_HEIGHT/2-((WIDTH*TILES+TILES)/2))
 | |
|      /* Score box */
 | |
| #    define SCORE_X (int)(((CENTER_X+WIDTH+4) + (LCD_WIDTH-(CENTER_X+WIDTH+4))/2)-(f_width/2))
 | |
| #    define SCORE_Y (int)((LCD_HEIGHT/2)-(f_height/2))
 | |
|      /* Max. size of bricks is 4 blocks */
 | |
| #    define NEXT_H (WIDTH*4+3)
 | |
| #    define NEXT_X (int)(CENTER_X/2-WIDTH/2)
 | |
| #    define NEXT_Y (int)(LCD_HEIGHT/2-NEXT_H/2)
 | |
| #else
 | |
|      /* Max size of one block */
 | |
| #    define WIDTH (int)((LCD_HEIGHT * 0.85) / TILES)
 | |
|      /* Align the playing left centered */
 | |
| #    define CENTER_X (int)(LCD_WIDTH*0.2)
 | |
| #    define CENTER_Y (int)(LCD_HEIGHT/2-((WIDTH*TILES+TILES)/2))
 | |
|      /* Score box */
 | |
| #    define SCORE_X (int)(((CENTER_X+WIDTH+4) + (LCD_WIDTH-(CENTER_X+WIDTH+4))/2)-(f_width/2))
 | |
| #    define SCORE_Y 16
 | |
|      /* Max. size of bricks is 4 blocks */
 | |
| #    define NEXT_H (WIDTH*4+3)
 | |
| #    define NEXT_X (score_x+f_width+7)
 | |
| #    define NEXT_Y (int)(LCD_HEIGHT-((4*WIDTH)+13))
 | |
| #endif
 | |
| #else
 | |
|    /* Max size of one block */
 | |
| #  define WIDTH (int)((LCD_HEIGHT * 0.8) / TILES)
 | |
|    /* Align the playing filed centered */
 | |
| #  define CENTER_X (int)(LCD_WIDTH/2-(WIDTH/2))
 | |
| #  define CENTER_Y 2
 | |
|    /* Score box */
 | |
| #  define SCORE_X (int)((LCD_WIDTH/2)-(f_width/2))
 | |
| #  define SCORE_Y (LCD_HEIGHT-(f_height+2))
 | |
|    /* Max. size of bricks is 4 blocks */
 | |
| #  define NEXT_H (WIDTH*4+3)
 | |
| #  define NEXT_X (int)(CENTER_X/2-WIDTH/2)
 | |
| #  define NEXT_Y (int)((LCD_HEIGHT * 0.8)/2-NEXT_H/2)
 | |
| #endif
 | |
| 
 | |
| static void draw_brick(int pos, int length) {
 | |
|     int i = pos;
 | |
|     rb->lcd_set_drawmode(DRMODE_BG|DRMODE_INVERSEVID);
 | |
|     rb->lcd_fillrect(CENTER_X, CENTER_Y, WIDTH, WIDTH * TILES + TILES);
 | |
|     rb->lcd_set_drawmode(DRMODE_SOLID);
 | |
| 
 | |
|     for (i = pos; i < length + pos; ++i) {
 | |
|         if (i >= 0) rb->lcd_fillrect(CENTER_X, CENTER_Y+i+(WIDTH*i), WIDTH, WIDTH);
 | |
|     }
 | |
| }
 | |
| 
 | |
| enum plugin_status plugin_start(const void* parameter)
 | |
| {
 | |
|     int i;
 | |
|     int f_width, f_height;
 | |
|     int score_x;
 | |
| 
 | |
|     bool quit = false;
 | |
|     int button;
 | |
|     
 | |
|     int cycletime = 300;
 | |
|     int end;
 | |
| 
 | |
|     int pos_cur_brick = 0;
 | |
|     int type_cur_brick = 0;
 | |
|     int type_next_brick = 0;
 | |
|     
 | |
|     unsigned long int score = 34126;
 | |
|     
 | |
|     (void)parameter;
 | |
| 
 | |
| #if LCD_DEPTH > 1
 | |
|     rb->lcd_set_backdrop(NULL);
 | |
|     rb->lcd_set_background(LCD_BLACK);
 | |
|     rb->lcd_set_foreground(LCD_WHITE);
 | |
| #endif
 | |
| 
 | |
|     rb->lcd_setfont(FONT_SYSFIXED);
 | |
|     
 | |
|     rb->lcd_getstringsize("100000000", &f_width, &f_height);
 | |
|     
 | |
|     rb->lcd_clear_display();
 | |
|     
 | |
|     /***********
 | |
|     ** Draw EVERYTHING
 | |
|     */
 | |
|     
 | |
|     /* Playing filed box */
 | |
|     rb->lcd_vline(CENTER_X-2, CENTER_Y, CENTER_Y + (WIDTH*TILES+TILES));
 | |
|     rb->lcd_vline(CENTER_X + WIDTH + 1, CENTER_Y,
 | |
|                   CENTER_Y + (WIDTH*TILES+TILES));
 | |
|     rb->lcd_hline(CENTER_X-2, CENTER_X + WIDTH + 1, 
 | |
|                   CENTER_Y + (WIDTH*TILES+TILES));
 | |
| 
 | |
|     /* Score box */
 | |
| #if (LCD_WIDTH > LCD_HEIGHT)
 | |
|     rb->lcd_drawrect(SCORE_X-4, SCORE_Y-5, f_width+8, f_height+9);
 | |
|     rb->lcd_putsxy(SCORE_X-4, SCORE_Y-6-f_height, "score");
 | |
| #else
 | |
|     rb->lcd_hline(0, LCD_WIDTH, SCORE_Y-5);
 | |
|     rb->lcd_putsxy(2, SCORE_Y-6-f_height, "score");
 | |
| #endif
 | |
|     score_x = SCORE_X;
 | |
|     
 | |
|     /* Next box */
 | |
|     rb->lcd_getstringsize("next", &f_width, NULL);
 | |
| #if (LCD_WIDTH > LCD_HEIGHT) && !(LCD_WIDTH > 132)
 | |
|     rb->lcd_drawrect(NEXT_X-5, NEXT_Y-5, WIDTH+10, NEXT_H+10);
 | |
|     rb->lcd_putsxy(score_x-4, NEXT_Y-5, "next");
 | |
| #else
 | |
|     rb->lcd_drawrect(NEXT_X-5, NEXT_Y-5, WIDTH+10, NEXT_H+10);
 | |
|     rb->lcd_putsxy(NEXT_X-5, NEXT_Y-5-f_height-1, "next");
 | |
| #endif
 | |
| 
 | |
|     /***********
 | |
|     ** GAMELOOP
 | |
|     */
 | |
|     rb->srand( *rb->current_tick );
 | |
|     
 | |
|     type_cur_brick = 2 + mrand(3);
 | |
|     type_next_brick = 2 + mrand(3);
 | |
|     
 | |
|     do {
 | |
|         end = *rb->current_tick + (cycletime * HZ) / 1000;
 | |
|         
 | |
|         draw_brick(pos_cur_brick, type_cur_brick);
 | |
| 
 | |
|         /* Draw next brick */
 | |
|         rb->lcd_set_drawmode(DRMODE_BG|DRMODE_INVERSEVID);
 | |
|         rb->lcd_fillrect(NEXT_X, NEXT_Y, WIDTH, WIDTH * 4 + 4);
 | |
|         rb->lcd_set_drawmode(DRMODE_SOLID);
 | |
| 
 | |
|         for (i = 0; i < type_next_brick; ++i) {
 | |
|             rb->lcd_fillrect(NEXT_X, 
 | |
|                              NEXT_Y + ((type_next_brick % 2) ? (int)(WIDTH/2) : ((type_next_brick == 2) ? (WIDTH+1) : 0)) + (WIDTH*i) + i,
 | |
|                              WIDTH, WIDTH);
 | |
|         } 
 | |
| 
 | |
|         /* Score box */
 | |
|         rb->lcd_putsxyf(score_x, SCORE_Y, "%8ld0", score);
 | |
| 
 | |
|         rb->lcd_update();
 | |
| 
 | |
|         /*We get button from PLA this way */
 | |
|         button = pluginlib_getaction(TIMEOUT_NOBLOCK, plugin_contexts,
 | |
|                                ARRAYLEN(plugin_contexts));
 | |
| 
 | |
|         switch(button) {
 | |
|             case ONEDROCKBLOX_DOWN:
 | |
|             case ONEDROCKBLOX_DOWN_REPEAT:
 | |
|                 cycletime = 100;
 | |
|                 break;
 | |
|             case ONEDROCKBLOX_QUIT:
 | |
|             case ONEDROCKBLOX_QUIT2:
 | |
|                 quit = true;
 | |
|                 break;
 | |
|             default:
 | |
|                 cycletime = 300;
 | |
|                 if(rb->default_event_handler(button) == SYS_USB_CONNECTED) {
 | |
|                     quit = true;
 | |
|                 }
 | |
|         }
 | |
|         
 | |
|         if ((pos_cur_brick + type_cur_brick) > 10) {
 | |
|              type_cur_brick = type_next_brick;
 | |
|              type_next_brick = 2 + mrand(3);
 | |
|              score += (type_cur_brick - 1) * 2;
 | |
|              pos_cur_brick = 1 - type_cur_brick;
 | |
|         } else {
 | |
|             ++pos_cur_brick;
 | |
|         }
 | |
| 
 | |
|         if (TIME_BEFORE(*rb->current_tick, end))
 | |
|             rb->sleep(end-*rb->current_tick);
 | |
|         else
 | |
|             rb->yield();
 | |
| 
 | |
|     } while (!quit);
 | |
|  
 | |
|     return PLUGIN_OK;
 | |
| }
 |