#include "lc_global.h" #include "opengl.h" #include "glwindow.h" #include "system.h" #include "tools.h" #include "resource.h" #ifndef WGL_ARB_extensions_string #define WGL_ARB_extensions_string 1 typedef const char* (WINAPI * PFNWGLGETEXTENSIONSSTRINGARBPROC) (HDC hdc); PFNWGLGETEXTENSIONSSTRINGARBPROC wglGetExtensionsStringARB = NULL; #endif // WGL_ARB_extensions_string #ifndef WGL_ARB_multisample #define WGL_SAMPLE_BUFFERS_ARB 0x2041 #define WGL_SAMPLES_ARB 0x2042 #endif #ifndef WGL_ARB_pixel_format #define WGL_ARB_pixel_format 1 #define WGL_NUMBER_PIXEL_FORMATS_ARB 0x2000 #define WGL_DRAW_TO_WINDOW_ARB 0x2001 #define WGL_DRAW_TO_BITMAP_ARB 0x2002 #define WGL_ACCELERATION_ARB 0x2003 #define WGL_NEED_PALETTE_ARB 0x2004 #define WGL_NEED_SYSTEM_PALETTE_ARB 0x2005 #define WGL_SWAP_LAYER_BUFFERS_ARB 0x2006 #define WGL_SWAP_METHOD_ARB 0x2007 #define WGL_NUMBER_OVERLAYS_ARB 0x2008 #define WGL_NUMBER_UNDERLAYS_ARB 0x2009 #define WGL_TRANSPARENT_ARB 0x200A #define WGL_SHARE_DEPTH_ARB 0x200C #define WGL_SHARE_STENCIL_ARB 0x200D #define WGL_SHARE_ACCUM_ARB 0x200E #define WGL_SUPPORT_GDI_ARB 0x200F #define WGL_SUPPORT_OPENGL_ARB 0x2010 #define WGL_DOUBLE_BUFFER_ARB 0x2011 #define WGL_STEREO_ARB 0x2012 #define WGL_PIXEL_TYPE_ARB 0x2013 #define WGL_COLOR_BITS_ARB 0x2014 #define WGL_RED_BITS_ARB 0x2015 #define WGL_RED_SHIFT_ARB 0x2016 #define WGL_GREEN_BITS_ARB 0x2017 #define WGL_GREEN_SHIFT_ARB 0x2018 #define WGL_BLUE_BITS_ARB 0x2019 #define WGL_BLUE_SHIFT_ARB 0x201A #define WGL_ALPHA_BITS_ARB 0x201B #define WGL_ALPHA_SHIFT_ARB 0x201C #define WGL_ACCUM_BITS_ARB 0x201D #define WGL_ACCUM_RED_BITS_ARB 0x201E #define WGL_ACCUM_GREEN_BITS_ARB 0x201F #define WGL_ACCUM_BLUE_BITS_ARB 0x2020 #define WGL_ACCUM_ALPHA_BITS_ARB 0x2021 #define WGL_DEPTH_BITS_ARB 0x2022 #define WGL_STENCIL_BITS_ARB 0x2023 #define WGL_AUX_BUFFERS_ARB 0x2024 #define WGL_NO_ACCELERATION_ARB 0x2025 #define WGL_GENERIC_ACCELERATION_ARB 0x2026 #define WGL_FULL_ACCELERATION_ARB 0x2027 #define WGL_SWAP_EXCHANGE_ARB 0x2028 #define WGL_SWAP_COPY_ARB 0x2029 #define WGL_SWAP_UNDEFINED_ARB 0x202A #define WGL_TYPE_RGBA_ARB 0x202B #define WGL_TYPE_COLORINDEX_ARB 0x202C #define WGL_TRANSPARENT_RED_VALUE_ARB 0x2037 #define WGL_TRANSPARENT_GREEN_VALUE_ARB 0x2038 #define WGL_TRANSPARENT_BLUE_VALUE_ARB 0x2039 #define WGL_TRANSPARENT_ALPHA_VALUE_ARB 0x203A #define WGL_TRANSPARENT_INDEX_VALUE_ARB 0x203B typedef BOOL (WINAPI * PFNWGLCHOOSEPIXELFORMATARBPROC) (HDC hdc, const int* piAttribIList, const FLOAT *pfAttribFList, UINT nMaxFormats, int *piFormats, UINT *nNumFormats); typedef BOOL (WINAPI * PFNWGLGETPIXELFORMATATTRIBFVARBPROC) (HDC hdc, int iPixelFormat, int iLayerPlane, UINT nAttributes, const int* piAttributes, FLOAT *pfValues); typedef BOOL (WINAPI * PFNWGLGETPIXELFORMATATTRIBIVARBPROC) (HDC hdc, int iPixelFormat, int iLayerPlane, UINT nAttributes, const int* piAttributes, int *piValues); PFNWGLCHOOSEPIXELFORMATARBPROC wglChoosePixelFormatARB = NULL; PFNWGLGETPIXELFORMATATTRIBFVARBPROC wglGetPixelFormatAttribfvARB = NULL; PFNWGLGETPIXELFORMATATTRIBIVARBPROC wglGetPixelFormatAttribivARB = NULL; #endif // WGL_ARB_pixel_format struct GLWindowPrivate { HGLRC m_hrc; CDC* m_pDC; CPalette* m_pPal; HWND m_hWnd; HCURSOR Cursor; bool Multisample; }; // ============================================================================ BOOL GLWindowPreTranslateMessage(GLWindow *wnd, MSG *pMsg) { switch (pMsg->message) { case WM_PAINT: { GLWindowPrivate* prv = (GLWindowPrivate*)wnd->GetData(); PAINTSTRUCT ps; BeginPaint(prv->m_hWnd, &ps); wnd->OnDraw(); EndPaint(prv->m_hWnd, &ps); } break; case WM_SIZE: wnd->OnSize(LOWORD(pMsg->lParam), HIWORD(pMsg->lParam)); break; case WM_LBUTTONDOWN: wnd->OnLeftButtonDown((SHORT)LOWORD(pMsg->lParam), wnd->GetHeight() - (SHORT)HIWORD(pMsg->lParam) - 1, (pMsg->wParam & MK_CONTROL) != 0, (pMsg->wParam & MK_SHIFT) != 0); break; case WM_LBUTTONUP: wnd->OnLeftButtonUp((SHORT)LOWORD(pMsg->lParam), wnd->GetHeight() - (SHORT)HIWORD(pMsg->lParam) - 1, (pMsg->wParam & MK_CONTROL) != 0, (pMsg->wParam & MK_SHIFT) != 0); break; case WM_LBUTTONDBLCLK: wnd->OnLeftButtonDoubleClick((SHORT)LOWORD(pMsg->lParam), wnd->GetHeight() - (SHORT)HIWORD(pMsg->lParam) - 1, (pMsg->wParam & MK_CONTROL) != 0, (pMsg->wParam & MK_SHIFT) != 0); break; case WM_MBUTTONDOWN: wnd->OnMiddleButtonDown((SHORT)LOWORD(pMsg->lParam), wnd->GetHeight() - (SHORT)HIWORD(pMsg->lParam) - 1, (pMsg->wParam & MK_CONTROL) != 0, (pMsg->wParam & MK_SHIFT) != 0); break; case WM_MBUTTONUP: wnd->OnMiddleButtonUp((SHORT)LOWORD(pMsg->lParam), wnd->GetHeight() - (SHORT)HIWORD(pMsg->lParam) - 1, (pMsg->wParam & MK_CONTROL) != 0, (pMsg->wParam & MK_SHIFT) != 0); break; case WM_RBUTTONDOWN: wnd->OnRightButtonDown((SHORT)LOWORD(pMsg->lParam), wnd->GetHeight() - (SHORT)HIWORD(pMsg->lParam) - 1, (pMsg->wParam & MK_CONTROL) != 0, (pMsg->wParam & MK_SHIFT) != 0); break; case WM_RBUTTONUP: wnd->OnRightButtonUp((SHORT)LOWORD(pMsg->lParam), wnd->GetHeight() - (SHORT)HIWORD(pMsg->lParam) - 1, (pMsg->wParam & MK_CONTROL) != 0, (pMsg->wParam & MK_SHIFT) != 0); break; case WM_MOUSEMOVE: wnd->OnMouseMove((SHORT)LOWORD(pMsg->lParam), wnd->GetHeight() - (SHORT)HIWORD(pMsg->lParam) - 1, (pMsg->wParam & MK_CONTROL) != 0, (pMsg->wParam & MK_SHIFT) != 0); break; case WM_ERASEBKGND: return TRUE; case WM_CREATE: wnd->OnInitialUpdate(); break; case WM_DESTROY: wnd->DestroyContext(); return FALSE; break; case WM_SETCURSOR: { GLWindowPrivate *prv = (GLWindowPrivate*)wnd->GetData(); if (prv->Cursor) { SetCursor(prv->Cursor); return TRUE; } return FALSE; } break; case WM_PALETTECHANGED: if ((HWND)pMsg->wParam == pMsg->hwnd) // Responding to own message. break; case WM_QUERYNEWPALETTE: { GLWindowPrivate *prv = (GLWindowPrivate*)wnd->GetData(); if (prv->m_pPal) { prv->m_pDC->SelectPalette(prv->m_pPal, FALSE); if (prv->m_pDC->RealizePalette() != 0) { // Some colors changed, so we need to do a repaint. InvalidateRect(prv->m_hWnd, NULL, TRUE); } } } break; default: return FALSE; } return TRUE; } LRESULT CALLBACK GLWindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { static CMapPtrToPtr WindowMap; GLWindow *wnd; if (uMsg == WM_CREATE) { LPCREATESTRUCT cs = (LPCREATESTRUCT)lParam; wnd = (GLWindow*)cs->lpCreateParams; wnd->CreateFromWindow(hwnd); WindowMap.SetAt(hwnd, wnd); } wnd = (GLWindow*)WindowMap[hwnd]; if (wnd) { MSG msg; msg.hwnd = hwnd; msg.message = uMsg; msg.wParam = wParam; msg.lParam = lParam; GLWindowPreTranslateMessage(wnd, &msg); if (uMsg == WM_DESTROY) { WindowMap.RemoveKey(hwnd); } } return DefWindowProc(hwnd, uMsg, wParam, lParam); } static LRESULT CALLBACK TempWindowProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { if (uMsg != WM_CREATE) return DefWindowProc(hWnd, uMsg, wParam, lParam); HDC hDC = GetDC(hWnd); PIXELFORMATDESCRIPTOR pfd; memset(&pfd, 0, sizeof(PIXELFORMATDESCRIPTOR)); pfd.nSize = sizeof(PIXELFORMATDESCRIPTOR); pfd.nVersion = 1; pfd.dwFlags = PFD_SUPPORT_OPENGL | PFD_DOUBLEBUFFER | PFD_DRAW_TO_WINDOW; pfd.iPixelType = PFD_TYPE_RGBA; pfd.cColorBits = 24; pfd.cDepthBits = 24; pfd.iLayerType = PFD_MAIN_PLANE; int PixelFormat = ChoosePixelFormat(hDC, &pfd); if (PixelFormat == 0) return 0; if (!SetPixelFormat(hDC, PixelFormat, &pfd)) return 0; HGLRC hGLRC = wglCreateContext(hDC); wglMakeCurrent(hDC, hGLRC); const GLubyte* Extensions; wglGetExtensionsStringARB = (PFNWGLGETEXTENSIONSSTRINGARBPROC)wglGetProcAddress("wglGetExtensionsStringARB"); if (wglGetExtensionsStringARB) Extensions = (GLubyte*)wglGetExtensionsStringARB(wglGetCurrentDC()); else Extensions = glGetString(GL_EXTENSIONS); GL_InitializeSharedExtensions(); if (GL_ExtensionSupported(Extensions, "WGL_ARB_pixel_format")) { wglChoosePixelFormatARB = (PFNWGLCHOOSEPIXELFORMATARBPROC)wglGetProcAddress("wglChoosePixelFormatARB"); wglGetPixelFormatAttribfvARB = (PFNWGLGETPIXELFORMATATTRIBFVARBPROC)wglGetProcAddress("wglGetPixelFormatAttribfvARB"); wglGetPixelFormatAttribivARB = (PFNWGLGETPIXELFORMATATTRIBIVARBPROC)wglGetProcAddress("wglGetPixelFormatAttribivARB"); } wglMakeCurrent(NULL, NULL); wglDeleteContext(hGLRC); ReleaseDC(hWnd, hDC); return 0; } void GL_InitializeExtensions() { HINSTANCE hInstance; WNDCLASS wc; hInstance = GetModuleHandle(NULL); wc.style = CS_OWNDC; wc.lpfnWndProc = TempWindowProc; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = hInstance; wc.hIcon = LoadIcon(NULL, IDI_WINLOGO); wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = (HBRUSH)GetStockObject(BLACK_BRUSH); wc.lpszMenuName = NULL; wc.lpszClassName = "LeoCADGLTempWindow"; RegisterClass(&wc); HWND hWnd = CreateWindow(wc.lpszClassName, "LeoCAD Temp Window", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, NULL, NULL, hInstance, NULL); if (hWnd) DestroyWindow(hWnd); } // ============================================================================ // GLWindow class GLWindow::GLWindow(GLWindow *share) { m_pShare = share; m_pData = (GLWindowPrivate*)malloc(sizeof(GLWindowPrivate)); memset(m_pData, 0, sizeof(GLWindowPrivate)); } GLWindow::~GLWindow() { GLWindowPrivate *prv = (GLWindowPrivate*)m_pData; delete prv->m_pDC; free(m_pData); } bool GLWindow::CreateFromWindow(void* data) { GLWindowPrivate* prv = (GLWindowPrivate*)m_pData; prv->m_hWnd = (HWND)data; prv->m_pDC = new CClientDC(CWnd::FromHandle(prv->m_hWnd)); ASSERT(prv->m_pDC != NULL); // Fill in the Pixel Format Descriptor PIXELFORMATDESCRIPTOR pfd; memset(&pfd, 0, sizeof(PIXELFORMATDESCRIPTOR)); pfd.nSize = sizeof(PIXELFORMATDESCRIPTOR); pfd.nVersion = 1; pfd.dwFlags = PFD_SUPPORT_OPENGL | PFD_DOUBLEBUFFER | PFD_DRAW_TO_WINDOW; pfd.iPixelType = PFD_TYPE_RGBA; pfd.cColorBits = 24; pfd.cDepthBits = 24; pfd.iLayerType = PFD_MAIN_PLANE; int PixelFormat = 0; int AASamples = Sys_ProfileLoadInt("Default", "AASamples", 1); if (wglChoosePixelFormatARB && AASamples > 1) { // Choose a Pixel Format Descriptor (PFD) with multisampling support. int iAttributes[] = { WGL_DRAW_TO_WINDOW_ARB, GL_TRUE, WGL_SUPPORT_OPENGL_ARB, GL_TRUE, WGL_ACCELERATION_ARB, WGL_FULL_ACCELERATION_ARB, WGL_COLOR_BITS_ARB, 24, WGL_DEPTH_BITS_ARB, 24, WGL_STENCIL_BITS_ARB, 0, WGL_DOUBLE_BUFFER_ARB, GL_TRUE, WGL_SAMPLE_BUFFERS_ARB, GL_TRUE, WGL_SAMPLES_ARB, AASamples, 0, 0 }; float fAttributes[] = {0,0}; UINT NumFormats; int MultisamplePixelFormat; int Status = wglChoosePixelFormatARB(prv->m_pDC->m_hDC, iAttributes, fAttributes, 1, &MultisamplePixelFormat, &NumFormats); if (Status && NumFormats > 0) { PixelFormat = MultisamplePixelFormat; prv->Multisample = true; } } if (!PixelFormat) PixelFormat = ChoosePixelFormat(prv->m_pDC->m_hDC, &pfd); if (!PixelFormat) return false; if (!SetPixelFormat(prv->m_pDC->m_hDC, PixelFormat, &pfd)) return false; prv->m_pPal = new CPalette; if (CreateRGBPalette(prv->m_pDC->m_hDC, &prv->m_pPal)) { prv->m_pDC->SelectPalette(prv->m_pPal, FALSE); prv->m_pDC->RealizePalette(); } else { delete prv->m_pPal; prv->m_pPal = NULL; } // Create a rendering context. prv->m_hrc = wglCreateContext(prv->m_pDC->m_hDC); if (!prv->m_hrc) return false; if (m_pShare) { GLWindowPrivate *share = (GLWindowPrivate*)m_pShare->m_pData; wglShareLists(share->m_hrc, prv->m_hrc); } return true; } bool GLWindow::CreateFromBitmap(void* Data) { GLWindowPrivate* prv = (GLWindowPrivate*)m_pData; prv->m_pDC = new CDC; prv->m_pDC->Attach((HDC)Data); ASSERT(prv->m_pDC != NULL); // Fill in the Pixel Format Descriptor PIXELFORMATDESCRIPTOR pfd; memset(&pfd, 0, sizeof(PIXELFORMATDESCRIPTOR)); pfd.nSize = sizeof(PIXELFORMATDESCRIPTOR); pfd.nVersion = 1; pfd.dwFlags = PFD_DRAW_TO_BITMAP | PFD_SUPPORT_OPENGL | PFD_SUPPORT_GDI; pfd.iPixelType = PFD_TYPE_RGBA; pfd.cColorBits = 24; pfd.cDepthBits = 16; pfd.iLayerType = PFD_MAIN_PLANE; int nPixelFormat = ChoosePixelFormat(prv->m_pDC->m_hDC, &pfd); if (nPixelFormat == 0) return false; if (!SetPixelFormat(prv->m_pDC->m_hDC, nPixelFormat, &pfd)) return false; prv->m_pPal = new CPalette; if (CreateRGBPalette(prv->m_pDC->m_hDC, &prv->m_pPal)) { prv->m_pDC->SelectPalette(prv->m_pPal, FALSE); prv->m_pDC->RealizePalette(); } else { delete prv->m_pPal; prv->m_pPal = NULL; } // Create a rendering context. prv->m_hrc = wglCreateContext(prv->m_pDC->m_hDC); if (!prv->m_hrc) return false; if (m_pShare) { GLWindowPrivate *share = (GLWindowPrivate*)m_pShare->m_pData; wglShareLists(share->m_hrc, prv->m_hrc); } return true; } void GLWindow::DestroyContext() { GLWindowPrivate *prv = (GLWindowPrivate*)m_pData; if (prv->m_pPal) { CPalette palDefault; palDefault.CreateStockObject(DEFAULT_PALETTE); prv->m_pDC->SelectPalette(&palDefault, FALSE); delete prv->m_pPal; prv->m_pPal = NULL; } if (prv->m_hrc) wglDeleteContext(prv->m_hrc); prv->m_hrc = NULL; if (prv->m_pDC) delete prv->m_pDC; prv->m_pDC = NULL; } void GLWindow::OnInitialUpdate() { GLWindowPrivate *prv = (GLWindowPrivate*)m_pData; MakeCurrent(); if (prv->Multisample) glEnable(GL_MULTISAMPLE_ARB); } bool GLWindow::MakeCurrent() { GLWindowPrivate *prv = (GLWindowPrivate*)m_pData; if (prv->m_pPal) { prv->m_pDC->SelectPalette(prv->m_pPal, FALSE); prv->m_pDC->RealizePalette(); } return (wglMakeCurrent(prv->m_pDC->m_hDC, prv->m_hrc) != 0); } void GLWindow::SwapBuffers() { GLWindowPrivate *prv = (GLWindowPrivate*)m_pData; ::SwapBuffers(prv->m_pDC->m_hDC); } void GLWindow::Redraw(bool ForceRedraw) { GLWindowPrivate *prv = (GLWindowPrivate*)m_pData; InvalidateRect(prv->m_hWnd, NULL, FALSE); if (ForceRedraw) UpdateWindow(prv->m_hWnd); } void GLWindow::CaptureMouse() { GLWindowPrivate* prv = (GLWindowPrivate*)m_pData; SetCapture(prv->m_hWnd); } void GLWindow::ReleaseMouse() { ReleaseCapture(); } void GLWindow::SetCursor(LC_CURSOR_TYPE Cursor) { const UINT CursorResources[LC_CURSOR_COUNT] = { 0, // LC_CURSOR_DEFAULT IDC_BRICK, // LC_CURSOR_BRICK IDC_LIGHT, // LC_CURSOR_LIGHT IDC_SPOTLIGHT, // LC_CURSOR_SPOTLIGHT IDC_CAMERA, // LC_CURSOR_CAMERA IDC_SELECT, // LC_CURSOR_SELECT IDC_SELECT_GROUP, // LC_CURSOR_SELECT_GROUP IDC_MOVE, // LC_CURSOR_MOVE IDC_ROTATE, // LC_CURSOR_ROTATE IDC_ROTX, // LC_CURSOR_ROTATEX IDC_ROTY, // LC_CURSOR_ROTATEY IDC_ERASER, // LC_CURSOR_DELETE IDC_PAINT, // LC_CURSOR_PAINT IDC_ZOOM, // LC_CURSOR_ZOOM IDC_ZOOM_REGION, // LC_CURSOR_ZOOM_REGION IDC_PAN, // LC_CURSOR_PAN IDC_ROLL, // LC_CURSOR_ROLL IDC_ANGLE, // LC_CURSOR_ROTATE_VIEW }; GLWindowPrivate *prv = (GLWindowPrivate*)m_pData; if (CursorResources[Cursor]) prv->Cursor = AfxGetApp()->LoadCursor(CursorResources[Cursor]); else prv->Cursor = NULL; }