/***************************************************************************** * Copyright (c) 2014 Ted John * OpenRCT2, an open source clone of Roller Coaster Tycoon 2. * * This file is part of OpenRCT2. * * OpenRCT2 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 3 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, see . *****************************************************************************/ #include #include #include #include "addresses.h" #include "gfx.h" #include "rct2.h" #include "strings.h" #include "window.h" uint8 _screenDirtyBlocks[5120]; static void gfx_draw_dirty_blocks(int x, int y, int columns, int rows); /** * * rct2: 0x00678998 */ void gfx_load_g1() { HANDLE hFile; DWORD bytesRead; DWORD header[2]; int i; int g1BufferSize; void* g1Buffer; rct_g1_element *g1Elements = RCT2_ADDRESS(RCT2_ADDRESS_G1_ELEMENTS, rct_g1_element); hFile = CreateFile(get_file_path(PATH_ID_G1), GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_FLAG_RANDOM_ACCESS | FILE_ATTRIBUTE_NORMAL, NULL); if (hFile != INVALID_HANDLE_VALUE) { ReadFile(hFile, header, 8, &bytesRead, NULL); if (bytesRead == 8) { g1BufferSize = header[1]; g1Buffer = rct2_malloc(g1BufferSize); ReadFile(hFile, g1Elements, 29294 * sizeof(rct_g1_element), &bytesRead, NULL); ReadFile(hFile, g1Buffer, g1BufferSize, &bytesRead, NULL); CloseHandle(hFile); for (i = 0; i < 29294; i++) g1Elements[i].offset += (int)g1Buffer; return; } } // exit with error fprintf(stderr, "Unable to load g1.dat"); assert(0); } /** * Clears the screen with the specified colour. * rct2: 0x00678A9F */ void gfx_clear(rct_drawpixelinfo *dpi, int colour) { int y, w, h; char* ptr; w = dpi->width >> dpi->var_0F; h = dpi->height >> dpi->var_0F; ptr = dpi->bits; for (y = 0; y < h; y++) { memset(ptr, colour, w); ptr += w + dpi->pitch; } } /** * * rct2: 0x00684466 * dpi (edi) * x1 (ax) * y1 (bx) * x2 (cx) * y2 (dx) * colour (ebp) */ void gfx_draw_line(rct_drawpixelinfo *dpi, int x1, int y1, int x2, int y2, int colour) { RCT2_CALLPROC_X(0x00684466, x1, y1, x2, y2, 0, dpi, colour); } /** * * rct2: 0x00678AD4 * dpi (edi) * left (ax) * top (cx) * right (bx) * bottom (dx) * colour (ebp) */ void gfx_fill_rect(rct_drawpixelinfo *dpi, int left, int top, int right, int bottom, int colour) { RCT2_CALLPROC_X(0x00678AD4, left, right, top, bottom, 0, dpi, colour); } /** * * rct2: 0x006E6F81 * dpi (edi) * left (ax) * top (cx) * right (bx) * bottom (dx) * colour (ebp) * _si (si) */ void gfx_fill_rect_inset(rct_drawpixelinfo* dpi, short left, short top, short right, short bottom, int colour, short _si) { RCT2_CALLPROC_X(0x006E6F81, left, right, top, bottom, _si, dpi, colour); } /** * * rct2: 0x0067A28E * image_id (ebx) * x (cx) * y (dx) */ void gfx_draw_sprite(rct_drawpixelinfo *dpi, int image_id, int x, int y) { RCT2_CALLPROC_X(0x0067A28E, 0, image_id, x, y, 0, dpi, 0); } /** * * rct2: 0x00683854 * a1 (ebx) * product (cl) */ void gfx_transpose_palette(int pal, unsigned char product) { int eax, ebx, ebp; uint8* esi, *edi; ebx = pal * 16; esi = (uint8*)(*((int*)(0x009EBD28 + ebx))); ebp = *((short*)(0x009EBD2C + ebx)); eax = *((short*)(0x009EBD30 + ebx)) * 4; edi = (uint8*)0x01424680 + eax; for (; ebp > 0; ebp--) { edi[0] = (esi[0] * product) >> 8; edi[1] = (esi[1] * product) >> 8; edi[2] = (esi[2] * product) >> 8; esi += 3; edi += 4; } RCT2_CALLPROC_3(0x00405595, int, int, int, 0x01424680, 10, 236); } /** * Draws i formatted text string centred at i specified position. * rct2: 0x006C1D6C * dpi (edi) * format (bx) * x (cx) * y (dx) * colour (al) * args (esi) */ void gfx_draw_string_centred(rct_drawpixelinfo *dpi, int format, int x, int y, int colour, void *args) { RCT2_CALLPROC_X(0x006C1D6C, colour, format, x, y, args, dpi, 0); } /** * * rct2: 0x006ED7E5 */ void gfx_invalidate_screen() { int width = RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_WIDTH, sint16); int height = RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_HEIGHT, sint16); gfx_set_dirty_blocks(0, 0, width, height); } /** * * rct2: 0x006E732D * left (ax) * top (bx) * right (dx) * bottom (bp) */ void gfx_set_dirty_blocks(int left, int top, int right, int bottom) { int x, y; uint8 *screenDirtyBlocks = RCT2_ADDRESS(0x00EDE408, uint8); left = max(left, 0); top = max(top, 0); right = min(right, RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_WIDTH, sint16)); bottom = min(bottom, RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_HEIGHT, sint16)); if (left >= right) return; if (top >= bottom) return; right--; bottom--; left >>= RCT2_GLOBAL(0x009ABDF0, sint8); right >>= RCT2_GLOBAL(0x009ABDF0, sint8); top >>= RCT2_GLOBAL(0x009ABDF1, sint8); bottom >>= RCT2_GLOBAL(0x009ABDF1, sint8); for (y = top; y <= bottom; y++) for (x = left; x <= right; x++) screenDirtyBlocks[y * RCT2_GLOBAL(0x009ABDE8, sint32) + x] = 0xFF; } /** * * rct2: 0x006E73BE */ void gfx_draw_all_dirty_blocks() { int x, y, xx, yy, columns, rows; short left, top, right, bottom; uint8 *screenDirtyBlocks = RCT2_ADDRESS(0x00EDE408, uint8); for (x = 0; x < RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_COLUMNS, sint32); x++) { for (y = 0; y < RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_ROWS, sint32); y++) { if (screenDirtyBlocks[y * RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_COLUMNS, sint32) + x] == 0) continue; // Determine columns for (xx = x; xx < RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_COLUMNS, sint32); xx++) if (screenDirtyBlocks[y * RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_COLUMNS, sint32) + xx] == 0) break; columns = xx - x; // Check rows for (yy = y; yy < RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_ROWS, sint32); yy++) for (xx = x; xx < x + columns; xx++) if (screenDirtyBlocks[yy * RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_COLUMNS, sint32) + xx] == 0) goto endRowCheck; endRowCheck: rows = yy - y; gfx_draw_dirty_blocks(x, y, columns, rows); } } } static void gfx_draw_dirty_blocks(int x, int y, int columns, int rows) { int left, top, right, bottom; uint8 *screenDirtyBlocks = RCT2_ADDRESS(0x00EDE408, uint8); // Unset dirty blocks for (top = y; top < y + rows; top++) for (left = x; left < x + columns; left++) screenDirtyBlocks[top * RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_COLUMNS, sint32) + left] = 0; // Determine region in pixels left = max(0, x * RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_WIDTH, sint16)); top = max(0, y * RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_HEIGHT, sint16)); right = min(RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_WIDTH, sint16), left + (columns * RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_WIDTH, sint16))); bottom = min(RCT2_GLOBAL(RCT2_ADDRESS_SCREEN_HEIGHT, sint16), top + (rows * RCT2_GLOBAL(RCT2_ADDRESS_DIRTY_BLOCK_HEIGHT, sint16))); if (right <= left || bottom <= top) return; // Draw region gfx_redraw_screen_rect(left, top, right, bottom); } /** * * rct2: 0x006E7499 * left (ax) * top (bx) * right (dx) * bottom (bp) */ void gfx_redraw_screen_rect(short left, short top, short right, short bottom) { rct_window* w; rct_drawpixelinfo *screenDPI = RCT2_ADDRESS(RCT2_ADDRESS_SCREEN_DPI, rct_drawpixelinfo); rct_drawpixelinfo *windowDPI = RCT2_ADDRESS(RCT2_ADDRESS_WINDOW_DPI, rct_drawpixelinfo); // Unsure what this does RCT2_CALLPROC_X(0x00683326, left, top, right - 1, bottom - 1, 0, 0, 0); windowDPI->bits = screenDPI->bits + left + ((screenDPI->width + screenDPI->pitch) * top); windowDPI->x = left; windowDPI->y = top; windowDPI->width = right - left; windowDPI->height = bottom - top; windowDPI->pitch = screenDPI->width + screenDPI->pitch + left - right; for (w = RCT2_ADDRESS(RCT2_ADDRESS_WINDOW_LIST, rct_window); w < RCT2_GLOBAL(RCT2_ADDRESS_NEW_WINDOW_PTR, rct_window*); w++) { if (w->flags & WF_TRANSPARENT) continue; if (right <= w->x || bottom <= w->y) continue; if (left >= w->x + w->width || top >= w->y + w->height) continue; window_draw(w, left, top, right, bottom); } } /** * * rct2: 0x006C2321 * buffer (esi) */ int gfx_get_string_width(char *buffer) { int eax, ebx, ecx, edx, esi, edi, ebp; esi = buffer; RCT2_CALLFUNC_X(0x006C2321, &eax, &ebx, &ecx, &edx, &esi, &edi, &ebp); return ecx & 0xFFFF; } /** * Draws i formatted text string left aligned at i specified position but clips * the text with an elipsis if the text width exceeds the specified width. * rct2: 0x006C1B83 * dpi (edi) * format (bx) * args (esi) * colour (al) * x (cx) * y (dx) * width (bp) */ void gfx_draw_string_left_clipped(rct_drawpixelinfo* dpi, int format, void* args, int colour, int x, int y, int width) { RCT2_CALLPROC_X(0x006C1B83, colour, format, x, y, args, dpi, width); //char* buffer; //buffer = (char*)0x0141ED68; //format_string(buffer, format, args); //rctmem->current_font_sprite_base = 224; //clip_text(buffer, width); //gfx_draw_string(dpi, buffer, colour, x, y); } /** * Draws i formatted text string centred at i specified position but clips the * text with an elipsis if the text width exceeds the specified width. * rct2: 0x006C1BBA * dpi (edi) * format (bx) * args (esi) * colour (al) * x (cx) * y (dx) * width (bp) */ void gfx_draw_string_centred_clipped(rct_drawpixelinfo *dpi, int format, void *args, int colour, int x, int y, int width) { RCT2_CALLPROC_X(0x006C1BBA, colour, format, x, y, args, dpi, width); //char* buffer; //short text_width; //buffer = (char*)0x0141ED68; //format_string(buffer, format, args); //rctmem->current_font_sprite_base = 224; //text_width = clip_text(buffer, width); //// Draw the text centred //x -= (text_width - 1) / 2; //gfx_draw_string(dpi, buffer, colour, x, y); } /** * * rct2: 0x006C1E53 * dpi (edi) * args (esi) * x (cx) * y (dx) * width (bp) * colour (al) * format (ebx) */ int gfx_draw_string_centred_wrapped(rct_drawpixelinfo *dpi, void *args, int x, int y, int width, int format, int colour) { int eax, ebx, ecx, edx, esi, edi, ebp; eax = colour; ebx = format; ecx = x; edx = y; esi = args; edi = dpi; ebp = width; RCT2_CALLFUNC_X(0x006C1E53, &eax, &ebx, &ecx, &edx, &esi, &edi, &ebp); return (sint16)(edx & 0xFFFF) - y; } /** * * rct2: 0x006C2105 * dpi (edi) * format (esi) * x (cx) * y (dx) * width (bp) * colour (bx) * unknown (al) */ int gfx_draw_string_left_wrapped(rct_drawpixelinfo *dpi, void *format, int x, int y, int width, int colour, int unknown) { int eax, ebx, ecx, edx, esi, edi, ebp; eax = unknown; ebx = colour; ecx = x; edx = y; esi = format; edi = dpi; ebp = width; RCT2_CALLFUNC_X(0x006C2105, &eax, &ebx, &ecx, &edx, &esi, &edi, &ebp); return (sint16)(edx & 0xFFFF) - y; } /** * Draws i formatted text string. * rct2: 0x006C1B2F * dpi (edi) * format (bx) * args (esi) * colour (al) * x (cx) * y (dx) */ void gfx_draw_string_left(rct_drawpixelinfo *dpi, int format, void *args, int colour, int x, int y) { char* buffer; buffer = (char*)0x0141ED68; format_string(buffer, format, args); gfx_draw_string(dpi, buffer, colour, x, y); } /** * * rct2: 0x00682702 * dpi (edi) * format (esi) * colour (al) * x (cx) * y (dx) */ void gfx_draw_string(rct_drawpixelinfo *dpi, char *format, int colour, int x, int y) { RCT2_CALLPROC_X(0x00682702, colour, 0, x, y, format, dpi, 0); }