/* Copyright (C) 1994-2003 Revolution Software Ltd * * 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 program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * $Header$ */ //============================================================================= // // Filename : menu.c // Created : 14th November 1996 // By : P.R.Porter // // Summary : This module holds the code for the driver96 menu system. // // Version Date By Description // ------- --------- --- ----------------------------------------------- // 1.0 15-Nov-96 PRP Initial Menu functions. // // 1.1 20-Nov-96 PRP Fixed the displaying of the bottom menu. // // 1.2 08-Nov-96 PRP Made the speed of the menubar dependent upon // the number of icons displayed. // // 1.3 24-Jan-97 PRP Changed the creation of menu icon sprite // surfaces depending upon whether the hardware // can stretch blit or not. Also, made it so // that the full size icon sprite is not stretch // blitted. // // 1.4 10-Feb-97 PRP Changed the creation of menu icon sprite // surfaces as the capability bits for the drivers // have been changed. Also, changed the error // reporting code (for directDraw) so that it // works. // // 1.5 04-Mar-97 PRP Tried to fix bug where running out of video // memory creating menubar icon surfaces. // // 1.6 16-Apr-97 PRP Got rid of bug where task switching causes // failure of icon draw. // // 1.7 23-Jul-97 PRP Checked error value of stretched blit. // // 1.8 13-Aug-97 PRP Added CloseMenuImmediately. // // 1.9 13-Aug-97 PRP Fixed spelling of above // // // Functions // --------- // // -------------------------------------------------------------------------- // // int32 ProcessMenu(void) // // This function should be called regularly to process the menuber system. // The rate at which this function is called will dictate how smooth the menu // system is. The menu cannot be drawn at a higher rate than the system // vbl rate. // // -------------------------------------------------------------------------- // // int32 ShowMenu(uint8 menu) // // This function brings the menu in to view. The choice of top or bottom menu // is defined by the parameter menu being either RDMENU_TOP or RDMENU_BOTTOM. // An error code is returned if the menu is already shown. // // -------------------------------------------------------------------------- // // int32 HideMenu(uint8 menu) // // This function hides the menu defined by the parameter menu. If the menu is // already hidden, an error code is returned. // // -------------------------------------------------------------------------- // // int32 SetMenuIcon(uint8 menu, uint8 pocket, uint8 *icon) // // This function sets a menubar icon to that passed in. If icon is NULL, the // pocket is cleared, otherwise, that icon is placed into pocket. The menu is // either RDMENU_TOP or RDMENU_BOTTOM. Valid error codes include // RDERR_INVALIDPOCKET if the pocket number does not exist. Initially, there // are 15 pockets. // // -------------------------------------------------------------------------- // // uint8 GetMenuStatus(uint8 menu) // // This function returns the status of the menu passed in menu. Return values // are RDMENU_OPENING, RDMENU_CLOSING, RDMENU_HIDDEN and RDMENU_SHOWN. // //============================================================================= #include "stdafx.h" #include "driver96.h" #include "menu.h" #include "d_draw.h" #include "render.h" #include "common/rect.h" #define MENUDEEP 40 #define MAXMENUANIMS 8 static uint8 menuStatus[2] = { RDMENU_HIDDEN, RDMENU_HIDDEN }; static byte *icons[2][RDMENU_MAXPOCKETS] = { { NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL }, { NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL } }; static uint8 pocketStatus[2][RDMENU_MAXPOCKETS] = { { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 } }; static uint8 iconCount = 0; void ClearIconArea(int menu, int pocket, ScummVM::Rect *r) { byte *dst; int i; r->top = menu * (RENDERDEEP + MENUDEEP) + (MENUDEEP - RDMENU_ICONDEEP) / 2; r->bottom = r->top + RDMENU_ICONDEEP; r->left = RDMENU_ICONSTART + pocket * (RDMENU_ICONWIDE + RDMENU_ICONSPACING); r->right = r->left + RDMENU_ICONWIDE; dst = lpBackBuffer + r->top * screenWide + r->left; for (i = 0; i < RDMENU_ICONDEEP; i++) { memset(dst, 0, RDMENU_ICONWIDE); dst += screenWide; } } int32 ProcessMenu(void) { byte *src, *dst; uint8 menu; uint8 i, j; uint8 complete; uint8 frameCount; int32 curx, xoff; int32 cury, yoff; ScummVM::Rect r1, r2; int32 delta; static int32 lastTime = 0; if (lastTime == 0) { lastTime = SVM_timeGetTime(); frameCount = 1; } else { delta = SVM_timeGetTime() - lastTime; if (delta > 250) { lastTime += delta; delta = 250; frameCount = 1; } else { frameCount = (uint8) ((iconCount + 8) * delta / 750); lastTime += frameCount * 750 / (iconCount + 8); } } while (frameCount-- > 0) { for (menu = RDMENU_TOP; menu <= RDMENU_BOTTOM; menu++) { if (menuStatus[menu] == RDMENU_OPENING) { // The menu is opening, so process it here complete = 1; // Propagate the animation from the first icon. for (i = RDMENU_MAXPOCKETS - 1; i > 0; i--) { pocketStatus[menu][i] = pocketStatus[menu][i - 1]; if (pocketStatus[menu][i] != MAXMENUANIMS) complete = 0; } if (pocketStatus[menu][i] != MAXMENUANIMS) complete = 0; // ... and animate the first icon if (pocketStatus[menu][0] != MAXMENUANIMS) pocketStatus[menu][0]++; // Check to see if the menu is fully open if (complete) menuStatus[menu] = RDMENU_SHOWN; } else if (menuStatus[menu] == RDMENU_CLOSING) { // The menu is closing, so process it here complete = 1; // Propagate the animation from the first icon. for (i = RDMENU_MAXPOCKETS - 1; i > 0; i--) { if (icons[menu][i] && pocketStatus[menu][i] != 0 && pocketStatus[menu][i - 1] == 0) { ClearIconArea(menu, i, &r1); UploadRect(&r1); } pocketStatus[menu][i] = pocketStatus[menu][i - 1]; if (pocketStatus[menu][i] != 0) complete = 0; } if (pocketStatus[menu][i] != 0) complete = 0; // ... and animate the first icon if (pocketStatus[menu][0] != 0) { pocketStatus[menu][0]--; if (pocketStatus[menu][0] == 0) { ClearIconArea(menu, 0, &r1); UploadRect(&r1); } } // Check to see if the menu is fully closed if (complete) menuStatus[menu] = RDMENU_HIDDEN; } } } // Does the menu need to be drawn? for (menu = RDMENU_TOP; menu <= RDMENU_BOTTOM; menu++) { if (menuStatus[menu] != RDMENU_HIDDEN) { // Draw the menu here. curx = RDMENU_ICONSTART + RDMENU_ICONWIDE / 2; cury = (MENUDEEP / 2) + (RENDERDEEP + MENUDEEP) * menu; for (i = 0; i < RDMENU_MAXPOCKETS; i++) { if (icons[menu][i]) { // Since we no longer clear the screen // after each frame we need to clear // the icon area. ClearIconArea(menu, i, &r1); if (pocketStatus[menu][i] == MAXMENUANIMS) { xoff = (RDMENU_ICONWIDE / 2); r2.left = curx - xoff; r2.right = r2.left + RDMENU_ICONWIDE; yoff = (RDMENU_ICONDEEP / 2); r2.top = cury - yoff; r2.bottom = r2.top + RDMENU_ICONDEEP; } else { xoff = (RDMENU_ICONWIDE / 2) * pocketStatus[menu][i] / MAXMENUANIMS; r2.left = curx - xoff; r2.right = curx + xoff; yoff = (RDMENU_ICONDEEP / 2) * pocketStatus[menu][i] / MAXMENUANIMS; r2.top = cury - yoff; r2.bottom = cury + yoff; } if (xoff != 0 && yoff != 0) { dst = lpBackBuffer + r2.top * screenWide + r2.left; src = icons[menu][i]; if (pocketStatus[menu][i] != MAXMENUANIMS) { SquashImage( dst, screenWide, r2.right - r2.left, r2.bottom - r2.top, src, RDMENU_ICONWIDE, RDMENU_ICONWIDE, RDMENU_ICONDEEP, NULL); } else { for (j = 0; j < RDMENU_ICONDEEP; j++) { memcpy(dst, src, RDMENU_ICONWIDE); src += RDMENU_ICONWIDE; dst += screenWide; } } UploadRect(&r1); } } curx += (RDMENU_ICONSPACING + RDMENU_ICONWIDE); r1.left += (RDMENU_ICONSPACING + RDMENU_ICONWIDE); r1.right += (RDMENU_ICONSPACING + RDMENU_ICONWIDE); } } } /* uint8 menu; uint8 i; uint8 complete; uint8 frameCount; // uint8 *src, *dst; int32 curx, xoff; int32 cury, yoff; HRESULT hr; RECT r; int32 delta; static int32 lastTime = 0; if (lastTime == 0) { lastTime = timeGetTime(); frameCount = 1; } else { delta = timeGetTime() - lastTime; if (delta > 250) { lastTime += delta; delta = 250; frameCount = 1; } else { frameCount = (uint8) ((iconCount+8) * delta / 750); lastTime += frameCount * 750 / (iconCount + 8); } } while (frameCount-- > 0) { for (menu = RDMENU_TOP; menu <= RDMENU_BOTTOM; menu++) { if (menuStatus[menu] == RDMENU_OPENING) { // The menu is opening, so process it here complete = 1; // Propagate the animation from the first icon. for (i=RDMENU_MAXPOCKETS-1; i>0; i--) { pocketStatus[menu][i] = pocketStatus[menu][i-1]; if (pocketStatus[menu][i] != MAXMENUANIMS) { complete = 0; } } if (pocketStatus[menu][i] != MAXMENUANIMS) complete = 0; // ... and animate the first icon if (pocketStatus[menu][0] != MAXMENUANIMS) pocketStatus[menu][0] += 1; // Check to see if the menu is fully open if (complete) { menuStatus[menu] = RDMENU_SHOWN; } } else if (menuStatus[menu] == RDMENU_CLOSING) { // The menu is closing, so process it here complete = 1; // Propagate the animation from the first icon. for (i=RDMENU_MAXPOCKETS-1; i>0; i--) { pocketStatus[menu][i] = pocketStatus[menu][i-1]; if (pocketStatus[menu][i] != 0) { complete = 0; } } if (pocketStatus[menu][i] != 0) complete = 0; // ... and animate the first icon if (pocketStatus[menu][0] != 0) pocketStatus[menu][0] -= 1; // Check to see if the menu is fully open if (complete) { menuStatus[menu] = RDMENU_HIDDEN; } } } } // Does the menu need to be drawn? for (menu = RDMENU_TOP; menu <= RDMENU_BOTTOM; menu++) { if (menuStatus[menu] != RDMENU_HIDDEN) { // Draw the menu here. curx = RDMENU_ICONSTART + RDMENU_ICONWIDE / 2; cury = (MENUDEEP / 2) + (RENDERDEEP + MENUDEEP) * menu; for (i=0; i RDMENU_BOTTOM) return RDERR_INVALIDMENU; // Check that the menu is not currently shown, or in the process of // being shown. if (menuStatus[menu] == RDMENU_SHOWN || menuStatus[menu] == RDMENU_OPENING) return RDERR_INVALIDCOMMAND; menuStatus[menu] = RDMENU_OPENING; return RD_OK; } int32 HideMenu(uint8 menu) { // Check for invalid menu parameter if (menu > RDMENU_BOTTOM) return(RDERR_INVALIDMENU); // Check that the menu is not currently hidden, or in the process of // being hidden. if (menuStatus[menu] == RDMENU_HIDDEN || menuStatus[menu] == RDMENU_CLOSING) return RDERR_INVALIDCOMMAND; menuStatus[menu] = RDMENU_CLOSING; return RD_OK; } int32 CloseMenuImmediately(void) { ScummVM::Rect r; int i; menuStatus[0] = RDMENU_HIDDEN; menuStatus[1] = RDMENU_HIDDEN; for (i = 0; i < RDMENU_MAXPOCKETS; i++) { if (icons[0][i]) { ClearIconArea(0, i, &r); UploadRect(&r); } if (icons[1][i]) { ClearIconArea(1, i, &r); UploadRect(&r); } } memset(pocketStatus, 0, sizeof(uint8) * 2 * RDMENU_MAXPOCKETS); return RD_OK; } int32 SetMenuIcon(uint8 menu, uint8 pocket, uint8 *icon) { ScummVM::Rect r; debug(5, "stub SetMenuIcon( %d, %d )", menu, pocket); // Check for invalid menu parameter. if (menu > RDMENU_BOTTOM) return RDERR_INVALIDMENU; // Check for invalid pocket parameter if (pocket >= RDMENU_MAXPOCKETS) return RDERR_INVALIDPOCKET; // If there is an icon in the requested menu/pocket, clear it out. if (icons[menu][pocket]) { iconCount--; free(icons[menu][pocket]); icons[menu][pocket] = NULL; ClearIconArea(menu, pocket, &r); UploadRect(&r); } // Only put the icon in the pocket if it is not NULL if (icon != NULL) { iconCount++; icons[menu][pocket] = (uint8 *) malloc(RDMENU_ICONWIDE * RDMENU_ICONDEEP); if (icons[menu][pocket] == NULL) return RDERR_OUTOFMEMORY; memcpy(icons[menu][pocket], icon, RDMENU_ICONWIDE * RDMENU_ICONDEEP); } return RD_OK; } uint8 GetMenuStatus(uint8 menu) { if (menu > RDMENU_BOTTOM) return RDMENU_HIDDEN; return menuStatus[menu]; }