mirror of
https://github.com/ClassiCube/ClassiCube.git
synced 2025-09-13 01:26:50 -04:00
638 lines
19 KiB
C
638 lines
19 KiB
C
#include "Core.h"
|
|
#if defined CC_BUILD_OS2 && !defined CC_BUILD_SDL2
|
|
#include "_WindowBase.h"
|
|
#include "String.h"
|
|
#include "Funcs.h"
|
|
#include "Bitmap.h"
|
|
#include "Options.h"
|
|
#include "Errors.h"
|
|
#include "Audio.h"
|
|
|
|
#define INCL_DOSPROCESS
|
|
#define INCL_DOSMEMMGR
|
|
#define INCL_DOSERRORS
|
|
#define INCL_PM
|
|
#define INCL_MMIOOS2
|
|
#define INCL_GRE_COLORTABLE
|
|
#define INCL_GRE_DEVMISC3
|
|
#include <os2.h>
|
|
#include <os2me.h>
|
|
#include <dive.h>
|
|
#include <fourcc.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#define WM_GetVideoModeTable 0x04A2
|
|
#define WM_SetVideoMode 0x04A0
|
|
#define WM_NotifyVideoModeChange 0x04A1
|
|
#define WS_DesktopDive 0x00000000L
|
|
#define WS_MaxDesktopDive 0x00000001L
|
|
#define WS_FullScreenDive 0x00000002L
|
|
|
|
#define CC_WIN_CLASSNAME "ClassiCube_Window"
|
|
#define Rect_Width(rect) (rect.xRight - rect.xLeft)
|
|
#define Rect_Height(rect) (rect.yTop - rect.yBottom)
|
|
|
|
static HAB habAnchor;
|
|
static HMQ msgQueue;
|
|
static HWND hwndFrame;
|
|
static HWND hwndClient;
|
|
static HDIVE hDive = NULLHANDLE;
|
|
static ULONG bufNum = 0;
|
|
static PBYTE imageBuffer = NULL;
|
|
BOOL fVrnDisabled = TRUE;
|
|
static ULONG bufferScanLineSize, bufferScanLines;
|
|
static ULONG clrColor;
|
|
RECTL rcls[50];
|
|
ULONG ulNumRcls;
|
|
static int win_width, win_height; /* Size of window including titlebar and borders */
|
|
static int windowX, windowY;
|
|
FOURCC fccColorFormat;
|
|
|
|
FILEDLG fileDialog;
|
|
void *mutex = NULL;
|
|
|
|
/**
|
|
* Handling native keystrokes
|
|
* Borrowed from SDL2 OS/2 code
|
|
**/
|
|
static void MapKeys(MPARAM mp1, MPARAM mp2) {
|
|
ULONG ulFlags = SHORT1FROMMP(mp1); /* WM_CHAR flags */
|
|
ULONG ulVirtualKey = SHORT2FROMMP(mp2); /* Virtual key code VK_* */
|
|
ULONG ulCharCode = SHORT1FROMMP(mp2); /* Character code */
|
|
ULONG ulScanCode = CHAR4FROMMP(mp1); /* Scan code */
|
|
static const int aScancode[] = {
|
|
INPUT_NONE, CCKEY_ESCAPE, CCKEY_1, CCKEY_2, CCKEY_3, CCKEY_4, CCKEY_5, CCKEY_6,
|
|
CCKEY_7, CCKEY_8, CCKEY_9, CCKEY_0, CCKEY_MINUS, CCKEY_EQUALS, CCKEY_BACKSPACE, CCKEY_TAB,
|
|
CCKEY_Q, CCKEY_W, CCKEY_E, CCKEY_R, CCKEY_T, CCKEY_Y, CCKEY_U, CCKEY_I,
|
|
CCKEY_O, CCKEY_P, CCKEY_LBRACKET, CCKEY_RBRACKET, CCKEY_ENTER, CCKEY_LCTRL, CCKEY_A, CCKEY_S,
|
|
CCKEY_D, CCKEY_F, CCKEY_G, CCKEY_H, CCKEY_J, CCKEY_K, CCKEY_L, CCKEY_SEMICOLON,
|
|
INPUT_NONE, INPUT_NONE, CCKEY_LSHIFT, CCKEY_BACKSLASH, CCKEY_Z, CCKEY_X, CCKEY_C, CCKEY_V,
|
|
CCKEY_B, CCKEY_N, CCKEY_M, CCKEY_COMMA, CCKEY_PERIOD, CCKEY_SLASH, CCKEY_RSHIFT, CCKEY_KP_MULTIPLY,
|
|
CCKEY_LALT, CCKEY_SPACE, CCKEY_CAPSLOCK, CCKEY_F1, CCKEY_F2, CCKEY_F3, CCKEY_F4, CCKEY_F5,
|
|
CCKEY_F6, CCKEY_F7, CCKEY_F8, CCKEY_F9, CCKEY_F10, CCKEY_NUMLOCK, CCKEY_SCROLLLOCK, CCKEY_KP7,
|
|
CCKEY_KP8, CCKEY_KP9, CCKEY_KP_MINUS, CCKEY_KP4, CCKEY_KP5, CCKEY_KP6, CCKEY_KP_PLUS, CCKEY_KP1,
|
|
CCKEY_KP2, CCKEY_KP3, CCKEY_KP0, CCKEY_KP_DECIMAL, INPUT_NONE, INPUT_NONE, CCKEY_BACKSLASH,CCKEY_F11,
|
|
CCKEY_F12, CCKEY_PAUSE, CCKEY_KP_ENTER, CCKEY_RCTRL, CCKEY_KP_DIVIDE, INPUT_NONE, CCKEY_RALT, INPUT_NONE,
|
|
CCKEY_HOME, CCKEY_UP, CCKEY_PAGEUP, CCKEY_LEFT, CCKEY_RIGHT, CCKEY_END, CCKEY_DOWN, CCKEY_PAGEDOWN,
|
|
CCKEY_F17, CCKEY_DELETE, CCKEY_F19, INPUT_NONE, INPUT_NONE, INPUT_NONE, INPUT_NONE, INPUT_NONE,
|
|
INPUT_NONE, INPUT_NONE, INPUT_NONE, INPUT_NONE,INPUT_NONE, INPUT_NONE, INPUT_NONE, INPUT_NONE,
|
|
INPUT_NONE, INPUT_NONE, INPUT_NONE, INPUT_NONE,INPUT_NONE,INPUT_NONE,CCKEY_LWIN, CCKEY_RWIN
|
|
};
|
|
|
|
if ((ulFlags & KC_CHAR) != 0 && (ulFlags & KC_KEYUP) == 0) {
|
|
if (WinQueryCapture(HWND_DESKTOP) != hwndClient) { // Mouse isn't captured
|
|
char c;
|
|
if (Convert_TryCodepointToCP437(ulCharCode, &c)) {
|
|
Event_RaiseInt(&InputEvents.Press, c);
|
|
return;
|
|
}
|
|
}
|
|
else {
|
|
Input_Set(ulCharCode, 0);
|
|
}
|
|
}
|
|
if ((ulFlags & KC_SCANCODE) != 0 && ulScanCode < sizeof(aScancode)/sizeof(int)) {
|
|
Input_Set(aScancode[ulScanCode], (ulFlags & KC_KEYUP) == 0);
|
|
}
|
|
}
|
|
|
|
static void RefreshWindowBounds() {
|
|
RECTL rect;
|
|
cc_bool changed = false;
|
|
|
|
if (WinQueryWindowRect(hwndClient, &rect)) {
|
|
Window_Main.Width = Rect_Width(rect);
|
|
Window_Main.Height = Rect_Height(rect);
|
|
if (Window_Main.Width != win_width || Window_Main.Height != win_height) {
|
|
changed = true;
|
|
win_width = Window_Main.Width;
|
|
win_height = Window_Main.Height;
|
|
}
|
|
if (changed) {
|
|
Event_RaiseVoid(&WindowEvents.StateChanged);
|
|
Event_RaiseVoid(&WindowEvents.Resized);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Window procedure
|
|
MRESULT EXPENTRY ClientWndProc(HWND hwnd, ULONG message, MPARAM mp1, MPARAM mp2) {
|
|
//printf("m %lu (%x)\n", message, message);
|
|
switch(message) {
|
|
case WM_CLOSE:
|
|
Event_RaiseVoid(&WindowEvents.Closing);
|
|
if (Window_Main.Exists) WinDestroyWindow(hwndFrame);
|
|
Window_Main.Exists = false;
|
|
break;
|
|
|
|
case WM_QUIT:
|
|
Window_Destroy();
|
|
Window_Main.Handle.ptr = NULL;
|
|
Window_Main.Exists = false;
|
|
return (MRESULT)0;
|
|
|
|
case WM_ACTIVATE:
|
|
Window_Main.Focused = SHORT1FROMMP(mp1) != 0;
|
|
Event_RaiseVoid(&WindowEvents.FocusChanged);
|
|
break;
|
|
|
|
case WM_SIZE:
|
|
RefreshWindowBounds();
|
|
break;
|
|
|
|
case WM_MOVE:
|
|
RefreshWindowBounds();
|
|
break;
|
|
|
|
case WM_ERASEBACKGROUND:
|
|
return (MRESULT)TRUE;
|
|
|
|
case WM_VRNDISABLED:
|
|
fVrnDisabled = TRUE;
|
|
return FALSE;
|
|
break;
|
|
|
|
case WM_VRNENABLED:
|
|
{
|
|
HPS hps;
|
|
HRGN hrgn;
|
|
RGNRECT rgnCtl;
|
|
|
|
hps = WinGetPS(hwnd);
|
|
hrgn = GpiCreateRegion(hps, 0L, NULL);
|
|
if (hrgn) {
|
|
// NOTE: If mp1 is zero, then this was just a move message.
|
|
// Illustrate the visible region on a WM_VRNENABLE.
|
|
//
|
|
WinQueryVisibleRegion(hwndClient, hrgn);
|
|
rgnCtl.ircStart = 0;
|
|
rgnCtl.crc = 50;
|
|
rgnCtl.ulDirection = 1;
|
|
|
|
// Get the all visible rectangles
|
|
if(!GpiQueryRegionRects(hps, hrgn, NULL, &rgnCtl, rcls)) {
|
|
DiveSetupBlitter (hDive, 0);
|
|
GpiDestroyRegion (hps, hrgn);
|
|
break;
|
|
}
|
|
|
|
ulNumRcls = rgnCtl.crcReturned;
|
|
GpiDestroyRegion(hps, hrgn);
|
|
|
|
// Release presentation space
|
|
WinReleasePS(hps);
|
|
}
|
|
if (mp1 == 0) Event_RaiseVoid(&WindowEvents.RedrawNeeded);
|
|
// Enable blitting
|
|
fVrnDisabled = FALSE;
|
|
}
|
|
return FALSE;
|
|
break;
|
|
|
|
case WM_MOUSEMOVE:
|
|
{
|
|
SHORT x = SHORT1FROMMP(mp1);
|
|
SHORT y = Window_Main.Height - SHORT2FROMMP(mp1);
|
|
Pointer_SetPosition(0, x, y);
|
|
}
|
|
break;
|
|
|
|
case WM_BUTTON1DOWN:
|
|
Input_SetPressed(CCMOUSE_L); break;
|
|
case WM_BUTTON3DOWN:
|
|
Input_SetPressed(CCMOUSE_M); break;
|
|
case WM_BUTTON2DOWN:
|
|
Input_SetPressed(CCMOUSE_R); break;
|
|
|
|
case WM_BUTTON1UP:
|
|
Input_SetReleased(CCMOUSE_L); break;
|
|
case WM_BUTTON3UP:
|
|
Input_SetReleased(CCMOUSE_M); break;
|
|
case WM_BUTTON2UP:
|
|
Input_SetReleased(CCMOUSE_R); break;
|
|
|
|
case WM_CHAR:
|
|
MapKeys(mp1, mp2);
|
|
return (MRESULT)TRUE;
|
|
}
|
|
//printf("default msg %lu (%x)\n", message, message);
|
|
return WinDefWindowProc(hwnd, message, mp1, mp2);
|
|
}
|
|
|
|
void DiveFreeBuffer(void) {
|
|
ULONG rc;
|
|
|
|
if (bufNum != 0) {
|
|
rc = DiveFreeImageBuffer(hDive, bufNum);
|
|
if (rc != DIVE_SUCCESS) {
|
|
// TODO Error messages
|
|
}
|
|
bufNum = 0;
|
|
}
|
|
if (imageBuffer != NULL) {
|
|
rc = DosFreeMem(imageBuffer);
|
|
if (rc != NO_ERROR) {
|
|
// TODO Error messages
|
|
}
|
|
imageBuffer = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
/*########################################################################################################################*
|
|
*--------------------------------------------------Public implementation--------------------------------------------------*
|
|
*#########################################################################################################################*/
|
|
void Window_PreInit(void) { }
|
|
void Window_Init(void) {
|
|
ULONG rc;
|
|
PPIB pib;
|
|
DIVE_CAPS caps = {0};
|
|
FOURCC fccFormats[100] = {0};
|
|
HDC hdc;
|
|
HPS hps;
|
|
LONG screencaps[CAPS_DEVICE_POLYSET_POINTS];
|
|
|
|
/* Change process type code for use Win* API from VIO session */
|
|
DosGetInfoBlocks(NULL, &pib);
|
|
if (pib->pib_ultype == 2 || pib->pib_ultype == 0) {
|
|
/* VIO windowable or fullscreen protect-mode session */
|
|
pib->pib_ultype = 3; /* Presentation Manager protect-mode session */
|
|
}
|
|
|
|
habAnchor = WinInitialize(0);
|
|
if (habAnchor == NULLHANDLE) {
|
|
Process_Abort2(LOUSHORT(WinGetLastError(0)), "Initialization failed");
|
|
}
|
|
msgQueue = WinCreateMsgQueue(habAnchor, 0);
|
|
if (msgQueue == NULLHANDLE) {
|
|
Process_Abort2(LOUSHORT(WinGetLastError(habAnchor)), "Window queue creation failed");
|
|
}
|
|
|
|
// Init Dive
|
|
caps.pFormatData = fccFormats; //TODO do something with it.
|
|
caps.ulFormatLength = 100;
|
|
caps.ulStructLen = sizeof(DIVE_CAPS);
|
|
if (rc = DiveQueryCaps(&caps, DIVE_BUFFER_SCREEN)) {
|
|
Process_Abort2(rc, "DIVE: Could not get capabilities.");
|
|
}
|
|
if (caps.ulDepth < 24) {
|
|
Process_Abort2(rc, "DIVE: Too few colours");
|
|
}
|
|
|
|
rc = DiveOpen(&hDive, FALSE, NULL);
|
|
if (rc != DIVE_SUCCESS) {
|
|
Process_Abort2(rc, "DIVE: Display engine instance open failed");
|
|
}
|
|
// Everything is fine, so we set infos.
|
|
hps = WinGetScreenPS(HWND_DESKTOP);
|
|
hdc = GpiQueryDevice(hps);
|
|
DevQueryCaps(hdc, CAPS_FAMILY, CAPS_DEVICE_POLYSET_POINTS, screencaps);
|
|
WinReleasePS(hps);
|
|
DisplayInfo.Width = screencaps[CAPS_WIDTH];
|
|
DisplayInfo.Height = screencaps[CAPS_HEIGHT];
|
|
DisplayInfo.Depth = screencaps[CAPS_COLOR_BITCOUNT];
|
|
DisplayInfo.ScaleX = 1.0;
|
|
DisplayInfo.ScaleY = 1.0;
|
|
DisplayInfo.DPIScaling = Options_GetBool(OPT_DPI_SCALING, false);
|
|
Input.Sources = INPUT_SOURCE_NORMAL;
|
|
}
|
|
|
|
void Window_Create(int width, int height) {
|
|
ULONG ulFlags = FCF_SHELLPOSITION | FCF_TASKLIST
|
|
| FCF_SIZEBORDER | FCF_ICON | FCF_MAXBUTTON |FCF_MINBUTTON
|
|
| FCF_MAXBUTTON | FCF_TITLEBAR | FCF_SYSMENU;
|
|
|
|
if (!WinRegisterClass(habAnchor, CC_WIN_CLASSNAME, ClientWndProc, 0, 0)) {
|
|
Process_Abort2(LOUSHORT(WinGetLastError(habAnchor)), "Window class registration failed");
|
|
}
|
|
hwndFrame = WinCreateStdWindow(HWND_DESKTOP, WS_VISIBLE | WS_SYNCPAINT, &ulFlags,
|
|
CC_WIN_CLASSNAME, "ClassiCube", 0L, NULLHANDLE, 0, &hwndClient);
|
|
if (hwndFrame == NULLHANDLE) {
|
|
Process_Abort2(LOUSHORT(WinGetLastError(habAnchor)), "Failed to create window");
|
|
}
|
|
Window_SetSize(width, height);
|
|
Window_Main.Exists = true;
|
|
Window_Main.Handle.ptr = &hwndFrame;
|
|
Window_Main.Focused = TRUE;
|
|
|
|
WinSetVisibleRegionNotify(hwndClient, TRUE);
|
|
// Send an invlaidation message to the client.
|
|
WinPostMsg(hwndFrame, WM_VRNENABLED, 0L, 0L);
|
|
}
|
|
|
|
void Window_Create2D(int width, int height) {
|
|
Window_Create(width, height);
|
|
}
|
|
|
|
void Window_Create3D(int width, int height) {
|
|
Window_Create(width, height);
|
|
}
|
|
|
|
void Window_Free(void) { }
|
|
|
|
void Window_Destroy(void) {
|
|
WinSetVisibleRegionNotify (hwndClient, FALSE);
|
|
DiveFreeBuffer();
|
|
if (hDive != NULLHANDLE) DiveClose(hDive);
|
|
hDive = NULLHANDLE;
|
|
AudioBackend_Free();
|
|
}
|
|
|
|
void Window_ProcessEvents(float delta) {
|
|
QMSG msg;
|
|
HWND hwndFocused;
|
|
int roundtrips = 5;
|
|
|
|
while (roundtrips && WinPeekMsg(habAnchor, &msg, NULLHANDLE, 0, 0, PM_NOREMOVE)) {
|
|
WinGetMsg(habAnchor, &msg, NULLHANDLE, 0, 0);
|
|
WinDispatchMsg(habAnchor, &msg);
|
|
roundtrips--;
|
|
}
|
|
}
|
|
|
|
void Window_SetTitle(const cc_string* title) {
|
|
char nativeTitle[title->capacity+1];
|
|
Mem_Set(nativeTitle, 0, title->capacity+1);
|
|
String_CopyToRaw(nativeTitle, title->capacity, title);
|
|
WinSetWindowText(hwndFrame, nativeTitle);
|
|
}
|
|
|
|
void Clipboard_GetText(cc_string* value) {
|
|
PSZ data;
|
|
|
|
if (WinOpenClipbrd(habAnchor)) {
|
|
data = (PSZ) WinQueryClipbrdData(habAnchor, CF_TEXT);
|
|
if (data) {
|
|
strcpy(value->buffer, data);
|
|
value->length = strlen(data);
|
|
}
|
|
WinCloseClipbrd(habAnchor);
|
|
}
|
|
}
|
|
|
|
void Clipboard_SetText(const cc_string* value) {
|
|
|
|
if (WinOpenClipbrd(habAnchor)) {
|
|
WinEmptyClipbrd(habAnchor);
|
|
WinSetClipbrdData(habAnchor, (ULONG) value->buffer, CF_TEXT, CFI_POINTER);
|
|
WinCloseClipbrd(habAnchor);
|
|
}
|
|
}
|
|
|
|
void Window_Show(void) {
|
|
WinShowWindow(hwndFrame, TRUE);
|
|
WinSetFocus(HWND_DESKTOP, hwndClient);
|
|
}
|
|
|
|
void Window_SetSize(int width, int height) {
|
|
WinSetWindowPos(hwndFrame, HWND_TOP, 0, 0, width, height, SWP_SIZE);
|
|
}
|
|
|
|
void Window_RequestClose(void) {
|
|
WinPostMsg(hwndClient, WM_CLOSE, 0, 0);
|
|
}
|
|
|
|
int Window_GetWindowState(void) {
|
|
ULONG style = WinQueryWindowULong(hwndFrame, QWL_STYLE);
|
|
if (style & WS_MINIMIZED) return WINDOW_STATE_MINIMISED;
|
|
if (style & WS_MAXIMIZED) return WINDOW_STATE_FULLSCREEN;
|
|
return WINDOW_STATE_NORMAL;
|
|
}
|
|
|
|
void Window_AllocFramebuffer(struct Bitmap* bmp, int width, int height) {
|
|
APIRET rc;
|
|
ULONG scanLineSize = width * (32 >> 3);
|
|
|
|
/* Destroy previous buffer. */
|
|
DiveFreeBuffer();
|
|
|
|
if (width == 0 || height == 0) return;
|
|
|
|
/* Bytes per line. */
|
|
scanLineSize = (scanLineSize + 3) & ~3; /* 4-byte aligning */
|
|
|
|
rc = DosAllocMem((PVOID)&imageBuffer,
|
|
(height * scanLineSize) + sizeof(ULONG),
|
|
PAG_COMMIT | PAG_READ | PAG_WRITE);
|
|
|
|
if (rc != NO_ERROR) {
|
|
// TODO Debug messages
|
|
return;
|
|
}
|
|
|
|
rc = DiveAllocImageBuffer(hDive, &bufNum, FOURCC_BGR4, width, height,
|
|
scanLineSize, imageBuffer);
|
|
if (rc != DIVE_SUCCESS) {
|
|
DosFreeMem(imageBuffer);
|
|
imageBuffer = NULL;
|
|
bufNum = 0;
|
|
bmp->scan0 = NULL;
|
|
Process_Abort2(rc, "Dive: Could not allocate image buffer");
|
|
return;
|
|
}
|
|
|
|
rc = DiveBeginImageBufferAccess(hDive, bufNum,
|
|
&imageBuffer, &bufferScanLineSize, &bufferScanLines);
|
|
if (rc != DIVE_SUCCESS) {
|
|
DiveFreeImageBuffer(hDive, bufNum);
|
|
DosFreeMem(imageBuffer);
|
|
imageBuffer = NULL;
|
|
bufNum = 0;
|
|
bmp->scan0 = NULL;
|
|
Process_Abort2(rc, "Dive: Access to image buffer failed");
|
|
}
|
|
|
|
bmp->scan0 = (unsigned int*)imageBuffer;
|
|
bmp->width = width;
|
|
bmp->height = height;
|
|
}
|
|
|
|
|
|
void Window_DrawFramebuffer(Rect2D r, struct Bitmap* bmp) {
|
|
SETUP_BLITTER setupBlitter;
|
|
POINTL pointl;
|
|
SWP swp;
|
|
RECTL rect;
|
|
APIRET rc;
|
|
|
|
rc = DiveEndImageBufferAccess(hDive, bufNum);
|
|
if (rc != DIVE_SUCCESS) {
|
|
// TODO Error messages
|
|
return;
|
|
}
|
|
if (fVrnDisabled) {
|
|
DiveSetupBlitter(hDive, 0);
|
|
}
|
|
else {
|
|
// Now find the window position and size, relative to parent.
|
|
WinQueryWindowPos(hwndClient, &swp);
|
|
// Convert the point to offset from desktop lower left.
|
|
pointl.x = swp.x;
|
|
pointl.y = swp.y;
|
|
WinMapWindowPoints(hwndFrame, HWND_DESKTOP, &pointl, 1);
|
|
WinQueryWindowRect(hwndClient, &rect);
|
|
|
|
// TODO CC's coordinate begins in the upper left corner.
|
|
// TODO Fix coordinate system for partial updates
|
|
// Tell DIVE about the new settings.
|
|
setupBlitter.ulStructLen = sizeof(SETUP_BLITTER);
|
|
setupBlitter.fccSrcColorFormat = FOURCC_BGR4;
|
|
setupBlitter.ulSrcWidth = swp.cx;
|
|
setupBlitter.ulSrcHeight = swp.cy;
|
|
setupBlitter.ulSrcPosX = 0;
|
|
setupBlitter.ulSrcPosY = 0;
|
|
setupBlitter.fInvert = FALSE;
|
|
setupBlitter.ulDitherType = 1;
|
|
setupBlitter.fccDstColorFormat = FOURCC_SCRN;
|
|
setupBlitter.ulDstWidth = swp.cx;
|
|
setupBlitter.ulDstHeight = swp.cy;
|
|
setupBlitter.lDstPosX = 0;
|
|
setupBlitter.lDstPosY = 0;
|
|
setupBlitter.lScreenPosX = pointl.x;
|
|
setupBlitter.lScreenPosY = pointl.y;
|
|
setupBlitter.ulNumDstRects = ulNumRcls;
|
|
setupBlitter.pVisDstRects = rcls;
|
|
DiveSetupBlitter(hDive, &setupBlitter);
|
|
DiveBlitImage(hDive, bufNum, DIVE_BUFFER_SCREEN);
|
|
}
|
|
}
|
|
|
|
void Window_FreeFramebuffer(struct Bitmap* bmp) {
|
|
DiveFreeBuffer();
|
|
}
|
|
|
|
cc_result Window_EnterFullscreen(void) {
|
|
SWP swp;
|
|
if (!WinGetMaxPosition(hwndFrame, &swp)) return ERROR_BASE; // TODO
|
|
if (!WinSetWindowPos(hwndFrame, HWND_TOP,
|
|
swp.x, swp.y, swp.cx, swp.cy, SWP_MAXIMIZE)
|
|
) return ERROR_BASE; // TODO
|
|
return 0;
|
|
}
|
|
|
|
cc_result Window_ExitFullscreen(void) {
|
|
if (!WinSetWindowPos(hwndFrame, HWND_TOP,
|
|
0, 0, 0, 0, SWP_RESTORE | SWP_SHOW | SWP_ACTIVATE)) return ERROR_BASE; // TODO
|
|
return 0;
|
|
}
|
|
|
|
int Window_IsObscured(void) { return 0; }
|
|
|
|
void ShowDialogCore(const char *title, const char *name) {
|
|
WinMessageBox(HWND_DESKTOP, hwndClient, name, title, 0, MB_OK|MB_APPLMODAL);
|
|
}
|
|
|
|
MRESULT EXPENTRY myFileDlgProc(HWND hwnd, ULONG msg, MPARAM mp1, MPARAM mp2) {
|
|
MRESULT result;
|
|
printf("Dialog message %04x, mp1 %u, mp2 %u hwnd %u (%u, %u)\n", msg, mp1, mp2, hwnd, hwndFrame, hwndClient);
|
|
result = WinDefFileDlgProc(hwnd, msg, mp1, mp2);
|
|
printf("Default %u\n", result);
|
|
return result;
|
|
}
|
|
|
|
cc_result Window_OpenFileDialog(const struct OpenFileDialogArgs* args) {
|
|
|
|
HWND hDialog;
|
|
|
|
WinSetVisibleRegionNotify(hwndClient, FALSE);
|
|
memset(&fileDialog, 0, sizeof(FILEDLG));
|
|
fileDialog.cbSize = sizeof(FILEDLG);
|
|
fileDialog.fl = FDS_OPEN_DIALOG | FDS_CENTER | FDS_HELPBUTTON;
|
|
fileDialog.pszTitle = (PSZ)args->description;
|
|
//fileDialog.pfnDlgProc = myFileDlgProc;
|
|
|
|
strcpy(fileDialog.szFullFile, *(args->filters));
|
|
hDialog = WinFileDlg(HWND_DESKTOP, hwndFrame, &fileDialog);
|
|
|
|
if (fileDialog.lReturn == DID_OK) {
|
|
cc_string temp = String_FromRaw(fileDialog.szFullFile, CCHMAXPATH);
|
|
args->Callback(&temp);
|
|
}
|
|
WinSetVisibleRegionNotify(hwndClient, TRUE);
|
|
return 0;
|
|
}
|
|
|
|
cc_result Window_SaveFileDialog(const struct SaveFileDialogArgs* args) {
|
|
FILEDLG fileDialog;
|
|
HWND hDialog;
|
|
|
|
memset(&fileDialog, 0, sizeof(FILEDLG));
|
|
fileDialog.cbSize = sizeof(FILEDLG);
|
|
fileDialog.fl = FDS_HELPBUTTON | FDS_CENTER | FDS_PRELOAD_VOLINFO | FDS_SAVEAS_DIALOG;
|
|
fileDialog.pszTitle = (PSZ)args->titles;
|
|
fileDialog.pszOKButton = NULL;
|
|
|
|
Mem_Copy(fileDialog.szFullFile, *(args->filters), CCHMAXPATH);
|
|
hDialog = WinFileDlg(HWND_DESKTOP, hwndClient, &fileDialog);
|
|
if (fileDialog.lReturn == DID_OK) {
|
|
cc_string temp = String_FromRaw(fileDialog.szFullFile, CCHMAXPATH);
|
|
args->Callback(&temp);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Displays on-screen keyboard for platforms that lack physical keyboard input. */
|
|
/* NOTE: On desktop platforms, this won't do anything. */
|
|
void OnscreenKeyboard_Open(struct OpenKeyboardArgs* args) { }
|
|
/* Sets the text used for keyboard input. */
|
|
/* NOTE: This is e.g. used for mobile on-screen keyboard input with the web client, */
|
|
/* because it is backed by a HTML input, rather than true keyboard input events. */
|
|
/* As such, this is necessary to ensure the HTML input is consistent with */
|
|
/* whatever text input widget is actually being displayed on screen. */
|
|
void OnscreenKeyboard_SetText(const cc_string* text) { }
|
|
void OnscreenKeyboard_Draw2D(Rect2D* r, struct Bitmap* bmp) { }
|
|
void OnscreenKeyboard_Draw3D(void) { }
|
|
/* Hides/Removes the previously displayed on-screen keyboard. */
|
|
void OnscreenKeyboard_Close(void) { }
|
|
/* Locks/Unlocks the landscape orientation. */
|
|
void Window_LockLandscapeOrientation(cc_bool lock) { }
|
|
|
|
void Window_EnableRawMouse(void) {
|
|
if (WinSetCapture(HWND_DESKTOP, hwndClient))
|
|
DefaultEnableRawMouse();
|
|
}
|
|
|
|
void Window_UpdateRawMouse(void) {
|
|
DefaultUpdateRawMouse();
|
|
}
|
|
|
|
void Window_DisableRawMouse(void) {
|
|
if (WinSetCapture(HWND_DESKTOP, NULLHANDLE))
|
|
DefaultDisableRawMouse();
|
|
}
|
|
|
|
void Gamepads_Init(void) { }
|
|
|
|
void Gamepads_Process(float delta) { }
|
|
|
|
void Cursor_GetRawPos(int *x, int *y) {
|
|
POINTL pointl;
|
|
ULONG ulRes;
|
|
|
|
WinQueryPointerPos(HWND_DESKTOP, &pointl);
|
|
*x = pointl.x;
|
|
*y = WinQuerySysValue(HWND_DESKTOP, SV_CYSCREEN) - pointl.y - 1;
|
|
}
|
|
|
|
void Cursor_SetPosition(int x, int y) {
|
|
POINTL pointl;
|
|
|
|
pointl.x = x;
|
|
pointl.y = win_height - y;
|
|
WinMapWindowPoints(hwndClient, HWND_DESKTOP, &pointl, 1);
|
|
WinSetPointerPos(HWND_DESKTOP, pointl.x, pointl.y);
|
|
}
|
|
|
|
static void Cursor_DoSetVisible(cc_bool visible) {
|
|
WinShowCursor(hwndClient, visible);
|
|
}
|
|
|
|
|
|
|
|
void Window_ProcessGamepads(float delta) { }
|
|
|
|
#endif
|