// This file is part of BOINC. // http://boinc.berkeley.edu // Copyright (C) 2010 University of California // // BOINC is free software; you can redistribute it and/or modify it // under the terms of the GNU Lesser General Public License // as published by the Free Software Foundation, // either version 3 of the License, or (at your option) any later version. // // BOINC 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 Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with BOINC. If not, see . // Event loop and support functions for Windows versions // of BOINC applications w/ graphics. // Platform-independent code should NOT be here. // #ifdef _WIN32 #include "boinc_win.h" #else #include "config.h" #endif #include "app_ipc.h" #include "boinc_api.h" #include "diagnostics.h" #include "filesys.h" #include "graphics2.h" #include "str_replace.h" #include "str_util.h" #include "util.h" #define BOINC_WINDOW_CLASS_NAME "BOINC_app" #define WM_SHUTDOWNGFX WM_USER+1 static HDC win_dc = NULL; static HGLRC gl_dc = NULL; static HWND window = NULL; static HINSTANCE instance; static RECT rect = {50, 50, 50+640, 50+480}; static int current_graphics_mode = MODE_HIDE_GRAPHICS; static POINT mousePos; static bool visible = true; static bool window_ready = false; static UINT_PTR gfx_timer_id = 0; static bool fullscreen; void boinc_close_window_and_quit(const char* p) { char buf[256]; fprintf(stderr, "%s Close event (%s) detected, shutting down.\n", boinc_msg_prefix(buf, sizeof(buf)), p ); window_ready = false; wglMakeCurrent(NULL, NULL); // release GL rendering context, if any if (gl_dc) { wglDeleteContext(gl_dc); } if (window && win_dc) { ReleaseDC(window, win_dc); } if (window) { DestroyWindow(window); } } void SetupPixelFormat(HDC win_dc) { int nPixelFormat; char buf[256]; static PIXELFORMATDESCRIPTOR pfd = { sizeof(PIXELFORMATDESCRIPTOR), // size of structure. 1, // always 1. PFD_DRAW_TO_WINDOW | // support window PFD_SUPPORT_OPENGL | // support OpenGl PFD_DOUBLEBUFFER, // support double buffering PFD_TYPE_RGBA, // support RGBA 32, // 32 bit color mode 0, 0, 0, 0, 0, 0, // ignore color bits 0, // no alpha buffer 0, // ignore shift bit 0, // no accumulation buffer 0, 0, 0, 0, // ignore accumulation bits. 16, // number of depth buffer bits. 0, // number of stencil buffer bits. 0, // 0 means no auxiliary buffer PFD_MAIN_PLANE, // The main drawing plane 0, // this is reserved 0, 0, 0 // layer masks ignored. }; // chooses the best pixel format // nPixelFormat = ChoosePixelFormat(win_dc, &pfd); // set pixel format to device context. // if (!SetPixelFormat(win_dc, nPixelFormat, &pfd)) { fprintf(stderr, "%s ERROR: Couldn't set pixel format for device context (0x%x).\n", boinc_msg_prefix(buf, sizeof(buf)), GetLastError() ); } } static void make_window(const char* title) { RECT WindowRect = {0,0,0,0}; int width, height; DWORD dwExStyle; DWORD dwStyle; char buf[256]; if (fullscreen) { HDC screenDC=GetDC(NULL); WindowRect.left = WindowRect.top = 0; WindowRect.right=GetDeviceCaps(screenDC, HORZRES); WindowRect.bottom=GetDeviceCaps(screenDC, VERTRES); ReleaseDC(NULL, screenDC); dwExStyle=WS_EX_TOPMOST; dwStyle=WS_POPUP; while(ShowCursor(false) >= 0); } else { // Version 5 screensaver logic kills all MODE_WINDOW graphics // before starting one in fullscreen mode, // then restarts the ones it killed when screensaver stops. // To be compatible with V5, // we remember and restore the MODE_WINDOW dimensions. // FILE *f = boinc_fopen("gfx_info", "r"); if (f) { // ToDo: change this to XML parsing fscanf(f, "%d %d %d %d\n", &rect.left, &rect.top, &rect.right, &rect.bottom ); fclose(f); } WindowRect = rect; dwExStyle=WS_EX_APPWINDOW|WS_EX_WINDOWEDGE; dwStyle=WS_OVERLAPPEDWINDOW; while(ShowCursor(true) < 0); } char window_title[256]; if (title) { strlcpy(window_title, title, sizeof(window_title)); } else { APP_INIT_DATA aid; boinc_get_init_data(aid); if (!strlen(aid.app_name)) { strlcpy(aid.app_name, "BOINC Application", sizeof(aid.app_name)); } get_window_title(window_title, 256); } //fprintf(stderr, "Setting window title to '%s'.\n", window_title); window = CreateWindowEx(dwExStyle, BOINC_WINDOW_CLASS_NAME, window_title, dwStyle|WS_CLIPSIBLINGS|WS_CLIPCHILDREN, WindowRect.left, WindowRect.top, WindowRect.right-WindowRect.left,WindowRect.bottom-WindowRect.top, NULL, NULL, instance, NULL ); if (!SetForegroundWindow(window)) { fprintf(stderr, "%s ERROR: SetForegroundWindow() failed (0x%x).\n", boinc_msg_prefix(buf, sizeof(buf)), GetLastError() ); } if (!GetCursorPos(&mousePos)) { fprintf(stderr, "%s ERROR: GetCursorPos() failed (0x%x).\n", boinc_msg_prefix(buf, sizeof(buf)), GetLastError() ); } win_dc = GetDC(window); if (!win_dc) { fprintf(stderr, "%s ERROR: GetDC() failed (0x%x).\n", boinc_msg_prefix(buf, sizeof(buf)), GetLastError() ); } SetupPixelFormat(win_dc); gl_dc = wglCreateContext(win_dc); if (!gl_dc) { fprintf(stderr, "%s ERROR: wglCreateContext() failed (0x%x).\n", boinc_msg_prefix(buf, sizeof(buf)), GetLastError() ); ReleaseDC(window, win_dc); return; } if(!wglMakeCurrent(win_dc, gl_dc)) { fprintf(stderr, "%s ERROR: wglMakeCurrent() failed (0x%x).\n", boinc_msg_prefix(buf, sizeof(buf)), GetLastError() ); ReleaseDC(window, win_dc); wglDeleteContext(gl_dc); return; } // use client area for resize when not fullscreen if (current_graphics_mode != MODE_FULLSCREEN) { GetClientRect(window, &WindowRect); } width = WindowRect.right-WindowRect.left; height = WindowRect.bottom-WindowRect.top; ShowWindow(window, SW_SHOW); SetFocus(window); app_graphics_init(); app_graphics_resize(width, height); window_ready=true; } void parse_mouse_event(UINT uMsg, int& which, bool& down) { switch(uMsg) { case WM_LBUTTONDOWN: which = 0; down = true; break; case WM_MBUTTONDOWN: which = 1; down = true; break; case WM_RBUTTONDOWN: which = 2; down = true; break; case WM_LBUTTONUP: which = 0; down = false; break; case WM_MBUTTONUP: which = 1; down = false; break; case WM_RBUTTONUP: which = 2; down = false; break; } } // message handler (includes timer, Windows msgs) // LRESULT CALLBACK WndProc( HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam ) { switch(uMsg) { case WM_ERASEBKGND: return 0; case WM_KEYDOWN: if(!window_ready) return 0; if (fullscreen) { boinc_close_window_and_quit("key down"); } else { boinc_app_key_press((int)wParam, (int)lParam); } return 0; case WM_KEYUP: if(!window_ready) return 0; if (fullscreen) { boinc_close_window_and_quit("key up"); } else { boinc_app_key_release((int)wParam, (int)lParam); } return 0; case WM_LBUTTONDOWN: case WM_MBUTTONDOWN: case WM_RBUTTONDOWN: case WM_LBUTTONUP: case WM_MBUTTONUP: case WM_RBUTTONUP: if(!window_ready) return 0; if (fullscreen) { boinc_close_window_and_quit("button up"); } else { int which; bool down; parse_mouse_event(uMsg, which, down); boinc_app_mouse_button( (int)(short)LOWORD(lParam), (int)(short)HIWORD(lParam), which, down ); } return 0; case WM_MOUSEMOVE: if(!window_ready) return 0; if (fullscreen) { if((int)(short)LOWORD(lParam) != mousePos.x || (int)(short)HIWORD(lParam) != mousePos.y) { boinc_close_window_and_quit("mouse move"); } } else { boinc_app_mouse_move( (int)(short)LOWORD(lParam), (int)(short)HIWORD(lParam), (wParam&MK_LBUTTON)!=0, (wParam&MK_MBUTTON)!=0, (wParam&MK_RBUTTON)!=0 ); } return 0; case WM_CLOSE: boinc_close_window_and_quit("WM_CLOSE"); return 0; case WM_DESTROY: PostQuitMessage(0); return 0; case WM_SHUTDOWNGFX: CloseWindow(hWnd); return 0; case WM_PAINT: PAINTSTRUCT ps; RECT winRect; HDC pdc; pdc = BeginPaint(hWnd, &ps); GetClientRect(hWnd, &winRect); FillRect(pdc, &winRect, (HBRUSH)GetStockObject(BLACK_BRUSH)); EndPaint(hWnd, &ps); return 0; case WM_SIZE: if ( SIZE_MINIMIZED == wParam ) { visible = FALSE; } else { visible = TRUE; } if(!window_ready) return 0; app_graphics_resize(LOWORD(lParam), HIWORD(lParam)); return 0; } return DefWindowProc(hWnd, uMsg, wParam, lParam); } BOOL reg_win_class() { WNDCLASS wc; wc.style = CS_HREDRAW | CS_VREDRAW | CS_OWNDC; wc.lpfnWndProc = (WNDPROC) WndProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = instance; wc.hIcon = LoadIcon(NULL, IDI_WINLOGO); wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = NULL; wc.lpszMenuName = NULL; wc.lpszClassName = BOINC_WINDOW_CLASS_NAME; if (!RegisterClass(&wc)) { MessageBox( NULL, "RegisterClass() failed.", "Error", MB_OK|MB_ICONEXCLAMATION ); return FALSE; } return TRUE; } BOOL unreg_win_class() { if (!UnregisterClass(BOINC_WINDOW_CLASS_NAME, instance)) { MessageBox( NULL, "UnregisterClass() failed.", "ERROR", MB_OK|MB_ICONINFORMATION ); instance = NULL; } return TRUE; } static VOID CALLBACK timer_handler(HWND, UINT, UINT, DWORD) { RECT rt; int width, height; static int size_changed = 0; GetWindowRect(window, &rt); width = rt.right-rt.left; height = rt.bottom-rt.top; if (throttled_app_render(width, height, dtime())) { SwapBuffers(win_dc); if (!fullscreen) { // If user has changed window size, wait until it stops // changing and then write the new dimensions to file // if ((rt.left != rect.left) || (rt.top != rect.top) || (rt.right != rect.right) || (rt.bottom != rect.bottom) ) { if (IsZoomed(window)) return; if ((rt.left < 0) && (rt.right < 0)) return; if ((rt.top < 0) && (rt.bottom < 0)) return; size_changed = 1; rect.left = rt.left; rect.top = rt.top; rect.right = rt.right; rect.bottom = rt.bottom; } else { if (size_changed && (++size_changed > 10)) { size_changed = 0; FILE *f = boinc_fopen("gfx_info", "w"); if (f) { // ToDo: change this to XML fprintf(f, "%d %d %d %d\n", rect.left, rect.top, rect.right, rect.bottom ); fclose(f); } } } // End if (new size != previous size) else } } } void boinc_graphics_loop(int argc, char** argv, const char* title) { char buf[256]; if (!diagnostics_is_initialized()) { boinc_init_graphics_diagnostics(BOINC_DIAG_DEFAULTS); } fprintf(stderr, "%s Starting graphics application.\n", boinc_msg_prefix(buf, sizeof(buf)) ); for (int i=1; i