FreeCalypso > hg > fc-tourmaline
diff src/ui/bmi/mmiGame.c @ 3:67bfe9f274f6
src/ui: import of src/ui3 from Magnetite
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Fri, 16 Oct 2020 06:33:10 +0000 |
parents | |
children | c0052fe355d3 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/ui/bmi/mmiGame.c Fri Oct 16 06:33:10 2020 +0000 @@ -0,0 +1,1910 @@ +/******************************************************************************* + + CONDAT (UK) + +******************************************************************************** + + This software product is the property of Condat (UK) Ltd and may not be + disclosed to any third party without the express permission of the owner. + +******************************************************************************** + + $Project name: Basic MMI + $Project code: BMI (6349) + $Module: Game + $File: Mmigame.c + $Revision: 1.0 + + $Author: Condat(UK) + $Date: 03/07/01 + +******************************************************************************** + + Description + Aug 16, 2004 REF: CRR 24323 Deepa M.D + Bug:Clenup of sprintf used for tracing + Fix:Replace the char buf[]; sprintf (buf, "...", ...); TRACE_EVENT (buf); statements by TRACE_EVENT_PX + + This provides the main game (four in a row) functionality + + +********************************************************************************/ + + +/******************************************************************************* + + Include files + +*******************************************************************************/ + +#define ENTITY_MFW + +/* includes */ +#include <string.h> +#include <stdio.h> +#include <stdlib.h> + +#if defined (NEW_FRAME) + +#include "typedefs.h" +#include "vsi.h" +#include "pei.h" +#include "custom.h" +#include "gsm.h" + +#else + +#include "STDDEFS.H" +#include "custom.h" +#include "gsm.h" +#include "vsi.h" + +#endif +#include "mfw_sys.h" + +#include "cus_aci.h" + +#include "mfw_mfw.h" +#include "mfw_win.h" +#include "mfw_kbd.h" +/* SPR#1428 - SH - New Editor changes */ +#ifndef NEW_EDITOR +#include "mfw_edt.h" +#endif +#include "mfw_lng.h" +#include "mfw_tim.h" +#include "mfw_icn.h" +#include "mfw_mnu.h" +#include "mfw_phb.h" +#include "mfw_cm.h" +#include "mfw_sim.h" +#include "mfw_nm.h" +#include "mfw_sat.h" +#include "mfw_ss.h" /*for convert*/ +#include "mfw_phb.h" +#include "ksd.h" +#include "psa.h" +#include "mfw_sms.h" +#include "mfw_cphs.h" +#include "mfw_sat.h" +#include "Mfw_band.h" +#include "mfw_ffs.h" +#include "Mmigame.h" + +#include "dspl.h" + +#include "MmiMmi.h" +#include "MmiDialogs.h" +#include "MmiLists.h" +#include "MmiBand.h" +#include "MmiCPHS.h" +/* SPR#1428 - SH - New Editor changes */ +#ifdef NEW_EDITOR +#include "ATBCommon.h" +#include "ATBDisplay.h" +#include "ATBEditor.h" +#include "AUIEditor.h" +#else +#include "MmiEditor.h" +#endif +#include"MmiBookShared.h" + + + + +#include "font_bitmaps.h" +#include "mmiColours.h" +#include "MmiResources.h" + +#include "Mmigame.h" +//GW 14/09/01 Disable game when not required. +#ifdef MMIGAME +void dspl_show_bitmap(int x,int y,t_font_bitmap* current_bitmap,U32 attr ); +/******************************************************************************* + + internal data + +*******************************************************************************/ + +#define FOUR_IN_A_ROW_INIT 121 +#define MAX_X_BOARD 11 +#define MAX_Y_BOARD 8 +/* +* The information related to every window must be encapsulated in such an structure +*/ + +typedef struct +{ + T_MMI_CONTROL mmi_control; // common control parameter + T_MFW_HND win; + T_MFW_HND kbd; + T_MFW_HND menu; + T_MFW_HND parent_win; + T_MFW_HND info_win; + char board_array[MAX_Y_BOARD][MAX_X_BOARD]; //the virtual board +} T_four_in_a_row; + +typedef struct +{ + /* administrative data */ + + T_MMI_CONTROL mmi_control; + T_MFW_HND win; + T_MFW_HND parent_win; +/* SPR#1428 - SH - New editor data */ +#ifdef NEW_EDITOR + T_AUI_EDITOR_DATA editor_data; +#else /* NEW_EDITOR */ + T_EDITOR_DATA editor_data; +#endif /* NEW_EDITOR */ + + /* internal data */ + char buffer[80]; + UBYTE status; + +} tShowInfo; + + +/* +* These are common functions xxx_create and xxx_destroy +*/ +T_MFW_HND four_in_a_row_create(MfwHnd parent); +void four_in_a_row_destroy (T_MFW_HND); + + +/* +* This dialog function (the same name as the window) +* is used to handle the comunication between different windows. The global macro SEND_EVENT can be used with parameter win +* and the corresponding events to send from one mmi dialog to another. +*/ +void four_in_a_row (T_MFW_HND win, USHORT event, SHORT value, void * parameter); + + +/* +* These are common optional functions handler +*/ +int four_in_a_row_kbd_cb (MfwEvt e, MfwKbd *k); +int four_in_a_row_win_cb (MfwEvt e, MfwWin *w); + +int Game_Result; + + +/* +* This an optional function, used often to call, create and init a new dialog, with different parameters depending +* on the context +*/ +T_MFW_HND four_in_a_row_start (T_MFW_HND win_parent,char *character); +int four_in_a_row_aktivate(MfwMnu* m, MfwMnuItem* i); + +int check_column(T_four_in_a_row *data, int where_from); +void drop_stone(T_four_in_a_row *data); + +int mobile_thinking(T_four_in_a_row *data); +int check_line(T_four_in_a_row *data, int x_value, int drop_position, int dx, int dy, BOOL where_from); +void check_for_winner(T_four_in_a_row *data, int check_out_position); +int random_scoring(void); // RAVI + +T_MFW_HND Game_Info(T_MFW_HND parent_window); +static T_MFW_HND ShowGame_Information(MfwHnd parent_window); +static void ShowGame_DialogCB(T_MFW_HND win, USHORT e, SHORT identifier, void *parameter); +void ShowGameInfoEditor(T_MFW_HND win); +static void ShowInfoCB( T_MFW_HND win, USHORT Identifier,UBYTE reason); +void showGameInfo_destroy(MfwHnd own_window); + +const char * info = "Welcome to Connect 4!"; +const char * playAgain = "Play Again?"; + + +//internal datas + +int column; //this is the current position of white player cursor +int x_runer; //goes through the horizontal lines +int y_runer; //goes through the vertical lines +char winner; //describes the winner 0->nobody 'B'->black(mobile) 'W'->white(human player) +int column_black; //this is the current position of black player cursor +int request_stone; //helps to decide between black or white (for strategical things) + + + +#ifndef BMP_FORMAT_BW_UNPACKED +#define BMP_FORMAT_BW_UNPACKED 1 +#endif + + +t_font_bitmap plWin1 ={ 0, BMP_FORMAT_BW_UNPACKED, 21, 24, 0, (char*)player_win1}; +t_font_bitmap plWin2 ={ 0, BMP_FORMAT_BW_UNPACKED, 21, 24, 0, (char*)player_win2}; +t_font_bitmap plLost1 ={ 0, BMP_FORMAT_BW_UNPACKED, 21, 24, 0, (char*)player_lost1}; +t_font_bitmap plLost2 ={ 0, BMP_FORMAT_BW_UNPACKED, 21, 24, 0, (char*)player_lost2}; +t_font_bitmap draw1 ={ 0, BMP_FORMAT_BW_UNPACKED, 21, 24, 0, (char*)board_full1}; +t_font_bitmap draw2 ={ 0, BMP_FORMAT_BW_UNPACKED, 21, 24, 0, (char*)board_full2}; +t_font_bitmap gameName1 ={ 0, BMP_FORMAT_BW_UNPACKED, 18, 24, 0, (char*)game_name1}; +t_font_bitmap gameName2 ={ 0, BMP_FORMAT_BW_UNPACKED, 24, 24, 0, (char*)game_name2}; + +#ifdef COLOURDISPLAY +char colBmp[256*4]; +t_font_bitmap allCol ={ 0, BMP_FORMAT_256_COLOUR, 32, 32, 0, colBmp }; +t_font_bitmap stone_colour ={ 0, BMP_FORMAT_BW_UNPACKED, 8, 8, 0, (char*)black_stone_bw}; +t_font_bitmap gameCursor = { 0, BMP_FORMAT_BW_UNPACKED, 8, 8, 0, (char*)game_cursor}; +#else +#ifdef LSCREEN +t_font_bitmap blackStone_bw ={ 0, BMP_FORMAT_BW_UNPACKED, 8, 8, 0, (char*)black_stone_bw}; +t_font_bitmap whiteStone_bw ={ 0, BMP_FORMAT_BW_UNPACKED, 8, 8, 0, (char*)white_stone_bw}; +t_font_bitmap gameCursor = { 0, BMP_FORMAT_BW_UNPACKED, 8, 8, 0, (char*)game_cursor}; +#else +t_font_bitmap blackStone_bw ={ 0, BMP_FORMAT_BW_UNPACKED, 4, 4, 0, (char*)black_stone_bw}; +t_font_bitmap whiteStone_bw ={ 0, BMP_FORMAT_BW_UNPACKED, 4, 4, 0, (char*)white_stone_bw}; +t_font_bitmap gameCursor = { 0, BMP_FORMAT_BW_UNPACKED, 6, 6, 0, (char*)game_cursor}; +#endif +#endif + +t_font_bitmap *blackStone, *whiteStone; + +/******************************************************************************* + + $Function: four_in_a_row_create + + $Description: + + $Returns: + + $Arguments: + +*******************************************************************************/ + + +T_MFW_HND four_in_a_row_create (T_MFW_HND parent_window) +{ + + T_MFW_WIN * win; + + /* + * This window is dynamic, for that reason the associated data are allocated in the mfw heap + */ + T_four_in_a_row * data = (T_four_in_a_row *)ALLOC_MEMORY (sizeof (T_four_in_a_row)); + + TRACE_FUNCTION ("four_in_a_row_create()"); + + /* + * Create window handler + */ + + data->win = win_create (parent_window, 0, E_WIN_VISIBLE, (T_MFW_CB)four_in_a_row_win_cb); + + if (data->win EQ 0) + return 0; + + /* + * These assignments are necessary to attach the data to the window, and to handle the mmi event communication. + */ + + data->mmi_control.dialog = (T_DIALOG_FUNC)four_in_a_row; + data->mmi_control.data = data; + data->parent_win = parent_window; + win = ((T_MFW_HDR *)data->win)->data; + win->user = (void *) data; + + /* + * Create any other handler + */ + + data->kbd = kbd_create (data->win,KEY_ALL,(T_MFW_CB)four_in_a_row_kbd_cb); + + column=0; //set start value for game cursor + winner ='0'; //game starts, so set winner to nobody + + /*clear the board*/ + for (y_runer=0;y_runer<MAX_Y_BOARD;y_runer++){ + for(x_runer=0;x_runer<MAX_X_BOARD;x_runer++){ + data->board_array[y_runer][x_runer] ='0'; + } + } + winShow(data->win); + + return data->win; +} + +/******************************************************************************* + + $Function: four_in_a_rowr_destroy + + $Description: + + $Returns: + + $Arguments: + +*******************************************************************************/ + +void four_in_a_row_destroy (T_MFW_HND own_window) +{ + T_MFW_WIN * win; + T_four_in_a_row * data; + + TRACE_EVENT("four_in_a_row_destroy"); + + if (own_window) + { + win = ((T_MFW_HDR *)own_window)->data; + data = (T_four_in_a_row *)win->user; + + if (data) + { + /* + * Exit Keyboard Handler + */ + /* + * Delete WIN Handler + */ + win_delete (data->win); + } + + /* + * In this case the data attached to window must be also deleted. + */ + FREE_MEMORY ((void *)data, sizeof (T_four_in_a_row)); + + column=0; //set the current column back to 0 + + } +} + +/******************************************************************************* + + $Function: four_in_a_row_start + + $Description: This function just creates and inits the new dialog + + $Returns: + + $Arguments: + +*******************************************************************************/ +T_MFW_HND four_in_a_row_start (T_MFW_HND win_parent,char *character) +{ + T_MFW_HND win; + /* + MmiTetrisStart(); + return; + */ + win = four_in_a_row_create (win_parent); + TRACE_EVENT("four_in_a_row_start"); + + if (win NEQ NULL) + { + SEND_EVENT(win,FOUR_IN_A_ROW_INIT,0,character); + } + return win; + +} + +/******************************************************************************* + + $Function: four_in_a_row + + $Description: + + $Returns: + + $Arguments: + +*******************************************************************************/ + + + +void four_in_a_row (T_MFW_HND win, USHORT event, SHORT value, void * parameter) +{ + T_MFW_WIN * win_data = ((T_MFW_HDR *) win)->data; + T_four_in_a_row * data = (T_four_in_a_row *)win_data->user; + + TRACE_FUNCTION ("four_in_a_row()"); + + /* + * In this case the communication is very simple (only one intern event) + */ + + + switch (event) + { + case FOUR_IN_A_ROW_INIT: + softKeys_displayId(TxtSearchName,TxtNull,0,COLOUR_IDLE); + winShow(data->win); + break; + + default: + return; + } + +} + +/******************************************************************************* + + $Function: four_in_a_row_win_cb + + $Description: win calback function of four in a row + + $Returns: + + $Arguments: + +*******************************************************************************/ + + +int four_in_a_row_win_cb (MfwEvt e, MfwWin *w) +{ +// T_DISPLAY_DATA display_info; // RAVI + T_four_in_a_row * data = (T_four_in_a_row *)w->user; + int xOfs,yOfs, xScale, yScale; + int manOfsX,manOfsY; + int xPos,yPos; + int y_axis; //goes through the vertical lines + int x_axis; //goes through the horizontal lines +// int x,y,i,b; // RAVI + int displayPosX, temp; + + TRACE_FUNCTION ("four_in_a_row_win_cb()"); + + switch (e) + { + case MfwWinVisible: +#ifdef COLOURDISPLAY + whiteStone = &stone_colour; + blackStone = &stone_colour; + +#else + whiteStone = &blackStone_bw; + blackStone = &whiteStone_bw; + +#endif +#ifdef LSCREEN + manOfsX = GAME_POS_X + 3; + manOfsY = GAME_POS_Y + 3; +#else + manOfsX = GAME_POS_X + 1; + manOfsY = GAME_POS_Y + 1; +#endif + xOfs = GAME_POS_X; + yOfs = GAME_POS_Y; + xScale = GAME_SCALE_X; + yScale = GAME_SCALE_Y; + + //dspl_ClearAll(); //Clears to white, not BGD! + TRACE_EVENT("display clear"); + resources_setColour( COLOUR_GAME ); + dspl_Clear( 0,0, SCREEN_SIZE_X,SCREEN_SIZE_Y); +#ifdef LSCREEN + softKeys_displayId(TxtHelp,TxtExit,0,COLOUR_LIST_SUBMENU); +#endif + + resources_setColour( COLOUR_GAME ); + dspl_Clear( xOfs, yOfs, xOfs+MAX_X_BOARD*xScale, yOfs+MAX_Y_BOARD*yScale ); //Clears to white, not BGD! + //dspl_BitBlt(column*xScale+xOfs+1,yOfs-yScale,8,8,0,(void*)game_cursor,0); //place the cursor bitmap + dspl_show_bitmap(column*xScale+xOfs+1,yOfs-yScale, &gameCursor, 0 ); + for (y_axis=0;y_axis<=MAX_Y_BOARD;y_axis++) + { + dspl_DrawLine( xOfs, yOfs+y_axis*yScale, + xOfs+MAX_X_BOARD*xScale, yOfs+y_axis*yScale); + } + for(x_axis=0;x_axis<=MAX_X_BOARD;x_axis++) + { + dspl_DrawLine( xOfs+x_axis*xScale, yOfs, + xOfs+x_axis*xScale, yOfs+MAX_Y_BOARD*yScale); + } + + //go through the board-array and check for W or B and set the stones + for (y_axis=0;y_axis<MAX_Y_BOARD;y_axis++) + { + for(x_axis=0;x_axis<MAX_X_BOARD;x_axis++) + { + xPos = manOfsX+x_axis*xScale; + yPos = manOfsY+y_axis*yScale; + if(data->board_array[y_axis][x_axis] EQ 'W'){ //found white + //set white_stone bitmap + dspl_SetFgdColour( COL_R ); + dspl_show_bitmap(xPos,yPos, whiteStone, 0 ); + } + if(data->board_array[y_axis][x_axis] EQ 'B'){ //found black + //set black_stone bitmap + dspl_SetFgdColour( COL_G ); //0x00404040 + dspl_show_bitmap(xPos,yPos, blackStone, 0 ); + } + } + } + dspl_SetFgdColour( COL_RB ); + dspl_SetBgdColour( COL_RG ); + dspl_show_bitmap(GAME_NAMEPOSX1, GAME_NAMEPOSY1, &gameName1, 0 ); + displayPosX = (SCREEN_SIZE_X/2); + temp = strlen((char*)playAgain); + //this if-clause is to set the bitmap, for win or lost game + if(winner EQ 'W'){ + //human player wins, so set win bitmap +#ifdef LSCREEN + dspl_SetFgdColour( COL_BLK ); + dspl_SetBgdColour( COL_TRANSPARENT ); + Game_Result = TRUE; + softKeys_displayId(TxtYes,TxtNo,0,COLOUR_LIST_SUBMENU); + dspl_TextOut((displayPosX-((temp*CHAR_WIDTH)/2)),170,DSPL_TXTATTR_NORMAL,(char*)playAgain); +#endif + dspl_SetFgdColour( COL_RB ); + dspl_SetBgdColour( COL_GB ); + dspl_show_bitmap(GAME_WINPOSX1, GAME_WINPOSY1, &plWin1, 0 ); + dspl_show_bitmap(GAME_WINPOSX2, GAME_WINPOSY2, &plWin2, 0 ); + } + else + if(winner EQ 'B'){ + //human player lost, so set lost bitmap +#ifdef LSCREEN + dspl_SetFgdColour( COL_BLK ); + dspl_SetBgdColour( COL_TRANSPARENT ); + Game_Result = TRUE; + softKeys_displayId(TxtYes,TxtNo,0,COLOUR_LIST_SUBMENU); + dspl_TextOut((displayPosX-((temp*CHAR_WIDTH)/2)),170,DSPL_TXTATTR_NORMAL,(char*)playAgain); +#endif + dspl_SetFgdColour( COL_RB ); + dspl_SetBgdColour( COL_GB ); + dspl_show_bitmap(GAME_WINPOSX1, GAME_WINPOSY1, &plLost1, 0 ); + dspl_show_bitmap(GAME_WINPOSX2, GAME_WINPOSY2, &plLost2, 0 ); + } + else + if(winner EQ 'N'){ + //board is full, nobody (N) wins +#ifdef LSCREEN + dspl_SetFgdColour( COL_BLK ); + dspl_SetBgdColour( COL_TRANSPARENT ); + Game_Result = TRUE; + softKeys_displayId(TxtYes,TxtNo,0,COLOUR_LIST_SUBMENU); + dspl_TextOut((displayPosX-((temp*CHAR_WIDTH)/2)),170,DSPL_TXTATTR_NORMAL,(char*)playAgain); +#endif + + dspl_SetFgdColour( COL_R ); + dspl_SetBgdColour( COL_G ); + dspl_show_bitmap(GAME_WINPOSX1, GAME_WINPOSY1, &draw1, 0 ); + dspl_show_bitmap(GAME_WINPOSX2, GAME_WINPOSY2, &draw2, 0 ); + } + else + { + dspl_SetFgdColour( COL_RB ); + dspl_SetBgdColour( COL_RG ); + dspl_show_bitmap(GAME_NAMEPOSX1, GAME_NAMEPOSY1, &gameName1, 0 ); + dspl_show_bitmap(GAME_NAMEPOSX2, GAME_NAMEPOSY2, &gameName2, 0 ); + dspl_show_bitmap(GAME_NAMEPOSX3, GAME_NAMEPOSY3, &gameName2, 0 ); + } + break; + + default: + return 0; + } + + return 1; +} + +/******************************************************************************* + + $Function: four_in_a_row_kbd_cb + + $Description: + + $Returns: + + $Arguments: + +*******************************************************************************/ + +int four_in_a_row_kbd_cb (MfwEvt e, MfwKbd *k) +{ + + T_MFW_HND win = mfw_parent (mfw_header()); + T_MFW_WIN * win_data = ((T_MFW_HDR *)win)->data; + T_four_in_a_row * data = (T_four_in_a_row *)win_data->user; + + TRACE_EVENT ("four_in_a_row_kbd_cb()"); + + switch (k->code) + { + case KCD_HUP: //quit the game "four in a row" + case KCD_RIGHT: //quit the game "four in a row" + TRACE_EVENT ("quit four_in_a_row"); + dspl_ClearAll(); //clear mobile screen + four_in_a_row_destroy (win); //destroy win-handler + break; + + case KCD_MNULEFT: + case KCD_1: + if(winner EQ '0'){ //while nobody has won.... + TRACE_EVENT ("KCD_1"); + if(column > 0) + { //cursor must be greater than 0 + column=column-1; + } + winShow(data->win); + } + break; + case KCD_MNUSELECT: + case KCD_2: + if(winner EQ '0'){ //while nobody has won.... + TRACE_EVENT ("KCD_2"); + drop_stone(data); //drop stone + } + winShow(data->win); + break; + case KCD_MNURIGHT: + case KCD_3: + if(winner EQ '0'){ //while nobody has won.... + TRACE_EVENT ("KCD_3"); + if(column<MAX_X_BOARD-1 ){ //cursor must be less than 50 + column=column+1; + } + winShow(data->win); + } + break; + case KCD_LEFT: + if(Game_Result == TRUE) + { + Game_Result = FALSE; + four_in_a_row_destroy(win); //destroy win-handler + four_in_a_row_start(win,0); + } + else + Game_Info(win); + break; + } + return MFW_EVENT_CONSUMED; +} + +/******************************************************************************* + + $Function: four_in_a_row_aktivate + + $Description: Starts the game function on user selection + + $Returns: MFW_EVENT_CONSUMED if event handled, otherwise + MFW_EVENT_PASSED + + $Arguments: menu, menu item + +*******************************************************************************/ +int four_in_a_row_aktivate(MfwMnu* m, MfwMnuItem* i) +{ + + T_MFW_HND parent = mfwParent( mfw_header()); + TRACE_EVENT("four_in_a_row_aktivate()"); + four_in_a_row_start(parent,0); + return MFW_EVENT_CONSUMED; +} + +/******************************************************************************* + + $Function: drop_stone + + $Description: decide whether a stone can be droped and where it can be droped + + $Returns: + + $Arguments: T_four_in_a_row + +*******************************************************************************/ +void drop_stone(T_four_in_a_row *data) +{ + int drop_position; //position (y in current board column) where to drop the stone + + TRACE_EVENT("drop_stone()"); + request_stone=0; //white is on turn + drop_position=check_column(data,request_stone); //set the drop position for the white stone in the current white, board column + if(drop_position>=0){ //if the current column is not filled up set stone + data->board_array[drop_position][column]='W'; //white is first and set a 'W' in the virtual board for a white stone + check_for_winner(data,drop_position); //check whether the last move(white) was a win-move... + if(winner EQ '0'){ //...if not, then the mobile(black) is on turn + drop_position=mobile_thinking(data); //set the drop position for the black stone in the current black, board column + data->board_array[drop_position][column_black]='B'; //set a 'B' for in the virtual board for a black stone + check_for_winner(data,drop_position); //check whether the last move(black) was a win-move... + } + } +} + +/******************************************************************************* + + $Function: check_for_winner + + $Description: checks the current drop position for a win-position + + $Returns: + + $Arguments: T_four_in_a_row + +*******************************************************************************/ +void check_for_winner(T_four_in_a_row *data, int check_out_position) +{ + int check_with_this_color; //set the color to check for + int color_regulator; //helps to switch between both colors + int check_this_column; //declares the column to check for + + TRACE_EVENT("check_for_winner()"); + if(data==NULL || check_out_position<0 || check_out_position>=MAX_Y_BOARD) + { + TRACE_ERROR("check_for_winner() invalid parameters"); + return; + } + color_regulator=0; //at first, use white to check for + winner='0'; //at first we don't know who wins, so nobody is set as winner + while(color_regulator<2){ //while checking for white(0) and black(1) is not ready.... + if(color_regulator EQ 0){ //if color_regulator is 0, the board will be checked for white stones ('W') + check_with_this_color='W'; + }else{ //if color_regulator is 1, the board will be checked for black stones ('B') + check_with_this_color='B'; + } + + if(request_stone EQ 0){ //if the board will be checked for white, use column as the current column + check_this_column=column; + }else{ //if the board will be checked for black, use column_black as the current column + check_this_column=column_black; + } + if(check_this_column<0 || check_this_column>=MAX_X_BOARD) + { + TRACE_ERROR("check_for_winner() invalid index check_this_column"); + break; + } + /*Begin with the checks, the mobile checks for vertical, horizontal and diagonal possibilities*/ + //check row + if(check_this_column>=3 && check_this_column<=10 && + data->board_array[check_out_position][check_this_column-3] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column-2] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_this_column>=2 && check_this_column<=9 && + data->board_array[check_out_position][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column-2] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_this_column>=0 && check_this_column<=7 && + data->board_array[check_out_position][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column+2] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column+3] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_this_column>=1 && check_this_column<=8 && + data->board_array[check_out_position][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column+2] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + //check column + }else if(check_out_position<=4 && + data->board_array[check_out_position+1][check_this_column] EQ check_with_this_color && + data->board_array[check_out_position+2][check_this_column] EQ check_with_this_color && + data->board_array[check_out_position+3][check_this_column] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + //check diagonal + }else if(check_out_position<=4 && check_this_column>=3 && check_this_column<=10 && + data->board_array[check_out_position+1][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position+2][check_this_column-2] EQ check_with_this_color && + data->board_array[check_out_position+3][check_this_column-3] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_out_position<=4 && check_this_column>=0 && check_this_column<=7 && + data->board_array[check_out_position+1][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position+2][check_this_column+2] EQ check_with_this_color && + data->board_array[check_out_position+3][check_this_column+3] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_out_position>=3 && check_this_column>=0 && check_this_column<=7 && + data->board_array[check_out_position-1][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position-2][check_this_column+2] EQ check_with_this_color && + data->board_array[check_out_position-3][check_this_column+3] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_out_position>=3 && check_this_column>=3 && check_this_column<=10 && + data->board_array[check_out_position-1][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position-2][check_this_column-2] EQ check_with_this_color && + data->board_array[check_out_position-3][check_this_column-3] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_out_position<=6 && check_out_position>=2 && check_this_column>=1 && check_this_column<=8 && + data->board_array[check_out_position+1][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position-1][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position-2][check_this_column+2] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_out_position<=5 && check_out_position>=1 && check_this_column>=2 && check_this_column<=9 && + data->board_array[check_out_position+1][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position+2][check_this_column-2] EQ check_with_this_color && + data->board_array[check_out_position-1][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_out_position<=5 && check_out_position>=1 && check_this_column>=1 && check_this_column<=8 && + data->board_array[check_out_position-1][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position+1][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position+2][check_this_column+2] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else if(check_out_position>=2 && check_out_position<=6 && check_this_column>=2 && check_this_column<=9 && + data->board_array[check_out_position-1][check_this_column-1] EQ check_with_this_color && + data->board_array[check_out_position-2][check_this_column-2] EQ check_with_this_color && + data->board_array[check_out_position+1][check_this_column+1] EQ check_with_this_color && + data->board_array[check_out_position][check_this_column] EQ check_with_this_color){ + winner=check_with_this_color; //set the current color to check for as the winner + break; + }else{ + winner='0'; //no win-moves found, so nobody is the winner and the game goes on + } + color_regulator++; //set the check color one up (to black) + } +} +/******************************************************************************* + + $Function: check_column + + $Description: checks for already set stones + + $Returns: int unused_position + + $Arguments: T_four_in_a_row + +*******************************************************************************/ +int check_column(T_four_in_a_row *data, int where_from) +{ + int unused_position; //is for a free drop position + int column_for_check; //column (vertical board line) to check for + + if(where_from EQ 0){ //is the current game color(player) 0 then use column + column_for_check=column; + }else{ //is the current game color(player) 1 then use column_black + column_for_check=column_black; + } + unused_position=-9999; + for(y_runer=0;y_runer<MAX_Y_BOARD;y_runer++){ //test all vertical fields whether they are empty('0') or not + if(data->board_array[y_runer][column_for_check] EQ '0'){ + unused_position=y_runer; //.this field is empty, so it is a unused position + } + } + return unused_position; //return the unused position +} + + +/******************************************************************************* + + $Function: mobile_thinking + + $Description: mobile moves + + $Returns: drop position of black stone (drop_me) + + $Arguments: T_four_in_a_row + +*******************************************************************************/ +int mobile_thinking(T_four_in_a_row *data) +{ + int position_score[11]; //this array takes the scoring values of each column in the game board + int drop_position; //drop position for stone + int drop_me; //buffer value + int scores[4]; //this array keeps the scores for row, column and diagonals + int add_random_score; + + TRACE_EVENT("mobile_thinking()"); + + request_stone=1; //black is + //check for good spots to drop the black stone + for(x_runer=0;x_runer<MAX_X_BOARD;x_runer++){ + //found the possible drop position + column_black=x_runer; //set the x_runer to the current column (column_black) + drop_position=check_column(data,request_stone); //check for possible drop positions + if(drop_position>=0){ //if there is a possible drop position..... + //check the scoring of the current point + add_random_score=random_scoring(); + data->board_array[drop_position][x_runer]='B'; //set a test black stone + scores[0]=check_line(data,x_runer,drop_position,1,0,0); //scores row + scores[1]=check_line(data,x_runer,drop_position,1,1,0); //scores diagonal right-up/left-down + scores[2]=check_line(data,x_runer,drop_position,1,-1,0); //scores diagonal left-up/right-down + scores[3]=check_line(data,x_runer,drop_position,0,1,0); //scores down(column) + position_score[x_runer]=scores[0]+scores[1]+scores[2]+scores[3]; //count all scores together to one total score + if(drop_position-1 >=0){ //Is it possible to set a future stone here? If Yes go on... + data->board_array[drop_position-1][x_runer]='W'; //set a test white stone for a possible next opponent move + scores[0]=check_line(data,x_runer,drop_position-1,1,0,1); //scores row + scores[1]=check_line(data,x_runer,drop_position-1,1,1,1); //scores diagonal right-up/left-down + scores[2]=check_line(data,x_runer,drop_position-1,1,-1,1); //scores diagonal left-up/right-down + scores[3]=check_line(data,x_runer,drop_position-1,0,1,1); //scores down(column) + position_score[x_runer]=position_score[x_runer]-scores[0]-scores[1]-scores[2]-scores[3]+add_random_score; //substraction of the future score from the normal score + data->board_array[drop_position-1][x_runer]='0'; //remove test white stone + } + data->board_array[drop_position][x_runer]='0'; //remove test black stone + }else{ //if there is no possible drop position set the score to -1 + position_score[x_runer]=-9999; + } + } + + drop_me=-9999; //buffer is set to absolutely minus + column_black=-9999; //position of black cursor is left outer side of board + for(x_runer=0;x_runer<MAX_X_BOARD;x_runer++){ //goes through the scoring array to find the highest score + if(position_score[x_runer]>drop_me){ + drop_me=position_score[x_runer]; + column_black=x_runer; //set the x_runer to the current column(column_black) + drop_position=check_column(data,request_stone); //this is the best drop position for the mobile + } + } + if(column_black EQ -9999){ //if the column is -1, then the game board is filled up and nobody has won + TRACE_EVENT("Board Full!!!"); + winner='N'; //set winner to nobody (N) + } + return drop_position; //return the mobile drop position +} + +/******************************************************************************* + + $Function: check_line + + $Description: set score points for the mobile, so the mobile knows where to set the next stone + + $Returns: total_score + + $Arguments: T_four_in_a_row, x_value, drop_position, dx, dy, where_from + +*******************************************************************************/ +int check_line(T_four_in_a_row *data, int x_value, int y_value, int dx, int dy, BOOL where_from) +{ + int lSc,nSc,oSc,total_score,sx,sy,i,j; + int pl_nSc; //no of empty squares that have a man under them (i.e. we/opponent can use next shot) + total_score=0; + for (i=0;i<4;i++){ + sx = x_value-i*dx; + sy = y_value-i*dy; + lSc=0; + nSc=0; pl_nSc=0; + oSc=0; + for (j=0;j<4;j++){ + if ((sx<0) || (sx>=MAX_X_BOARD) ||(sy<0) || (sy>=MAX_Y_BOARD)){ /*a0393213 lint warnings removal - 'Possible access of out-of-bounds pointer'*/ + lSc = -1; + }else if (lSc>=0){ + if(where_from EQ 0){ + if (data->board_array[sy][sx] EQ 'B'){ + lSc++; + }else if (data->board_array[sy][sx] EQ '0'){ + nSc++; + if ((sy EQ (MAX_Y_BOARD-1)) || (data->board_array[sy+1][sx] NEQ '0')) /*a0393213 lint warnings removal - 'Possible access of out-of-bounds pointer'*/ + pl_nSc++; + }else if (data->board_array[sy][sx] EQ 'W'){ + oSc++; + + }else //edge of board found + lSc=-1; + sx = sx+dx; + sy = sy+dy; + }else{ + if (data->board_array[sy][sx] EQ 'W'){ + lSc++; + }else if (data->board_array[sy][sx] EQ '0'){ + nSc++; + if ((sy EQ (MAX_Y_BOARD-1)) || (data->board_array[sy+1][sx] NEQ '0')) /*a0393213 lint warnings removal - 'Possible access of out-of-bounds pointer'*/ + pl_nSc++; + }else if (data->board_array[sy][sx] EQ 'B'){ + oSc++; + + }else //edge of board found + lSc=-1; + sx = sx+dx; + sy = sy+dy; + } + } + } + if (lSc >= 0){ + if (lSc EQ 4){ + return(9999);//got 4 in a row. + }else if (oSc EQ 3) //and ISc==1 + { + // 3 opp men - good spot! + if(where_from EQ 0){ + total_score = total_score + 1000; //this is for the normal, current position score + }else{ + total_score = -total_score - 1000; //this is for the possible future score, it must be minus, because it will be substracted from normal score + } + }else if ((lSc EQ 3) && (nSc EQ 1)){ + //playing here will form 3 in a row and a blank - good spot + if (pl_nSc EQ 0) //can't reach the 4th position + total_score = total_score + 75; + else + total_score = total_score + 125;//better if we can reach 4th position + }else if ((oSc EQ 2) && (nSc EQ 1)){ + // 2 opp men + 1 blank- good spot! + if (pl_nSc EQ 0) //can't reach the 4th position-not too bad + total_score = total_score + 50; + else + total_score = total_score + 200;//can get to both positions - have to block. + }else if ((lSc EQ 2) && (nSc EQ 2)){ + // 2 in a row and 2 blanks - OK spot + if (pl_nSc EQ 0) //can't reach the 3rd or 4th position-not too good + total_score = total_score + 20; + else if (pl_nSc EQ 1) //can reach one of the 3rd or 4th position - good + total_score = total_score + 45; + else if (pl_nSc EQ 2) //can reach both 3rd and 4th position - very good + total_score = total_score + 95; + }else if ((lSc EQ 1) && (nSc EQ 3)){ + // 1 in a row and 3 blanks - OK spot, better if we can reach the other 3 points. + total_score = total_score + 2*(pl_nSc+1); //pl_nSc=0,1,2 or 3 sc=sc+2,4,6 or 8 + } + //else //line is neither good nor bad. + } + else //line is not valid + total_score = total_score - 1; + } + return(total_score); +} +/******************************************************************************* + + $Function: check_line + + $Description: generates a random number to add on the field scores + + $Returns: random_score_result + + $Arguments: + +*******************************************************************************/ +int random_scoring(void) +{ + return(rand()%100); +} + + + +#define MAX_TET_X 11 +#define MAX_TET_Y 20 + +typedef struct +{ + + T_MMI_CONTROL mmi_control; // common control parameter + T_MFW_HND win; + T_MFW_HND hKbd; + T_MFW_HND menu; + T_MFW_HND parent_win; + T_MFW_HND info_tim; + + int dropTime; + int maxDropTime; + int timeStep; + + int xPos; + int yPos; + + int px; + int py; + int shapeId; + char shapeStr[20]; + int rotateId; + + int score; + int level; + int linesToGo; + + int resetSpeed; + + int gameOver; + + char tetrisScreen[MAX_TET_X][MAX_TET_Y]; + +} T_TETRIS; + + +static MfwHnd TetrisCreate(MfwHnd hParentWin); +static void TetrisExecCb (MfwHnd hWin, USHORT uiE, SHORT iValue, void *pParameter); +static int TetrisKbdCb (MfwEvt uiE, MfwKbd *psK); +static int tetris_tim_cb(MfwEvt e, MfwTim *t); +static int TetrisWinCb (MfwEvt uiE, MfwWin *psWin); + + +#define E_INIT 0x5000 +#define E_EXIT 0x5001 + +void initTetGame(T_TETRIS *data) +{ + int x,y; + for (y=0;y<MAX_TET_Y;y++) + { + for (x=0;x<MAX_TET_X;x++) + data->tetrisScreen[x][y]=' '; + data->tetrisScreen[0][y]='#'; + data->tetrisScreen[MAX_TET_X-1][y]='#'; + + } + for (x=0;x<MAX_TET_X;x++) + data->tetrisScreen[x][MAX_TET_Y-1]='#'; +} + +#define MAX_SHAPE 7 +void tetris_getShape(int shapeId, int rotate, char* shapeStr) +{ + int j,k; + int index; + const char* tetShape[4*4*MAX_SHAPE] = + { " ", " A ", " B ", " C ", " ", "F ", " G ", + "EEEE", " AA ", " BB ", "CCC ", " DD ", "FFF ", "GGG ", + " ", " A ", " B ", " ", " DD ", " ", " ", + " ", " ", " ", " ", " ", " ", " ", + " E ", " AA ", "BB ", " C ", " ", " FF ", " G ", + " E ", "AA ", " BB ", " CC ", " DD ", " F ", " G ", + " E ", " ", " ", " C ", " DD ", " F ", " GG ", + " E ", " ", " ", " ", " ", " ", " ", + " ", " A ", " B ", " ", " ", " ", " ", + "EEEE", " AA ", " BB ", "CCC ", " DD ", "FFF ", "GGG ", + " ", " A ", " B ", " C ", " DD ", " F ", "G ", + " ", " ", " ", " ", " ", " ", " ", + " E ", " AA ", "BB ", " C ", " ", " F ", "GG ", + " E ", "AA ", " BB ", "CC ", " DD ", " F ", " G ", + " E ", " ", " ", " C ", " DD ", "FF ", " G ", + " E ", " ", " ", " ", " ", " ", " " + }; + + + for (k=0;k<4;k++) + { + index = rotate*4*MAX_SHAPE+k*MAX_SHAPE; + for (j=0;j<4;j++) + shapeStr[j+k*4] = tetShape[index+shapeId][j]; + } +} + +int tetris_getNextShapeId( void ) +{ + static int shapeIndex = 0; + shapeIndex = (shapeIndex+1) % MAX_SHAPE; + return (shapeIndex); +} + + + +void tetris_getNextShape(T_TETRIS *data) +{ + data->shapeId = tetris_getNextShapeId( ); + data->rotateId = 0; + tetris_getShape(data->shapeId, data->rotateId, data->shapeStr); + data->px = (MAX_TET_X-2)/2; + data->py = 0; +} + +void tetris_getNextLevel(T_TETRIS *data) +{ + initTetGame(data); + tetris_getNextShape(data); + data->level = data->level+1; + data->linesToGo = 5*(data->level+1); + data->resetSpeed = 0; + data->maxDropTime = (120-data->level*5); +} + +void initTetrisData( T_TETRIS *data ) +{ + data->level = 0; + data->score = 0; + data->gameOver = 0; + tetris_getNextLevel(data); +} + +void dspl_FillRect( int x1, int y1, int sx, int sy) +{ + dspl_DrawFilledBgdRect(x1,y1,x1+sx,y1+sy); +} + +void tetris_drawScreen( int allScreen, T_TETRIS *data) +{ + int x1,y1,x2,y2; + int bCol; + int xOfs =5; + int yOfs=10; + int xOfs2=30; + int yOfs2=10; + int sx,sy; + int px,py; +// U32 oldfCol=0; // RAVI +// U32 oldbCol=0; // RAVI + int x = data->px; + int y = data->py; + +// char dbg[80]; // RAVI + + +#ifdef LSCREEN + //Calculate size based on the screen properties + sx = (SCREEN_SIZE_X-xOfs-xOfs2)/MAX_TET_X; + sy = (SCREEN_SIZE_Y-yOfs-yOfs2-Mmi_layout_softkeyHeight())/MAX_TET_Y; +#else + //Screen is very small - ensure shapes are as large as possible + sy = SCREEN_SIZE_Y/MAX_TET_Y; + sx = SCREEN_SIZE_X/MAX_TET_X; + if (sx >sy+1) + sx = sy+1; + xOfs = 5; + xOfs2 = SCREEN_SIZE_X - (sx*MAX_TET_X+xOfs); + yOfs = 0; + yOfs2 = SCREEN_SIZE_Y - (sy*MAX_TET_Y+xOfs); +#endif +//allScreen= 1; + + resources_setColour(COLOUR_GAME); + + if (allScreen) + { //redraw everything + x1 = 0; + y1 = 0; + x2 = MAX_TET_X; + y2 = MAX_TET_Y; + + } + else //just draw+1 box around current man + { + x1 = x-1; + y1 = y-1; + x2 = x+5; + y2 = y+4; + if (x1<1)x1=1; + if (y1<0)y1=0; + if (x2>MAX_TET_X-1) x2=MAX_TET_X-1; + if (y2>MAX_TET_Y-1) y2=MAX_TET_Y-1; + } + +#ifndef COLOURDISPLAY + dspl_Clear (xOfs+x1*sx,yOfs+y1*sy,xOfs+x2*sx-1,yOfs+y2*sy-1); +#endif + for (px = x1;px<x2;px++) + { + for (py = y1;py<y2;py++) + { + char boardChr = data->tetrisScreen[px][py]; + if ((px >= x ) && (px < x+4) && + (py >= y ) && (py < y+4)) + { + int shapeX,shapeY; + shapeX = px-x; + shapeY = py-y; + if (data->shapeStr[shapeX+shapeY*4] != ' ') + boardChr = data->shapeStr[shapeX+shapeY*4]; + } +#ifdef COLOURDISPLAY + switch (boardChr) + { + case '#': bCol = 0x00FFFFFF; break; + case ' ': bCol = 0x00404040; break; + case 'A': bCol = 0x000000FF; break; + case 'B': bCol = 0x0000FF00; break; + case 'C': bCol = 0x00FF0000; break; + case 'D': bCol = 0x0000FFFF; break; + case 'E': bCol = 0x00FF00FF; break; + case 'F': bCol = 0x00FFFF00; break; + default: bCol = 0x00FFFFFF; break; + } + dspl_SetBgdColour(bCol); + dspl_FillRect(xOfs+px*sx,yOfs+py*sy,sx,sy); +// dspl_SetFgdColour(0x00802040); + if (boardChr != ' ') + dspl_DrawRect (xOfs+px*sx,yOfs+py*sy,xOfs+px*sx+sx-1,yOfs+py*sy+sy-1); +#else + if (boardChr != ' ') + dspl_DrawRect (xOfs+px*sx,yOfs+py*sy,xOfs+px*sx+sx-1,yOfs+py*sy+sy-1); +#endif + } + } + if (data->gameOver) + { + int oldfCol = dspl_SetFgdColour(0x00FFFF00); + dspl_SetBgdColour(0x008000FF); + dspl_FillRect(xOfs+sx*3-5,yOfs+sy*3-5, xOfs+sx*3+5+6*4,yOfs+sy*3+5+18); + dspl_TextOut(xOfs+sx*3,yOfs+sy*3, 0, "GAME"); + dspl_TextOut(xOfs+sx*3,yOfs+sy*3+Mmi_layout_line_height()+4, 0, "OVER"); + dspl_SetFgdColour(oldfCol); + + } + resources_restoreColour(); +} + + + +int tetris_addShapeToScreen(T_TETRIS *data) +{ + int x,y; + for (x=0;x<4;x++) + for (y=0;y<4;y++) + if (data->shapeStr[x+y*4] != ' ') + { + if (data->tetrisScreen[data->px+x][data->py+y]!= ' ') + return(1); + else + data->tetrisScreen[data->px+x][data->py+y] = data->shapeStr[x+y*4]; + } + return(0); + +} + +int tetris_testForCompleteLines( T_TETRIS *data ) +{ + int nLines=0; + int x,y; + int blanks; + for (y=MAX_TET_Y-2;y>0;y--) + { + blanks = 0; + + for (x=1;x<MAX_TET_X-1;x++) + { + if (data->tetrisScreen[x][y] == ' ') + blanks++; + if (nLines >0) + data->tetrisScreen[x][y+nLines] = data->tetrisScreen[x][y]; + } + if (blanks==0) + nLines++; + + } + return (nLines); +} + + +void tetris_testGameOver( T_TETRIS *data ) +{ + int linesComplete; + + data->gameOver = tetris_addShapeToScreen(data); + if (!data->gameOver) + { + + linesComplete = tetris_testForCompleteLines(data); + if (linesComplete >0) + { + data->score = data->score + linesComplete*linesComplete; + data->linesToGo = data->linesToGo - linesComplete; + if (data->linesToGo <0) + tetris_getNextLevel(data); + } + tetris_getNextShape(data); + } +} + + + + + +void tetris_destroy (T_MFW_HND own_window) +{ + T_MFW_WIN * win = ((T_MFW_HDR *)own_window)->data; + T_TETRIS * data = (T_TETRIS *)win->user; + + TRACE_EVENT ("tetris_destroy()"); + + if (own_window == NULL) + { + TRACE_EVENT ("Error :- Called with NULL Pointer"); + return; + } + + if (data) + { + /* + * Exit ICON & KEYBOARD Handle + */ + kbdDelete (data->hKbd); + timDelete(data->info_tim); + /* + * Delete WIN Handler + */ + winDelete (data->win); + /* + * Free Memory + */ + FREE_MEMORY ((void *)data, sizeof (T_TETRIS)); + } +} + + + +int MmiTetrisStart(void ) +{ + MfwHnd hWin; + + MfwHnd hParentWin = mfwParent(mfwHeader()); + + TRACE_FUNCTION ("TetrisStart()"); + + hWin = TetrisCreate (hParentWin); + + if (hWin NEQ NULL) + { + SEND_EVENT (hWin, E_INIT, NULL, NULL); + } + + return 1; +} + +void initTetrisData( T_TETRIS *data ); +static MfwHnd TetrisCreate(MfwHnd hParentWin) +{ + T_TETRIS *psData = (T_TETRIS *)ALLOC_MEMORY (sizeof( T_TETRIS)); + MfwWin *psWin; + MfwWinAttr *win_attr; + + TRACE_FUNCTION ("TetrisCreate()"); + + if (psData == NULL) + { + TRACE_EVENT ("ALLOC_MEMORY() failed"); + return NULL; + } + + win_attr = (MfwWinAttr *) ALLOC_MEMORY (sizeof(MfwWinAttr)); + if (win_attr) + { + win_attr->win.px = 0; + win_attr->win.py = 0; + win_attr->win.sx = SCREEN_SIZE_X; + win_attr->win.sy = SCREEN_SIZE_Y; + } + psData->win = winCreate (hParentWin, win_attr, E_WIN_VISIBLE, (T_MFW_CB)TetrisWinCb); + if (psData->win == NULL) + { + mfwFree((U8*) win_attr, sizeof(MfwWinAttr)); + FREE_MEMORY((void *)psData, sizeof (T_TETRIS)); + return NULL; + } + psData->mmi_control.dialog = (T_DIALOG_FUNC)TetrisExecCb; + psData->mmi_control.data = psData; + psData->timeStep = 150; + psData->maxDropTime = 120; + psData->dropTime = psData->maxDropTime; + psWin = ((MfwHdr *)psData->win)->data; + psWin->user = (void *)psData; + psData->parent_win = hParentWin; + initTetrisData( psData ); + + return psData->win; +} + +static int TetrisWinCb (MfwEvt uiE, MfwWin *psWin) +{ + T_TETRIS *psData = (T_TETRIS *)psWin->user; + + TRACE_FUNCTION ("TetrisWinCb()"); + + switch (uiE) + { + case MfwWinVisible: + dspl_ResetWindow(); + dspl_SetBgdColour(0x000000FF); + dspl_Clear(0,0,SCREEN_SIZE_X-1,SCREEN_SIZE_Y-1 ); + tetris_drawScreen( 1, psData); + + break; + + case MfwWinFocussed: + case MfwWinDelete: + default: + return MFW_EVENT_REJECTED; + } + TRACE_FUNCTION ("TetrisWinCb() - end"); + return MFW_EVENT_CONSUMED; +} + +static void TetrisExecCb (MfwHnd hWin, USHORT uiE, SHORT iValue, void *pParameter) +{ + MfwWin *psWin = ((MfwHdr *) hWin)->data; + T_TETRIS *psData = (T_TETRIS *)psWin->user; + + TRACE_FUNCTION ("TetrisExecCb()"); + + switch (uiE) + { + case E_INIT: + + psData->info_tim = timCreate(hWin, psData->timeStep, (MfwCb)tetris_tim_cb); + psData->hKbd = kbdCreate(psData->win, KEY_ALL, (MfwCb)TetrisKbdCb); + timStart(psData->info_tim); + + /* */ + winShow(hWin); + + break; + + case E_EXIT: + tetris_destroy(hWin); + break; + + default: + return; + } + + return; +} + + +static int tetris_checkPos( T_TETRIS *data, int px, int py, char* shape) +{ + int x,y; + for (x=0;x<4;x++) + for (y=0;y<4;y++) + if (shape[x+y*4] != ' ') + { + if (data->tetrisScreen[px+x][py+y]!= ' ') + return(1); + } + return(0); +} + +static int tetris_moveDown(T_TETRIS *data,int *px, int *py) +{ + int cannotMove; + cannotMove = tetris_checkPos(data,*px,*py+1,data->shapeStr); + if (cannotMove) + return (1); + else + *py = *py+1; + return (0); +} +static int tetris_moveLeft(T_TETRIS *data, int *px, int *py) +{ + int cannotMove; + cannotMove = tetris_checkPos(data,*px-1,*py,data->shapeStr); + if (cannotMove) + return (1); + else + *px = *px-1; + return (0); +} +static int tetris_moveRight( T_TETRIS *data, int *px, int *py) +{ + int cannotMove; + cannotMove = tetris_checkPos(data,*px+1,*py,data->shapeStr); + if (cannotMove) + return (1); + else + *px = *px+1; + return (0); +} + +static int tetris_rotateLeft( T_TETRIS *data, int *px, int *py ) +{ + int i; + int cannotMove; + char tmpShape[20]; + tetris_getShape(data->shapeId,(data->rotateId+1)%4,tmpShape); + cannotMove = tetris_checkPos(data,*px,*py,tmpShape); + if (cannotMove) + return (1); + for (i=0;i<16;i++) + data->shapeStr[i] = tmpShape[i]; + data->rotateId = (data->rotateId+1)%4; + return (0); +} + +static int tetris_rotateRight( T_TETRIS *data, int *px, int *py ) +{ + int i; + int cannotMove; + char tmpShape[20]; + tetris_getShape(data->shapeId,(data->rotateId+3)%4,tmpShape); + cannotMove = tetris_checkPos(data,*px,*py,tmpShape); + if (cannotMove) + return (1); + for (i=0;i<16;i++) + data->shapeStr[i] = tmpShape[i]; + data->rotateId = (data->rotateId+3)%4; + return (0); +} +#define KEY_TET_MOVELEFT KCD_1 +#define KEY_TET_MOVERIGHT KCD_3 +#define KEY_TET_ROTATELEFT KCD_4 +#define KEY_TET_ROTATERIGHT KCD_6 +#define KEY_TET_DROPALL KCD_5 +#define KEY_TET_DROP KCD_2 + + +static int TetrisKbdCb (MfwEvt uiE, MfwKbd *psK) +{ + MfwHnd hWin = mfwParent(mfwHeader()); + MfwWin *psWin = ((MfwHdr *)hWin)->data; + T_TETRIS *psData = (T_TETRIS *)psWin->user; + int cannotMove=0; + + + TRACE_FUNCTION ("TetrisKbdCb()"); + /***************************Go-lite Optimization changes Start***********************/ + // Aug 16, 2004 REF: CRR 24323 Deepa M.D + TRACE_EVENT_P1 ("Code : %d",(int)psK->code); + /***************************Go-lite Optimization changes Start***********************/ + + + switch(psK->code ) + { + case KEY_TET_MOVELEFT: + case KCD_MNULEFT: + tetris_moveLeft(psData,&psData->px,&psData->py); + break; + case KEY_TET_MOVERIGHT: + case KCD_MNURIGHT: + tetris_moveRight(psData,&psData->px,&psData->py); + break; + case KEY_TET_ROTATELEFT: + tetris_rotateLeft(psData,&psData->px,&psData->py); + break; + case KCD_MNUSELECT: + case KEY_TET_ROTATERIGHT: + tetris_rotateRight(psData,&psData->px,&psData->py); + break; + case KEY_TET_DROP: + cannotMove = tetris_moveDown(psData,&psData->px,&psData->py); + if (cannotMove) + tetris_testGameOver( psData ); + break; + case KEY_TET_DROPALL: + while(tetris_moveDown(psData,&psData->px,&psData->py) ==0) + { //loop until we hit the bottom + + } + cannotMove = 1; + tetris_testGameOver( psData ); + break; + + + case KCD_RIGHT: + SEND_EVENT (hWin, E_EXIT, 0, 0); + return MFW_EVENT_CONSUMED; + + default: + + return MFW_EVENT_CONSUMED; + } + tetris_drawScreen(cannotMove, psData); + + TRACE_FUNCTION ("TetrisKbdCb()-end"); + return MFW_EVENT_CONSUMED; +} + +static int tetris_tim_cb(MfwEvt e, MfwTim *t) +{ + T_MFW_HND win = mfw_parent (mfw_header()); + T_MFW_WIN * win_data = ((T_MFW_HDR *)win)->data; + T_TETRIS * data = (T_TETRIS *)win_data->user; + static int nCycles = 0; + int cannotMove; + + char bfr[80]; + nCycles = nCycles+ data->timeStep; + if (nCycles > 1000) + { + nCycles = nCycles - 1000; + data->maxDropTime = data->maxDropTime-1; + } + TRACE_EVENT("tetris_tim_cb"); + data->dropTime = data->dropTime - data->timeStep; + if (data->dropTime < 0) + { + if (data->maxDropTime > 90) + data->dropTime = data->dropTime+75*10; + else if (data->maxDropTime > 60) + data->dropTime = data->dropTime+(60+(data->maxDropTime-60)/2)*10; + else if (data->maxDropTime > 30) + data->dropTime = data->dropTime+data->maxDropTime*10; + else + data->dropTime = data->dropTime+(data->maxDropTime+90)/4*10; + + if (data->dropTime < data->timeStep) + data->dropTime = data->timeStep; + else if (data->dropTime < data->timeStep) + data->dropTime = data->timeStep; + + resources_setColour(COLOUR_GAME); + sprintf(bfr,"%d ",data->score); + dspl_TextOut(SCREEN_SIZE_X-30 ,10, 0, bfr); + sprintf(bfr,"%d ",data->linesToGo); + dspl_TextOut(SCREEN_SIZE_X-30 ,12+Mmi_layout_line_height(), 0, bfr); + resources_restoreColour(); + cannotMove = tetris_moveDown(data,&data->px,&data->py); + tetris_getNextShapeId( );//To randomize it + + if (cannotMove) + { + tetris_testGameOver( data ); + } + tetris_drawScreen(cannotMove, data); + } + + timStart(data->info_tim); + TRACE_EVENT("tetris_tim_cb-end"); + return 1; +} + +T_MFW_HND Game_Info(T_MFW_HND parent_window) +{ + T_MFW_HND win = ShowGame_Information(parent_window); + TRACE_FUNCTION("ShowVerion()"); + if (win NEQ NULL) + { + SEND_EVENT (win, SHOWGAMEINFO_INIT, 0, 0); + } + + return win; +} + +static T_MFW_HND ShowGame_Information(MfwHnd parent_window) +{ + tShowInfo* data = (tShowInfo*)ALLOC_MEMORY (sizeof (tShowInfo)); + T_MFW_WIN* win; + + if (data EQ NULL) + { + return NULL; + } + + // Create the window handler + data->win = win_create (parent_window, 0, E_WIN_VISIBLE, NULL); + if (data->win EQ NULL) + { + return NULL; + } + + // connect the dialog data to the MFW window + data->mmi_control.dialog = (T_DIALOG_FUNC)ShowGame_DialogCB; + data->mmi_control.data = data; + + win = ((T_MFW_HDR *)data->win)->data; + win->user = (void *)data; + data->parent_win = parent_window; + + return data->win; +} + +static void ShowGame_DialogCB(T_MFW_HND win, USHORT e, SHORT identifier, void *parameter) +{ + T_MFW_WIN *win_data = ((T_MFW_HDR *) win)->data; + tShowInfo* data = (tShowInfo*) win_data->user; + /* SPR#1428 - SH - New editor data */ +#ifdef NEW_EDITOR +// T_AUI_EDITOR_DATA editor_data; // RAVI +#else /* NEW_EDITOR */ +// T_EDITOR_DATA editor_data; // RAVI +#endif /* NEW_EDITOR */ + + switch (e) + { + case SHOWGAMEINFO_INIT: + { + strcat(data->buffer, info); + + ShowGameInfoEditor(win); + + /* SPR#1428 - SH - New editor changes */ +#ifdef NEW_EDITOR + data->editor_data.Callback = (T_AUI_EDIT_CB)ShowInfoCB; + AUI_edit_Start(win, &(data->editor_data)); +#else /* NEW_EDITOR */ + data->editor_data.Callback = (T_EDIT_CB)ShowInfoCB; + editor_start(win, &(data->editor_data)); +#endif /* NEW_EDITOR */ + + winShow(data->win); + } + break; + + default: + { + TRACE_EVENT("Show Game Info : Unknown Event"); + } + break; + } +} + +void ShowGameInfoEditor (T_MFW_HND win) +{ + T_MFW_WIN *win_data = ((T_MFW_HDR *) win)->data; + tShowInfo* data = (tShowInfo*) win_data->user; + +/* SPR#1428 - SH - New editor changes */ +#ifdef NEW_EDITOR + AUI_edit_SetDefault(&data->editor_data); + AUI_edit_SetDisplay(&data->editor_data, NULL, COLOUR_EDITOR, EDITOR_FONT); + AUI_edit_SetTextStr(&data->editor_data, TxtSoftBack, TxtNull, TxtNull, NULL); + AUI_edit_SetEvents(&data->editor_data, 0, TRUE, FOREVER, NULL); + AUI_edit_SetBuffer(&data->editor_data, ATB_DCS_ASCII, (UBYTE *)data->buffer, 35); + AUI_edit_SetMode(&data->editor_data, ED_MODE_READONLY, ED_CURSOR_NONE); +#else /* NEW_EDITOR */ + editor_attr_init(&((data->editor_data).editor_attr), NULL, edtCurNone, 0, (char*)data->buffer, 35, COLOUR_EDITOR); + editor_data_init(&data->editor_data, NULL, TxtSoftBack, TxtNull, 0, 1, READ_ONLY_MODE, FOREVER); + + data->editor_data.hide = FALSE; + data->editor_data.Identifier = 0; + data->editor_data.TextString = NULL; + data->editor_data.destroyEditor = TRUE; +#endif /* NEW_EDITOR */ +} + +static void ShowInfoCB( T_MFW_HND win, USHORT Identifier,UBYTE reason) +{ + T_MFW_WIN *win_data = ( (T_MFW_HDR *) win )->data; + tShowInfo*data = (tShowInfo*) win_data->user; + + + TRACE_FUNCTION("ShowVerionCB()"); + + switch (reason ) + { + case INFO_KCD_LEFT: + case INFO_KCD_HUP: + case INFO_KCD_RIGHT: + + default: + showGameInfo_destroy(data->win); + break; + } +} + +/******************************************************************************* + + $Function: showVersion_destroy + + $Description: Destroys the editor + + $Returns: + + $Arguments: + +*******************************************************************************/ +void showGameInfo_destroy(MfwHnd own_window) +{ + T_MFW_WIN * win_data; + + tShowInfo* data = NULL; + + if (own_window == NULL) + { + TRACE_EVENT ("Error : showGameInfo_destroy called with NULL Pointer"); + return; + } + + + win_data = ((T_MFW_HDR *)own_window)->data; + if (win_data != NULL) + data = (tShowInfo*)win_data->user; + + if (data) + { + TRACE_EVENT ("calc_destroy()"); + + win_delete (data->win); + // Free Memory + FREE_MEMORY ((void *)data, sizeof (tShowInfo)); + + } + else + { + TRACE_EVENT ("calc_destroy() called twice"); + } +} +#endif +