#include "compiler.h"
#include <gx.h>
#include "bmpdata.h"
#include "scrnmng.h"
#include "scrndraw.h"
#include "vramhdl.h"
#include "menubase.h"
#include "nekop2.res"
#if defined(SUPPORT_SOFTKBD)
#include "softkbd.res"
#endif
#if defined(GX_DLL)
#pragma comment(lib, "gx.lib")
#endif
typedef struct {
BOOL rotate;
VRAMHDL vram;
UINT posx;
UINT posy;
int width;
int height;
long start;
long xalign;
long yalign;
#if defined(SUPPORT_SOFTKBD)
int allflash;
BMPFILE *kbd;
#endif
} SCRNMNG;
typedef struct {
int width;
int height;
} SCRNSTAT;
GXDisplayProperties gx_dp;
static SCRNMNG scrnmng;
static SCRNSTAT scrnstat;
static SCRNSURF scrnsurf;
static BYTE gx_disable = 1;
extern GXKeyList gx_keylist;
static const TCHAR errmsg[] = STRLITERAL("Error");
typedef struct {
int xalign;
int yalign;
int width;
int height;
int srcpos;
int dstpos;
} DRAWRECT;
static BOOL calcdrawrect(DRAWRECT *dr, VRAMHDL s, const RECT_T *rt) {
int pos;
dr->xalign = scrnmng.xalign;
dr->yalign = scrnmng.yalign;
dr->srcpos = 0;
dr->dstpos = scrnmng.start;
dr->width = min(scrnmng.width, s->width);
dr->height = min(scrnmng.height, s->height);
if (rt) {
pos = max(rt->left, 0);
dr->srcpos += pos;
dr->dstpos += pos * dr->xalign;
dr->width = min(rt->right, dr->width) - pos;
pos = max(rt->top, 0);
dr->srcpos += pos * s->width;
dr->dstpos += pos * dr->yalign;
dr->height = min(rt->bottom, dr->height) - pos;
}
if ((dr->width <= 0) || (dr->height <= 0)) {
return(FAILURE);
}
return(SUCCESS);
}
static void bmp16draw(BMPFILE *bf, BYTE *dst, int width, int height,
int xalign, int yalign) {
const BMPINFO *bi;
const BYTE *pal;
BMPDATA inf;
BYTE *src;
int bmpalign;
UINT pals;
UINT16 paltbl[16];
UINT c;
int x;
int y;
if (bf == NULL) {
goto b16d_err;
}
bi = (BMPINFO *)(bf + 1);
pal = (BYTE *)(bi + 1);
if (((bf->bfType[0] != 'B') && (bf->bfType[1] != 'M')) ||
(bmpdata_getinfo(bi, &inf) != SUCCESS) || (inf.bpp != 4)) {
goto b16d_err;
}
src = (BYTE *)bf + (LOADINTELDWORD(bf->bfOffBits));
bmpalign = bmpdata_getalign(bi);
if (inf.height > 0) {
src += (inf.height - 1) * bmpalign;
bmpalign *= -1;
}
else {
inf.height *= -1;
}
if ((width < inf.width) || (height < inf.height)) {
goto b16d_err;
}
dst += ((width - inf.width) / 2) * xalign;
dst += ((height - inf.height) / 2) * yalign;
yalign -= (inf.width * xalign);
pals = LOADINTELDWORD(bi->biClrUsed);
pals = min(pals, 16);
ZeroMemory(paltbl, sizeof(paltbl));
for (c=0; c<pals; c++) {
paltbl[c] = ((pal[c*4+2] & 0xf8) << 8) +
((pal[c*4+1] & 0xfc) << 3) + (pal[c*4+0] >> 3);
}
for (y=0; y<inf.height; y++) {
for (x=0; x<inf.width; x++) {
if (!(x & 1)) {
c = src[x >> 1] >> 4;
}
else {
c = src[x >> 1] & 15;
}
*(UINT16 *)dst = paltbl[c];
dst += xalign;
}
src += bmpalign;
dst += yalign;
}
b16d_err:
return;
}
// ----
void scrnmng_initialize(void) {
scrnstat.width = 640;
scrnstat.height = 400;
}
BOOL scrnmng_create(HWND hWnd, LONG width, LONG height) {
TCHAR msg[32];
if ((width <= 0) || (height <= 0)) {
return(FAILURE);
}
if (GXOpenDisplay(hWnd, GX_FULLSCREEN) == 0) {
MessageBox(hWnd, STRLITERAL("Couldn't GameX Object"),
errmsg, MB_OK | MB_ICONSTOP);
return(FAILURE);
}
gx_dp = GXGetDisplayProperties();
if (gx_dp.cBPP != 16) {
MessageBox(hWnd, STRLITERAL("Only 16bit color support..."),
errmsg, MB_OK | MB_ICONSTOP);
return(FAILURE);
}
if (!(gx_dp.ffFormat & kfDirect565)) {
MessageBox(hWnd, STRLITERAL("Only 16bit(565) support..."),
errmsg, MB_OK | MB_ICONSTOP);
return(FAILURE);
}
if ((gx_dp.cxWidth >= (DWORD)width) &&
(gx_dp.cyHeight >= (DWORD)height)) {
scrnmng.rotate = FALSE;
}
else if ((gx_dp.cxWidth >= (DWORD)height) &&
(gx_dp.cyHeight >= (DWORD)width)) {
scrnmng.rotate = TRUE;
}
else {
wsprintf(msg, STRLITERAL("Required %dx%d..."), width, height);
MessageBox(hWnd, msg, errmsg, MB_OK | MB_ICONSTOP);
return(FAILURE);
}
if (!scrnmng.rotate) {
scrnmng.posx = ((gx_dp.cxWidth - width) / 2);
scrnmng.posy = ((gx_dp.cyHeight - height) / 2);
scrnmng.start = scrnmng.posx * gx_dp.cbxPitch +
scrnmng.posy * gx_dp.cbyPitch;
scrnmng.width = width;
scrnmng.height = height;
scrnmng.xalign = gx_dp.cbxPitch;
scrnmng.yalign = gx_dp.cbyPitch;
}
else {
scrnmng.posx = ((gx_dp.cxWidth - height) / 2);
scrnmng.posy = ((gx_dp.cyHeight - width) / 2);
scrnmng.start = ((scrnmng.posy + width - 1) * gx_dp.cbyPitch) +
(scrnmng.posx * gx_dp.cbxPitch);
scrnmng.width = width;
scrnmng.height = height;
scrnmng.xalign = -gx_dp.cbyPitch;
scrnmng.yalign = gx_dp.cbxPitch;
}
gx_disable = 0;
scrnmng_clear(TRUE);
#if defined(SUPPORT_SOFTKBD)
scrnmng.kbd = (BMPFILE *)bmpdata_solvedata(np2kbd_bmp);
#endif
return(SUCCESS);
}
void scrnmng_destroy(void) {
gx_disable = 1;
GXCloseDisplay();
#if defined(SUPPORT_SOFTKBD)
if (scrnmng.kbd) {
_MFREE(scrnmng.kbd);
scrnmng.kbd = NULL;
}
#endif
}
#if defined(SUPPORT_SOFTKBD)
void scrnmng_allflash(void) {
scrnmng.allflash = 1;
}
#endif
RGB16 scrnmng_makepal16(RGB32 pal32) {
RGB16 ret;
ret = (pal32.p.r & 0xf8) << 8;
#if defined(SIZE_QVGA) && !defined(SIZE_VGATEST)
ret += (pal32.p.g & 0xfc) << (3 + 16);
#else
ret += (pal32.p.g & 0xfc) << 3;
#endif
ret += pal32.p.b >> 3;
return(ret);
}
void scrnmng_setwidth(int posx, int width) {
scrnstat.width = width;
}
void scrnmng_setheight(int posy, int height) {
scrnstat.height = height;
}
const SCRNSURF *scrnmng_surflock(void) {
if (gx_disable) {
return(NULL);
}
if (scrnmng.vram == NULL) {
scrnsurf.ptr = (BYTE *)GXBeginDraw() + scrnmng.start;
scrnsurf.xalign = scrnmng.xalign;
scrnsurf.yalign = scrnmng.yalign;
}
else {
scrnsurf.ptr = scrnmng.vram->ptr;
scrnsurf.xalign = 2;
scrnsurf.yalign = scrnmng.vram->yalign;
}
#if !defined(SIZE_VGATEST)
scrnsurf.width = min(scrnstat.width, 640);
scrnsurf.height = min(scrnstat.height, 400);
#else
scrnsurf.width = min(scrnstat.width, 320);
scrnsurf.height = min(scrnstat.height, 200);
#endif
scrnsurf.bpp = 16;
scrnsurf.extend = 0;
return(&scrnsurf);
}
static void draw_onmenu(void) {
RECT_T rt;
DRAWRECT dr;
const BYTE *p;
BYTE *q;
const BYTE *a;
int salign;
int dalign;
int r;
rt.left = 0;
rt.top = 0;
#if defined(SUPPORT_SOFTKBD)
if (!scrnmng.allflash) {
#endif
rt.right = min(scrnstat.width, 640);
rt.bottom = min(scrnstat.height, 400);
#if defined(SIZE_QVGA)
rt.right >>= 1;
rt.bottom >>= 1;
#endif
#if defined(SUPPORT_SOFTKBD)
}
else {
scrnmng.allflash = 0;
rt.right = scrnmng.width;
rt.bottom = scrnmng.height;
}
#endif
if (calcdrawrect(&dr, menuvram, &rt) != SUCCESS) {
return;
}
p = scrnmng.vram->ptr + (dr.srcpos * 2);
q = (BYTE *)GXBeginDraw() + dr.dstpos;
a = menuvram->alpha + dr.srcpos;
salign = menuvram->width - dr.width;
dalign = dr.yalign - (dr.width * dr.xalign);
do {
r = dr.width;
do {
if (a[0] == 0) {
*(UINT16 *)q = *(UINT16 *)p;
}
p += 2;
q += dr.xalign;
a += 1;
} while(--r);
p += salign * 2;
q += dalign;
a += salign;
} while(--dr.height);
GXEndDraw();
}
void scrnmng_surfunlock(const SCRNSURF *surf) {
if (surf) {
if (scrnmng.vram == NULL) {
#if defined(SUPPORT_SOFTKBD)
if (scrnmng.allflash) {
scrnmng.allflash = 0;
}
bmp16draw(scrnmng.kbd, surf->ptr + (surf->yalign * 200), 320, 40,
surf->xalign, surf->yalign);
#endif
GXEndDraw();
}
else {
if (menuvram) {
draw_onmenu();
}
}
}
}
// ---- for wince
void scrnmng_enable(BOOL enable) {
if (enable) {
gx_disable &= ~2;
}
else {
gx_disable |= 2;
}
}
BOOL scrnmng_mousepos(LPARAM *lp) {
UINT x;
UINT y;
if (!scrnmng.rotate) {
x = LOWORD(*lp) - scrnmng.posx;
y = HIWORD(*lp) - scrnmng.posy;
}
else {
x = scrnmng.width - HIWORD(*lp) - 1 - scrnmng.posy;
y = LOWORD(*lp) - scrnmng.posx;
}
if ((x >= (UINT)scrnmng.width) || (y >= (UINT)scrnmng.height)) {
return(FAILURE);
}
*lp = (x & 0xffff) | ((y << 16) & 0xffff0000);
return(SUCCESS);
}
void scrnmng_clear(BOOL logo) {
BMPFILE *bf;
BYTE *p;
BYTE *q;
int y;
int x;
long yalign;
if (gx_disable) {
return;
}
bf = NULL;
if (logo) {
bf = (BMPFILE *)bmpdata_solvedata(nekop2_bmp);
}
p = (BYTE *)GXBeginDraw();
q = p;
y = gx_dp.cyHeight;
yalign = gx_dp.cbyPitch - (gx_dp.cbxPitch * gx_dp.cxWidth);
do {
x = gx_dp.cxWidth;
do {
*(UINT16 *)q = 0;
q += gx_dp.cbxPitch;
} while(--x);
q += yalign;
} while(--y);
bmp16draw(bf, p + scrnmng.start, scrnmng.width, scrnmng.height,
scrnmng.xalign, scrnmng.yalign);
GXEndDraw();
if (bf) {
_MFREE(bf);
}
}
void scrnmng_keybinds(void) {
short tmp;
if (scrnmng.rotate) {
tmp = gx_keylist.vkLeft;
gx_keylist.vkLeft = gx_keylist.vkDown;
gx_keylist.vkDown = gx_keylist.vkRight;
gx_keylist.vkRight = gx_keylist.vkUp;
gx_keylist.vkUp = tmp;
}
}
// ---- for menubase
BOOL scrnmng_entermenu(SCRNMENU *smenu) {
if (smenu == NULL) {
goto smem_err;
}
vram_destroy(scrnmng.vram);
scrnmng.vram = vram_create(scrnmng.width, scrnmng.height, FALSE, 16);
if (scrnmng.vram == NULL) {
goto smem_err;
}
scrndraw_redraw();
#if defined(SUPPORT_SOFTKBD)
bmp16draw(scrnmng.kbd, scrnmng.vram->ptr + (640 * 200), 320, 40, 2, 640);
#endif
smenu->width = scrnmng.width;
smenu->height = scrnmng.height;
smenu->bpp = 16;
return(SUCCESS);
smem_err:
return(FAILURE);
}
void scrnmng_leavemenu(void) {
VRAM_RELEASE(scrnmng.vram);
}
void scrnmng_menudraw(const RECT_T *rct) {
DRAWRECT dr;
const BYTE *p;
const BYTE *q;
BYTE *r;
BYTE *a;
int salign;
int dalign;
int x;
if ((gx_disable) || (menuvram == NULL)) {
return;
}
if (calcdrawrect(&dr, menuvram, rct) != SUCCESS) {
return;
}
p = scrnmng.vram->ptr + (dr.srcpos * 2);
q = menuvram->ptr + (dr.srcpos * 2);
r = (BYTE *)GXBeginDraw() + dr.dstpos;
a = menuvram->alpha + dr.srcpos;
salign = menuvram->width;
dalign = dr.yalign - (dr.width * dr.xalign);
do {
x = 0;
do {
if (a[x]) {
if (a[x] & 2) {
*(UINT16 *)r = *(UINT16 *)(q + x * 2);
}
else {
a[x] = 0;
*(UINT16 *)r = *(UINT16 *)(p + x * 2);
}
}
r += dr.xalign;
} while(++x < dr.width);
p += salign * 2;
q += salign * 2;
r += dalign;
a += salign;
} while(--dr.height);
GXEndDraw();
}
RetroPC.NET-CVS <cvs@retropc.net>