This was caused by character set incompatibility between message and map windows. Apparently, Nethack is using IBM character set (CP 437) but fonts were create for Windows ANSI codepage. I fixed most of it by changing character set of the fonts except for the Rogue level. I had to make changes to src/drawing.c for Rogue level since most of Windows fonts are not capable of displaying control characters (char. code<0x20) And of course, disabling IBMGraphics option fixes it all.
766 lines
20 KiB
C
766 lines
20 KiB
C
/* Copyright (C) 2001 by Alex Kompel <shurikk@pacbell.net> */
|
|
/* NetHack may be freely redistributed. See license for details. */
|
|
|
|
#include "winMS.h"
|
|
#include "resource.h"
|
|
#include "mhmap.h"
|
|
#include "mhmsg.h"
|
|
#include "mhinput.h"
|
|
|
|
#define NHMAP_FONT_NAME TEXT("Terminal")
|
|
#define MAXWINDOWTEXT 255
|
|
|
|
extern short glyph2tile[];
|
|
|
|
/* map window data */
|
|
typedef struct mswin_nethack_map_window {
|
|
int map[COLNO][ROWNO]; /* glyph map */
|
|
|
|
int mapMode; /* current map mode */
|
|
boolean bAsciiMode; /* switch ASCII/tiled mode */
|
|
int xPos, yPos; /* scroll position */
|
|
int xPageSize, yPageSize; /* scroll page size */
|
|
int xCur, yCur; /* position of the cursor */
|
|
int xScrTile, yScrTile; /* size of display tile */
|
|
POINT map_orig; /* map origin point */
|
|
|
|
HFONT hMapFont; /* font for ASCII mode */
|
|
} NHMapWindow, *PNHMapWindow;
|
|
|
|
static TCHAR szNHMapWindowClass[] = TEXT("MSNethackMapWndClass");
|
|
LRESULT CALLBACK MapWndProc(HWND, UINT, WPARAM, LPARAM);
|
|
static void register_map_window_class();
|
|
static void onMSNHCommand(HWND hWnd, WPARAM wParam, LPARAM lParam);
|
|
static void onMSNH_VScroll(HWND hWnd, WPARAM wParam, LPARAM lParam);
|
|
static void onMSNH_HScroll(HWND hWnd, WPARAM wParam, LPARAM lParam);
|
|
static void onPaint(HWND hWnd);
|
|
static void onCreate(HWND hWnd, WPARAM wParam, LPARAM lParam);
|
|
static void nhcoord2display(PNHMapWindow data, int x, int y, LPRECT lpOut);
|
|
static void nhglyph2charcolor(short glyph, uchar* ch, int* color);
|
|
static COLORREF nhcolor_to_RGB(int c);
|
|
|
|
HWND mswin_init_map_window () {
|
|
static int run_once = 0;
|
|
HWND ret;
|
|
|
|
if( !run_once ) {
|
|
register_map_window_class();
|
|
run_once = 1;
|
|
}
|
|
|
|
ret = CreateWindow(
|
|
szNHMapWindowClass, /* registered class name */
|
|
NULL, /* window name */
|
|
WS_CHILD | WS_HSCROLL | WS_VSCROLL | WS_CLIPSIBLINGS, /* window style */
|
|
0, /* horizontal position of window - set it later */
|
|
0, /* vertical position of window - set it later */
|
|
0, /* window width - set it later */
|
|
0, /* window height - set it later*/
|
|
GetNHApp()->hMainWnd, /* handle to parent or owner window */
|
|
NULL, /* menu handle or child identifier */
|
|
GetNHApp()->hApp, /* handle to application instance */
|
|
NULL ); /* window-creation data */
|
|
if( !ret ) {
|
|
panic("Cannot create map window");
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
void mswin_map_stretch(HWND hWnd, LPSIZE lpsz, BOOL redraw)
|
|
{
|
|
PNHMapWindow data;
|
|
RECT client_rt;
|
|
SCROLLINFO si;
|
|
SIZE wnd_size;
|
|
LOGFONT lgfnt;
|
|
|
|
/* check arguments */
|
|
if( !IsWindow(hWnd) ||
|
|
!lpsz ||
|
|
lpsz->cx<=0 ||
|
|
lpsz->cy<=0 ) return;
|
|
|
|
/* calculate window size */
|
|
GetClientRect(hWnd, &client_rt);
|
|
wnd_size.cx = client_rt.right - client_rt.left;
|
|
wnd_size.cy = client_rt.bottom - client_rt.top;
|
|
|
|
/* set new screen tile size */
|
|
data = (PNHMapWindow)GetWindowLong(hWnd, GWL_USERDATA);
|
|
data->xScrTile = ((data->mapMode==NHMAP_VIEW_FIT_TO_SCREEN)? wnd_size.cx : lpsz->cx) / COLNO;
|
|
data->yScrTile = ((data->mapMode==NHMAP_VIEW_FIT_TO_SCREEN)? wnd_size.cy : lpsz->cy) / ROWNO;
|
|
|
|
/* set map origin point */
|
|
data->map_orig.x = max(0, client_rt.left + (wnd_size.cx - data->xScrTile*COLNO)/2 );
|
|
data->map_orig.y = max(0, client_rt.top + (wnd_size.cy - data->yScrTile*ROWNO)/2 );
|
|
|
|
data->map_orig.x -= data->map_orig.x % data->xScrTile;
|
|
data->map_orig.y -= data->map_orig.y % data->yScrTile;
|
|
|
|
/* adjust horizontal scroll bar */
|
|
if( data->mapMode==NHMAP_VIEW_FIT_TO_SCREEN )
|
|
data->xPageSize = COLNO+1; /* disable scroll bar */
|
|
else
|
|
data->xPageSize = wnd_size.cx/data->xScrTile;
|
|
if( wnd_size.cx/data->xScrTile >= COLNO ) {
|
|
data->xPos = 0;
|
|
GetNHApp()->bNoHScroll = TRUE;
|
|
} else {
|
|
GetNHApp()->bNoHScroll = FALSE;
|
|
data->xPos = max(0, min(COLNO, u.ux - data->xPageSize/2));
|
|
}
|
|
|
|
si.cbSize = sizeof(si);
|
|
si.fMask = SIF_RANGE | SIF_PAGE | SIF_POS;
|
|
si.nMin = 0;
|
|
si.nMax = COLNO;
|
|
si.nPage = data->xPageSize;
|
|
si.nPos = data->xPos;
|
|
SetScrollInfo(hWnd, SB_HORZ, &si, TRUE);
|
|
|
|
/* adjust vertical scroll bar */
|
|
if( data->mapMode==NHMAP_VIEW_FIT_TO_SCREEN )
|
|
data->yPageSize = ROWNO+1; /* disable scroll bar */
|
|
else
|
|
data->yPageSize = wnd_size.cy/data->yScrTile;
|
|
|
|
if( wnd_size.cy/data->yScrTile >= ROWNO ) {
|
|
data->yPos = 0;
|
|
GetNHApp()->bNoVScroll = TRUE;
|
|
} else {
|
|
GetNHApp()->bNoVScroll = FALSE;
|
|
data->yPos = max(0, min(ROWNO, u.uy - data->yPageSize/2));
|
|
}
|
|
|
|
si.cbSize = sizeof(si);
|
|
si.fMask = SIF_RANGE | SIF_PAGE | SIF_POS;
|
|
si.nMin = 0;
|
|
si.nMax = ROWNO;
|
|
si.nPage = data->yPageSize;
|
|
si.nPos = data->yPos;
|
|
SetScrollInfo(hWnd, SB_VERT, &si, TRUE);
|
|
|
|
/* create font */
|
|
if( data->hMapFont ) DeleteObject(data->hMapFont);
|
|
ZeroMemory(&lgfnt, sizeof(lgfnt));
|
|
lgfnt.lfHeight = -data->yScrTile; // height of font
|
|
lgfnt.lfWidth = -data->xScrTile; // average character width
|
|
lgfnt.lfEscapement = 0; // angle of escapement
|
|
lgfnt.lfOrientation = 0; // base-line orientation angle
|
|
lgfnt.lfWeight = FW_NORMAL; // font weight
|
|
lgfnt.lfItalic = FALSE; // italic attribute option
|
|
lgfnt.lfUnderline = FALSE; // underline attribute option
|
|
lgfnt.lfStrikeOut = FALSE; // strikeout attribute option
|
|
lgfnt.lfCharSet = OEM_CHARSET; // character set identifier
|
|
lgfnt.lfOutPrecision = OUT_DEFAULT_PRECIS; // output precision
|
|
lgfnt.lfClipPrecision = CLIP_DEFAULT_PRECIS; // clipping precision
|
|
lgfnt.lfQuality = DEFAULT_QUALITY; // output quality
|
|
lgfnt.lfPitchAndFamily = FIXED_PITCH; // pitch and family
|
|
_tcscpy(lgfnt.lfFaceName, NHMAP_FONT_NAME);
|
|
data->hMapFont = CreateFontIndirect(&lgfnt);
|
|
|
|
mswin_cliparound(data->xCur, data->yCur);
|
|
|
|
if(redraw) InvalidateRect(hWnd, NULL, TRUE);
|
|
}
|
|
|
|
/* set map mode */
|
|
int mswin_map_mode(HWND hWnd, int mode)
|
|
{
|
|
PNHMapWindow data;
|
|
int oldMode;
|
|
SIZE mapSize;
|
|
|
|
data = (PNHMapWindow)GetWindowLong(hWnd, GWL_USERDATA);
|
|
if( mode == data->mapMode ) return mode;
|
|
|
|
oldMode = data->mapMode;
|
|
data->mapMode = mode;
|
|
|
|
switch( data->mapMode ) {
|
|
|
|
case NHMAP_VIEW_ASCII4x6:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 4*COLNO;
|
|
mapSize.cy = 6*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_ASCII6x8:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 6*COLNO;
|
|
mapSize.cy = 8*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_ASCII8x8:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 8*COLNO;
|
|
mapSize.cy = 8*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_ASCII16x8:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 16*COLNO;
|
|
mapSize.cy = 8*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_ASCII7x12:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 7*COLNO;
|
|
mapSize.cy = 12*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_ASCII8x12:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 8*COLNO;
|
|
mapSize.cy = 12*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_ASCII16x12:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 16*COLNO;
|
|
mapSize.cy = 12*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_ASCII12x16:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 12*COLNO;
|
|
mapSize.cy = 16*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_ASCII10x18:
|
|
data->bAsciiMode = TRUE;
|
|
mapSize.cx = 10*COLNO;
|
|
mapSize.cy = 18*ROWNO;
|
|
break;
|
|
|
|
case NHMAP_VIEW_FIT_TO_SCREEN: {
|
|
RECT client_rt;
|
|
GetClientRect(hWnd, &client_rt);
|
|
mapSize.cx = client_rt.right - client_rt.left;
|
|
mapSize.cy = client_rt.bottom - client_rt.top;
|
|
|
|
data->bAsciiMode = TRUE;
|
|
} break;
|
|
|
|
case NHMAP_VIEW_TILES:
|
|
default:
|
|
data->bAsciiMode = FALSE;
|
|
mapSize.cx = TILE_X*COLNO;
|
|
mapSize.cy = TILE_Y*ROWNO;
|
|
break;
|
|
}
|
|
|
|
mswin_map_stretch(hWnd, &mapSize, TRUE);
|
|
|
|
return oldMode;
|
|
}
|
|
|
|
/* register window class for map window */
|
|
void register_map_window_class()
|
|
{
|
|
WNDCLASS wcex;
|
|
ZeroMemory( &wcex, sizeof(wcex));
|
|
|
|
/* window class */
|
|
wcex.style = CS_NOCLOSE;
|
|
wcex.lpfnWndProc = (WNDPROC)MapWndProc;
|
|
wcex.cbClsExtra = 0;
|
|
wcex.cbWndExtra = 0;
|
|
wcex.hInstance = GetNHApp()->hApp;
|
|
wcex.hIcon = NULL;
|
|
wcex.hCursor = LoadCursor(NULL, IDC_ARROW);
|
|
wcex.hbrBackground = CreateSolidBrush(RGB(0, 0, 0)); /* set backgroup here */
|
|
wcex.lpszMenuName = NULL;
|
|
wcex.lpszClassName = szNHMapWindowClass;
|
|
|
|
if( !RegisterClass(&wcex) ) {
|
|
panic("cannot register Map window class");
|
|
}
|
|
}
|
|
|
|
/* map window procedure */
|
|
LRESULT CALLBACK MapWndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
PNHMapWindow data;
|
|
|
|
data = (PNHMapWindow)GetWindowLong(hWnd, GWL_USERDATA);
|
|
switch (message)
|
|
{
|
|
case WM_CREATE:
|
|
onCreate( hWnd, wParam, lParam );
|
|
break;
|
|
|
|
case WM_MSNH_COMMAND:
|
|
onMSNHCommand(hWnd, wParam, lParam);
|
|
break;
|
|
|
|
case WM_PAINT:
|
|
onPaint(hWnd);
|
|
break;
|
|
|
|
case WM_SETFOCUS:
|
|
/* transfer focus back to the main window */
|
|
SetFocus(GetNHApp()->hMainWnd);
|
|
break;
|
|
|
|
case WM_HSCROLL:
|
|
onMSNH_HScroll(hWnd, wParam, lParam);
|
|
break;
|
|
|
|
case WM_VSCROLL:
|
|
onMSNH_VScroll(hWnd, wParam, lParam);
|
|
break;
|
|
|
|
case WM_SIZE:
|
|
{
|
|
SIZE size;
|
|
|
|
if( data->mapMode == NHMAP_VIEW_FIT_TO_SCREEN ) {
|
|
size.cx = LOWORD(lParam);
|
|
size.cy = HIWORD(lParam);
|
|
} else {
|
|
/* mapping factor is unchaged we just need to adjust scroll bars */
|
|
size.cx = data->xScrTile*COLNO;
|
|
size.cy = data->yScrTile*ROWNO;
|
|
}
|
|
mswin_map_stretch(hWnd, &size, TRUE);
|
|
}
|
|
break;
|
|
|
|
case WM_LBUTTONDOWN:
|
|
NHEVENT_MS(
|
|
max(0, min(COLNO, data->xPos + (LOWORD(lParam)-data->map_orig.x)/data->xScrTile)),
|
|
max(0, min(ROWNO, data->yPos + (HIWORD(lParam)-data->map_orig.y)/data->yScrTile))
|
|
);
|
|
break;
|
|
|
|
case WM_DESTROY:
|
|
if( data->hMapFont ) DeleteObject(data->hMapFont);
|
|
free(data);
|
|
SetWindowLong(hWnd, GWL_USERDATA, (LONG)0);
|
|
break;
|
|
|
|
default:
|
|
return DefWindowProc(hWnd, message, wParam, lParam);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* on WM_COMMAND */
|
|
void onMSNHCommand(HWND hWnd, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
PNHMapWindow data;
|
|
RECT rt;
|
|
|
|
data = (PNHMapWindow)GetWindowLong(hWnd, GWL_USERDATA);
|
|
switch(wParam) {
|
|
case MSNH_MSG_PRINT_GLYPH:
|
|
{
|
|
PMSNHMsgPrintGlyph msg_data = (PMSNHMsgPrintGlyph)lParam;
|
|
data->map[msg_data->x][msg_data->y] = msg_data->glyph;
|
|
|
|
/* invalidate the update area */
|
|
nhcoord2display(data, msg_data->x, msg_data->y, &rt);
|
|
InvalidateRect(hWnd, &rt, TRUE);
|
|
}
|
|
break;
|
|
|
|
case MSNH_MSG_CLIPAROUND:
|
|
{
|
|
PMSNHMsgClipAround msg_data = (PMSNHMsgClipAround)lParam;
|
|
int x, y;
|
|
|
|
/* get page size and center horizontally on x-position*/
|
|
if( !GetNHApp()->bNoHScroll ) {
|
|
x = max(0, min(COLNO, msg_data->x - data->xPageSize/2));
|
|
SendMessage( hWnd, WM_HSCROLL, (WPARAM)MAKELONG(SB_THUMBTRACK, x), (LPARAM)NULL );
|
|
}
|
|
|
|
/* get page size and center vertically on y-position*/
|
|
if( !GetNHApp()->bNoVScroll ) {
|
|
y = max(0, min(ROWNO, msg_data->y - data->yPageSize/2));
|
|
SendMessage( hWnd, WM_VSCROLL, (WPARAM)MAKELONG(SB_THUMBTRACK, y), (LPARAM)NULL );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case MSNH_MSG_CLEAR_WINDOW:
|
|
{
|
|
int i, j;
|
|
for(i=0; i<COLNO; i++)
|
|
for(j=0; j<ROWNO; j++) {
|
|
data->map[i][j] = -1;
|
|
}
|
|
InvalidateRect(hWnd, NULL, TRUE);
|
|
} break;
|
|
|
|
case MSNH_MSG_CURSOR:
|
|
{
|
|
PMSNHMsgCursor msg_data = (PMSNHMsgCursor)lParam;
|
|
HDC hdc;
|
|
RECT rt;
|
|
|
|
/* move focus rectangle at the cursor postion */
|
|
hdc = GetDC(hWnd);
|
|
|
|
nhcoord2display(data, data->xCur, data->yCur, &rt);
|
|
if( data->bAsciiMode ) {
|
|
PatBlt(hdc, rt.left, rt.top, rt.right-rt.left, rt.bottom-rt.top, DSTINVERT);
|
|
} else {
|
|
DrawFocusRect(hdc, &rt);
|
|
}
|
|
|
|
data->xCur = msg_data->x;
|
|
data->yCur = msg_data->y;
|
|
|
|
nhcoord2display(data, data->xCur, data->yCur, &rt);
|
|
if( data->bAsciiMode ) {
|
|
PatBlt(hdc, rt.left, rt.top, rt.right-rt.left, rt.bottom-rt.top, DSTINVERT);
|
|
} else {
|
|
DrawFocusRect(hdc, &rt);
|
|
}
|
|
|
|
ReleaseDC(hWnd, hdc);
|
|
} break;
|
|
}
|
|
}
|
|
|
|
/* on WM_CREATE */
|
|
void onCreate(HWND hWnd, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
PNHMapWindow data;
|
|
int i,j;
|
|
|
|
/* set window data */
|
|
data = (PNHMapWindow)malloc(sizeof(NHMapWindow));
|
|
if( !data ) panic("out of memory");
|
|
|
|
ZeroMemory(data, sizeof(NHMapWindow));
|
|
for(i=0; i<COLNO; i++)
|
|
for(j=0; j<ROWNO; j++) {
|
|
data->map[i][j] = -1;
|
|
}
|
|
|
|
data->bAsciiMode = FALSE;
|
|
|
|
data->xScrTile = TILE_X;
|
|
data->yScrTile = TILE_Y;
|
|
|
|
SetWindowLong(hWnd, GWL_USERDATA, (LONG)data);
|
|
}
|
|
|
|
/* on WM_PAINT */
|
|
void onPaint(HWND hWnd)
|
|
{
|
|
PNHMapWindow data;
|
|
PAINTSTRUCT ps;
|
|
HDC hDC;
|
|
HDC tileDC;
|
|
HGDIOBJ saveBmp;
|
|
RECT paint_rt;
|
|
int i, j;
|
|
|
|
/* get window data */
|
|
data = (PNHMapWindow)GetWindowLong(hWnd, GWL_USERDATA);
|
|
|
|
hDC = BeginPaint(hWnd, &ps);
|
|
|
|
/* calculate paint rectangle */
|
|
if( !IsRectEmpty(&ps.rcPaint) ) {
|
|
/* calculate paint rectangle */
|
|
paint_rt.left = max(data->xPos + (ps.rcPaint.left - data->map_orig.x)/data->xScrTile, 0);
|
|
paint_rt.top = max(data->yPos + (ps.rcPaint.top - data->map_orig.y)/data->yScrTile, 0);
|
|
paint_rt.right = min(data->xPos + (ps.rcPaint.right - data->map_orig.x)/data->xScrTile+1, COLNO);
|
|
paint_rt.bottom = min(data->yPos + (ps.rcPaint.bottom - data->map_orig.y)/data->yScrTile+1, ROWNO);
|
|
|
|
if( data->bAsciiMode
|
|
#ifdef REINCARNATION
|
|
|| Is_rogue_level(&u.uz)
|
|
/* You enter a VERY primitive world! */
|
|
#endif
|
|
) {
|
|
HGDIOBJ oldFont;
|
|
|
|
oldFont = SelectObject(hDC, data->hMapFont);
|
|
SetBkMode(hDC, TRANSPARENT);
|
|
|
|
/* draw the map */
|
|
for(i=paint_rt.left; i<paint_rt.right; i++)
|
|
for(j=paint_rt.top; j<paint_rt.bottom; j++)
|
|
if(data->map[i][j]>0) {
|
|
uchar ch;
|
|
TCHAR wch;
|
|
RECT glyph_rect;
|
|
int color;
|
|
|
|
nhglyph2charcolor(data->map[i][j], &ch, &color);
|
|
|
|
if( color == NO_COLOR ) continue;
|
|
else SetTextColor( hDC, nhcolor_to_RGB(color) );
|
|
|
|
nhcoord2display(data, i, j, &glyph_rect);
|
|
DrawText(hDC,
|
|
NH_A2W(&ch, &wch, 1),
|
|
1,
|
|
&glyph_rect,
|
|
DT_CENTER | DT_VCENTER | DT_NOPREFIX
|
|
);
|
|
}
|
|
SelectObject(hDC, oldFont);
|
|
} else {
|
|
/* prepare tiles DC for mapping */
|
|
tileDC = CreateCompatibleDC(hDC);
|
|
saveBmp = SelectObject(tileDC, GetNHApp()->bmpTiles);
|
|
|
|
/* draw the map */
|
|
for(i=paint_rt.left; i<paint_rt.right; i++)
|
|
for(j=paint_rt.top; j<paint_rt.bottom; j++)
|
|
if(data->map[i][j]>0) {
|
|
short ntile;
|
|
int t_x, t_y;
|
|
RECT glyph_rect;
|
|
|
|
ntile = glyph2tile[ data->map[i][j] ];
|
|
t_x = (ntile % TILES_PER_LINE)*TILE_X;
|
|
t_y = (ntile / TILES_PER_LINE)*TILE_Y;
|
|
|
|
nhcoord2display(data, i, j, &glyph_rect);
|
|
|
|
StretchBlt(
|
|
hDC,
|
|
glyph_rect.left,
|
|
glyph_rect.top,
|
|
data->xScrTile,
|
|
data->yScrTile,
|
|
tileDC,
|
|
t_x,
|
|
t_y,
|
|
TILE_X,
|
|
TILE_Y,
|
|
SRCCOPY
|
|
);
|
|
}
|
|
SelectObject(tileDC, saveBmp);
|
|
DeleteDC(tileDC);
|
|
}
|
|
|
|
/* draw focus rect */
|
|
nhcoord2display(data, data->xCur, data->yCur, &paint_rt);
|
|
if( data->bAsciiMode ) {
|
|
PatBlt( hDC,
|
|
paint_rt.left, paint_rt.top,
|
|
paint_rt.right-paint_rt.left, paint_rt.bottom-paint_rt.top,
|
|
DSTINVERT );
|
|
} else {
|
|
DrawFocusRect(hDC, &paint_rt);
|
|
}
|
|
}
|
|
EndPaint(hWnd, &ps);
|
|
}
|
|
|
|
/* on WM_VSCROLL */
|
|
void onMSNH_VScroll(HWND hWnd, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
PNHMapWindow data;
|
|
SCROLLINFO si;
|
|
int yNewPos;
|
|
int yDelta;
|
|
|
|
/* get window data */
|
|
data = (PNHMapWindow)GetWindowLong(hWnd, GWL_USERDATA);
|
|
|
|
switch(LOWORD (wParam))
|
|
{
|
|
/* User clicked shaft left of the scroll box. */
|
|
case SB_PAGEUP:
|
|
yNewPos = data->yPos-data->yPageSize;
|
|
break;
|
|
|
|
/* User clicked shaft right of the scroll box. */
|
|
case SB_PAGEDOWN:
|
|
yNewPos = data->yPos+data->yPageSize;
|
|
break;
|
|
|
|
/* User clicked the left arrow. */
|
|
case SB_LINEUP:
|
|
yNewPos = data->yPos-1;
|
|
break;
|
|
|
|
/* User clicked the right arrow. */
|
|
case SB_LINEDOWN:
|
|
yNewPos = data->yPos+1;
|
|
break;
|
|
|
|
/* User dragged the scroll box. */
|
|
case SB_THUMBTRACK:
|
|
yNewPos = HIWORD(wParam);
|
|
break;
|
|
|
|
default:
|
|
yNewPos = data->yPos;
|
|
}
|
|
|
|
yNewPos = max(0, yNewPos);
|
|
yNewPos = min(ROWNO-data->yPageSize+1, yNewPos);
|
|
if( yNewPos == data->yPos ) return;
|
|
|
|
yDelta = yNewPos - data->yPos;
|
|
data->yPos = yNewPos;
|
|
|
|
ScrollWindowEx (hWnd, 0, -data->yScrTile * yDelta,
|
|
(CONST RECT *) NULL, (CONST RECT *) NULL,
|
|
(HRGN) NULL, (LPRECT) NULL, SW_INVALIDATE | SW_ERASE);
|
|
|
|
si.cbSize = sizeof(si);
|
|
si.fMask = SIF_POS;
|
|
si.nPos = data->yPos;
|
|
SetScrollInfo(hWnd, SB_VERT, &si, TRUE);
|
|
}
|
|
|
|
/* on WM_HSCROLL */
|
|
void onMSNH_HScroll(HWND hWnd, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
PNHMapWindow data;
|
|
SCROLLINFO si;
|
|
int xNewPos;
|
|
int xDelta;
|
|
|
|
/* get window data */
|
|
data = (PNHMapWindow)GetWindowLong(hWnd, GWL_USERDATA);
|
|
|
|
switch(LOWORD (wParam))
|
|
{
|
|
/* User clicked shaft left of the scroll box. */
|
|
case SB_PAGEUP:
|
|
xNewPos = data->xPos-data->xPageSize;
|
|
break;
|
|
|
|
/* User clicked shaft right of the scroll box. */
|
|
case SB_PAGEDOWN:
|
|
xNewPos = data->xPos+data->xPageSize;
|
|
break;
|
|
|
|
/* User clicked the left arrow. */
|
|
case SB_LINEUP:
|
|
xNewPos = data->xPos-1;
|
|
break;
|
|
|
|
/* User clicked the right arrow. */
|
|
case SB_LINEDOWN:
|
|
xNewPos = data->xPos+1;
|
|
break;
|
|
|
|
/* User dragged the scroll box. */
|
|
case SB_THUMBTRACK:
|
|
xNewPos = HIWORD(wParam);
|
|
break;
|
|
|
|
default:
|
|
xNewPos = data->xPos;
|
|
}
|
|
|
|
xNewPos = max(0, xNewPos);
|
|
xNewPos = min(COLNO-data->xPageSize+1, xNewPos);
|
|
if( xNewPos == data->xPos ) return;
|
|
|
|
xDelta = xNewPos - data->xPos;
|
|
data->xPos = xNewPos;
|
|
|
|
ScrollWindowEx (hWnd, -data->xScrTile * xDelta, 0,
|
|
(CONST RECT *) NULL, (CONST RECT *) NULL,
|
|
(HRGN) NULL, (LPRECT) NULL, SW_INVALIDATE | SW_ERASE);
|
|
|
|
|
|
si.cbSize = sizeof(si);
|
|
si.fMask = SIF_POS;
|
|
si.nPos = data->xPos;
|
|
SetScrollInfo(hWnd, SB_HORZ, &si, TRUE);
|
|
}
|
|
|
|
/* map nethack map coordinates to the screen location */
|
|
void nhcoord2display(PNHMapWindow data, int x, int y, LPRECT lpOut)
|
|
{
|
|
lpOut->left = (x - data->xPos)*data->xScrTile + data->map_orig.x;
|
|
lpOut->top = (y - data->yPos)*data->yScrTile + data->map_orig.y;
|
|
lpOut->right = lpOut->left + data->xScrTile;
|
|
lpOut->bottom = lpOut->top + data->yScrTile;
|
|
}
|
|
|
|
/* map glyph to character/color combination */
|
|
void nhglyph2charcolor(short g, uchar* ch, int* color)
|
|
{
|
|
int offset;
|
|
#ifdef TEXTCOLOR
|
|
|
|
#define zap_color(n) *color = iflags.use_color ? zapcolors[n] : NO_COLOR
|
|
#define cmap_color(n) *color = iflags.use_color ? defsyms[n].color : NO_COLOR
|
|
#define obj_color(n) *color = iflags.use_color ? objects[n].oc_color : NO_COLOR
|
|
#define mon_color(n) *color = iflags.use_color ? mons[n].mcolor : NO_COLOR
|
|
#define pet_color(n) *color = iflags.use_color ? mons[n].mcolor : NO_COLOR
|
|
#define warn_color(n) *color = iflags.use_color ? def_warnsyms[n].color : NO_COLOR
|
|
|
|
# else /* no text color */
|
|
|
|
#define zap_color(n)
|
|
#define cmap_color(n)
|
|
#define obj_color(n)
|
|
#define mon_color(n)
|
|
#define pet_color(c)
|
|
#define warn_color(c)
|
|
*color = CLR_WHITE;
|
|
#endif
|
|
|
|
if ((offset = (g - GLYPH_WARNING_OFF)) >= 0) { /* a warning flash */
|
|
*ch = warnsyms[offset];
|
|
warn_color(offset);
|
|
} else if ((offset = (g - GLYPH_SWALLOW_OFF)) >= 0) { /* swallow */
|
|
/* see swallow_to_glyph() in display.c */
|
|
*ch = (uchar) showsyms[S_sw_tl + (offset & 0x7)];
|
|
mon_color(offset >> 3);
|
|
} else if ((offset = (g - GLYPH_ZAP_OFF)) >= 0) { /* zap beam */
|
|
/* see zapdir_to_glyph() in display.c */
|
|
*ch = showsyms[S_vbeam + (offset & 0x3)];
|
|
zap_color((offset >> 2));
|
|
} else if ((offset = (g - GLYPH_CMAP_OFF)) >= 0) { /* cmap */
|
|
*ch = showsyms[offset];
|
|
cmap_color(offset);
|
|
} else if ((offset = (g - GLYPH_OBJ_OFF)) >= 0) { /* object */
|
|
*ch = oc_syms[(int)objects[offset].oc_class];
|
|
obj_color(offset);
|
|
} else if ((offset = (g - GLYPH_BODY_OFF)) >= 0) { /* a corpse */
|
|
*ch = oc_syms[(int)objects[CORPSE].oc_class];
|
|
mon_color(offset);
|
|
} else if ((offset = (g - GLYPH_PET_OFF)) >= 0) { /* a pet */
|
|
*ch = monsyms[(int)mons[offset].mlet];
|
|
pet_color(offset);
|
|
} else { /* a monster */
|
|
*ch = monsyms[(int)mons[g].mlet];
|
|
mon_color(g);
|
|
}
|
|
// end of wintty code
|
|
}
|
|
|
|
/* map nethack color to RGB */
|
|
COLORREF nhcolor_to_RGB(int c)
|
|
{
|
|
switch(c) {
|
|
case CLR_BLACK: return RGB( 0, 0, 0);
|
|
case CLR_RED: return RGB(255, 0, 0);
|
|
case CLR_GREEN: return RGB( 0, 128, 0);
|
|
case CLR_BROWN: return RGB(165, 42, 42);
|
|
case CLR_BLUE: return RGB( 0, 0, 255);
|
|
case CLR_MAGENTA: return RGB(255, 0, 255);
|
|
case CLR_CYAN: return RGB( 0, 255, 255);
|
|
case CLR_GRAY: return RGB(192, 192, 192);
|
|
case NO_COLOR: return RGB( 0, 0, 0);
|
|
case CLR_ORANGE: return RGB(255, 165, 0);
|
|
case CLR_BRIGHT_GREEN: return RGB( 0, 255, 0);
|
|
case CLR_YELLOW: return RGB(255, 255, 0);
|
|
case CLR_BRIGHT_BLUE: return RGB(0, 191, 255);
|
|
case CLR_BRIGHT_MAGENTA: return RGB(255, 127, 255);
|
|
case CLR_BRIGHT_CYAN: return RGB(127, 255, 255); /* something close to aquamarine */
|
|
case CLR_WHITE: return RGB(255, 255, 255);
|
|
default: return RGB( 0, 0, 0); /* black */
|
|
}
|
|
}
|