Changeset 326 for trunk/src/gl/gl_window.cc
- Timestamp:
- 08/26/14 18:39:10 (11 years ago)
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/gl/gl_window.cc
r323 r326 7 7 #include "nv/core/logging.hh" 8 8 #include "nv/lib/gl.hh" 9 #include "nv/lib/sdl.hh"10 9 11 10 using namespace nv; 12 11 13 static bool sdl_key_event_to_io_event( const SDL_KeyboardEvent& ke, io_event& kevent )14 {15 kevent.type = EV_KEY;16 kevent.key.pressed = ( ke.state != SDL_RELEASED );17 kevent.key.ascii = 0;18 kevent.key.code = KEY_NONE;19 20 #if NV_SDL_VERSION == NV_SDL_2021 uint32 ucode = (uint32)ke.keysym.sym;22 #else23 uint32 ucode = (uint32)ke.keysym.unicode;24 #endif25 26 // if result is a typable char place it into the structure27 if (ucode >= 32 && ucode < 128 )28 {29 kevent.key.ascii = static_cast<char8>( ucode );30 #if NV_SDL_VERSION == NV_SDL_2031 if (ucode >= 'a' && ucode <= 'z')32 {33 int shifted = !!(ke.keysym.mod & KMOD_SHIFT);34 int capslock = !!(ke.keysym.mod & KMOD_CAPS);35 if ((shifted ^ capslock) != 0) {36 kevent.key.ascii = (char8)SDL_toupper((int)ucode);37 }38 }39 #endif40 }41 42 // Get the Key value from the SDL Keyboard event43 int result = ke.keysym.sym;44 45 // Check the control and shift states46 kevent.key.alt = (ke.keysym.mod & KMOD_ALT ) != 0;47 kevent.key.control = (ke.keysym.mod & KMOD_CTRL ) != 0;48 kevent.key.shift = (ke.keysym.mod & KMOD_SHIFT ) != 0;49 50 // if result is a typable char from the directly transformable ranges51 if ( ( result >= KEY_A && result <= KEY_Z ) ||52 ( result >= KEY_0 && result <= KEY_9 ) ||53 result == ' ' )54 {55 kevent.key.code = static_cast<key_code>(result);56 }57 58 // if result is a typable char from the a..z range59 if ( result >= KEY_A + 32 && result <= KEY_Z + 32 )60 {61 kevent.key.code = static_cast<key_code>(result - 32);62 }63 64 if ( result >= SDLK_F1 && result <= SDLK_F12 )65 {66 kevent.key.code = static_cast<key_code>(result - SDLK_F1 + KEY_F1 );67 }68 69 // other recognized codes70 switch ( result )71 {72 case SDLK_BACKSPACE : kevent.key.code = KEY_BACK; break;73 case SDLK_TAB : kevent.key.code = KEY_TAB; break;74 case SDLK_RETURN : kevent.key.code = KEY_ENTER; break;75 case SDLK_PAGEUP : kevent.key.code = KEY_PGUP; break;76 case SDLK_PAGEDOWN : kevent.key.code = KEY_PGDOWN; break;77 case SDLK_END : kevent.key.code = KEY_END; break;78 case SDLK_HOME : kevent.key.code = KEY_HOME; break;79 case SDLK_LEFT : kevent.key.code = KEY_LEFT; break;80 case SDLK_UP : kevent.key.code = KEY_UP; break;81 case SDLK_RIGHT : kevent.key.code = KEY_RIGHT; break;82 case SDLK_DOWN : kevent.key.code = KEY_DOWN; break;83 case SDLK_DELETE : kevent.key.code = KEY_DELETE; break;84 case SDLK_INSERT : kevent.key.code = KEY_INSERT; break;85 //case SDLK_KP5 : kevent.key.code = KEY_CENTER; break;86 case SDLK_ESCAPE : kevent.key.code = KEY_ESCAPE; break;87 case SDLK_QUOTE : kevent.key.code = KEY_QUOTE; break;88 case SDLK_COMMA : kevent.key.code = KEY_COMMA; break;89 case SDLK_MINUS : kevent.key.code = KEY_MINUS; break;90 case SDLK_PERIOD : kevent.key.code = KEY_PERIOD; break;91 case SDLK_SLASH : kevent.key.code = KEY_SLASH; break;92 case SDLK_SEMICOLON : kevent.key.code = KEY_SCOLON; break;93 case SDLK_LEFTBRACKET : kevent.key.code = KEY_LBRACKET; break;94 case SDLK_BACKSLASH : kevent.key.code = KEY_BSLASH; break;95 case SDLK_RIGHTBRACKET : kevent.key.code = KEY_RBRACKET; break;96 case SDLK_BACKQUOTE : kevent.key.code = KEY_BQUOTE; break;97 default : break;98 }99 100 // If key was understood by nv, then it's valid, otherwise ignored101 return kevent.key.ascii != 0 || kevent.key.code != 0;102 }103 104 static bool sdl_mouse_button_to_io_event( const SDL_MouseButtonEvent& mb, io_event& mevent )105 {106 mevent.type = EV_MOUSE_BUTTON;107 mevent.mbutton.button = MOUSE_NONE;108 mevent.mbutton.pressed = (mb.state != SDL_RELEASED);109 mevent.mbutton.x = static_cast< uint16 >( mb.x );110 mevent.mbutton.y = static_cast< uint16 >( mb.y );111 112 switch ( mb.button )113 {114 case SDL_BUTTON_LEFT : mevent.mbutton.button = MOUSE_LEFT; break;115 case SDL_BUTTON_MIDDLE : mevent.mbutton.button = MOUSE_MIDDLE; break;116 case SDL_BUTTON_RIGHT : mevent.mbutton.button = MOUSE_RIGHT; break;117 #if NV_SDL_VERSION == NV_SDL_12118 case SDL_BUTTON_WHEELUP :119 mevent.type = EV_MOUSE_WHEEL;120 mevent.mwheel.x = 0;121 mevent.mwheel.y = 3;122 return true;123 case SDL_BUTTON_WHEELDOWN :124 mevent.type = EV_MOUSE_WHEEL;125 mevent.mwheel.x = 0;126 mevent.mwheel.y = -3;127 return true;128 #endif129 default : break;130 }131 132 return mevent.mbutton.button != MOUSE_NONE;133 }134 135 #if NV_SDL_VERSION == NV_SDL_20136 static bool sdl_mouse_wheel_to_io_event( const SDL_MouseWheelEvent& mm, io_event& mevent )137 {138 mevent.type = EV_MOUSE_WHEEL;139 mevent.mwheel.x = static_cast< sint32 >( mm.x );140 mevent.mwheel.y = static_cast< sint32 >( mm.y );141 return true;142 }143 #endif144 145 static bool sdl_mouse_motion_to_io_event( const SDL_MouseMotionEvent& mm, io_event& mevent )146 {147 mevent.type = EV_MOUSE_MOVE;148 mevent.mmove.pressed = (mm.state != SDL_RELEASED);149 mevent.mmove.x = static_cast< uint16 >( mm.x );150 mevent.mmove.y = static_cast< uint16 >( mm.y );151 mevent.mmove.rx = static_cast< sint16 >( mm.xrel );152 mevent.mmove.ry = static_cast< sint16 >( mm.yrel );153 return true;154 }155 156 static bool sdl_joy_button_event_to_io_event( const SDL_JoyButtonEvent& jb, io_event& jevent )157 {158 jevent.type = EV_JOY_BUTTON;159 jevent.jbutton.id = jb.which;160 jevent.jbutton.button = jb.button;161 jevent.jbutton.pressed = (jb.type == SDL_PRESSED);162 return true;163 }164 165 static bool sdl_joy_axis_event_to_io_event( const SDL_JoyAxisEvent& ja, io_event& jevent )166 {167 jevent.type = EV_JOY_AXIS;168 jevent.jaxis.id = ja.which;169 jevent.jaxis.axis = ja.axis;170 jevent.jaxis.value = ja.value;171 return true;172 }173 174 static bool sdl_joy_hat_event_to_io_event( const SDL_JoyHatEvent& jh, io_event& jevent )175 {176 jevent.type = EV_JOY_HAT;177 jevent.jhat.id = jh.which;178 jevent.jhat.hat = jh.hat;179 jevent.jhat.value = jh.value;180 return true;181 }182 183 static bool sdl_joy_ball_event_to_io_event( const SDL_JoyBallEvent& jb, io_event& jevent )184 {185 jevent.type = EV_JOY_HAT;186 jevent.jball.id = jb.which;187 jevent.jball.ball = jb.ball;188 jevent.jball.rx = jb.xrel;189 jevent.jball.ry = jb.yrel;190 return true;191 }192 193 static bool sdl_event_to_io_event( const SDL_Event& e, io_event& ioevent )194 {195 switch ( e.type )196 {197 case SDL_KEYDOWN : return sdl_key_event_to_io_event( e.key, ioevent );198 case SDL_KEYUP : return sdl_key_event_to_io_event( e.key, ioevent );199 case SDL_MOUSEMOTION : return sdl_mouse_motion_to_io_event( e.motion, ioevent );200 case SDL_MOUSEBUTTONDOWN : return sdl_mouse_button_to_io_event( e.button, ioevent );201 case SDL_MOUSEBUTTONUP : return sdl_mouse_button_to_io_event( e.button, ioevent );202 #if NV_SDL_VERSION == NV_SDL_20203 case SDL_MOUSEWHEEL : return sdl_mouse_wheel_to_io_event( e.wheel, ioevent );204 #endif205 /* // SDL 2.0 incompatible206 case SDL_ACTIVEEVENT :207 ioevent.type = EV_ACTIVE;208 ioevent.active.gain = (e.active.gain != 0);209 return e.active.state == SDL_APPINPUTFOCUS;210 case SDL_VIDEORESIZE :211 ioevent.type = EV_RESIZE;212 ioevent.resize.x = e.resize.w;213 ioevent.resize.y = e.resize.h;214 return true;215 case SDL_NOEVENT : return false;216 case SDL_VIDEOEXPOSE : return false;217 */218 case SDL_SYSWMEVENT : ioevent.type = EV_SYSTEM; return true;219 case SDL_QUIT : ioevent.type = EV_QUIT; return true;220 case SDL_JOYAXISMOTION : return sdl_joy_axis_event_to_io_event( e.jaxis, ioevent );221 case SDL_JOYBALLMOTION : return sdl_joy_ball_event_to_io_event( e.jball, ioevent );222 case SDL_JOYHATMOTION : return sdl_joy_hat_event_to_io_event( e.jhat, ioevent );223 case SDL_JOYBUTTONDOWN : return sdl_joy_button_event_to_io_event( e.jbutton, ioevent );224 case SDL_JOYBUTTONUP : return sdl_joy_button_event_to_io_event( e.jbutton, ioevent );225 default : return false;226 }227 }228 229 230 gl_window::gl_window( device* dev, uint16 width, uint16 height, bool fullscreen )231 : m_device( dev ), m_width( width ), m_height( height ), m_title("NV Engine"), m_handle( nullptr ), m_hwnd( 0 ), m_adopted( false )232 {233 // bpp = m_info->vfmt->BitsPerPixel;234 235 SDL_GL_SetAttribute( SDL_GL_RED_SIZE, 8 );236 SDL_GL_SetAttribute( SDL_GL_GREEN_SIZE, 8 );237 SDL_GL_SetAttribute( SDL_GL_BLUE_SIZE, 8 );238 SDL_GL_SetAttribute( SDL_GL_DEPTH_SIZE, 24 );239 SDL_GL_SetAttribute( SDL_GL_DOUBLEBUFFER, 1 );240 241 // SDL_GL_SetAttribute( SDL_GL_MULTISAMPLEBUFFERS, 1 );242 // SDL_GL_SetAttribute( SDL_GL_MULTISAMPLESAMPLES, 4 );243 244 #if NV_SDL_VERSION == NV_SDL_20245 SDL_GL_SetAttribute(SDL_GL_CONTEXT_MAJOR_VERSION, 2);246 SDL_GL_SetAttribute(SDL_GL_CONTEXT_MINOR_VERSION, 1);247 SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 1);248 SDL_GL_SetAttribute(SDL_GL_DEPTH_SIZE, 24);249 #endif250 251 252 #if NV_SDL_VERSION == NV_SDL_12253 uint32 flags = SDL_OPENGL;254 if (fullscreen) flags |= SDL_FULLSCREEN;255 m_handle = SDL_SetVideoMode( width, height, 32, flags );256 #elif NV_SDL_VERSION == NV_SDL_20257 uint32 flags = SDL_WINDOW_OPENGL | SDL_WINDOW_SHOWN;258 if (fullscreen) flags |= SDL_WINDOW_FULLSCREEN;259 m_handle = SDL_CreateWindow("Nova Engine", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED,260 width, height, flags );261 #endif262 if ( m_handle == 0 )263 {264 NV_LOG( LOG_CRITICAL, "Video mode set failed: " << SDL_GetError( ) );265 return; // TODO: Error report266 }267 268 // NV_LOG( LOG_INFO, "Joystick count : " << SDL_NumJoysticks() );269 // SDL_Joystick* j = SDL_JoystickOpen(0);270 // if (j)271 // {272 // NV_LOG( LOG_INFO, "Joystick Name: " << SDL_JoystickNameForIndex(0) );273 // NV_LOG( LOG_INFO, "Joystick Number of Axes: " << SDL_JoystickNumAxes(j));274 // NV_LOG( LOG_INFO, "Joystick Number of Buttons: " << SDL_JoystickNumButtons(j));275 // NV_LOG( LOG_INFO, "Joystick Number of Balls: " << SDL_JoystickNumBalls(j));276 // }277 278 279 m_context = new sdl_gl_context( m_device, m_handle );280 m_context->set_viewport( nv::ivec4( 0, 0, m_width, m_height ) );281 }282 283 uint16 gl_window::get_width() const284 {285 return m_width;286 }287 288 uint16 gl_window::get_height() const289 {290 return m_height;291 }292 293 string gl_window::get_title() const294 {295 return m_title;296 }297 298 void gl_window::set_title( const string& title )299 {300 if ( m_adopted ) return;301 #if NV_SDL_VERSION == NV_SDL_20302 SDL_SetWindowTitle( static_cast<SDL_Window*>( m_handle ), title.c_str() );303 #else304 SDL_WM_SetCaption( title.c_str(), title.c_str() );305 #endif306 m_title = title;307 }308 309 bool gl_window::is_event_pending()310 {311 return SDL_PollEvent( nullptr ) != 0;312 }313 314 bool gl_window::poll_event( io_event& event )315 {316 SDL_Event sdl_event;317 if ( SDL_PollEvent( &sdl_event ) == 0 ) return false;318 return sdl_event_to_io_event( sdl_event, event );319 }320 321 12 void gl_window::swap_buffers() 322 13 { 323 if ( m_adopted )324 {325 14 #if NV_PLATFORM == NV_WINDOWS 326 15 ::SwapBuffers( (HDC)m_hwnd ); 327 16 #else 328 NV_ASSERT( false, "Native GL context only working with SDL 2.0!" ); 329 #endif 330 331 } 332 #if NV_SDL_VERSION == NV_SDL_20 333 SDL_GL_SwapWindow( static_cast<SDL_Window*>( m_handle ) ); 334 #else 335 SDL_GL_SwapBuffers(); 17 NV_ASSERT( false, "Native GL context currently only working on Windows!" ); 336 18 #endif 337 19 } … … 339 21 gl_window::~gl_window() 340 22 { 23 if ( m_context ) 24 { 25 #if NV_PLATFORM == NV_WINDOWS 26 dynwglDeleteContext( (HGLRC)m_context->get_native_handle() ); 27 #endif 28 } 341 29 delete m_context; 342 #if NV_SDL_VERSION == NV_SDL_20 343 if ( !m_adopted ) SDL_DestroyWindow( static_cast<SDL_Window*>( m_handle ) ); 344 #endif 30 m_context = nullptr; 31 delete m_input; 345 32 } 346 33 347 nv::gl_window::gl_window( device* dev, void* handle, void* dc )348 : m_device( dev ), m_width( 0 ), m_height( 0 ), m_ title(), m_handle( nullptr ), m_adopted( true)34 nv::gl_window::gl_window( device* dev, window_manager* wm, input* a_input, void* handle, void* dc ) 35 : m_device( dev ), m_width( 0 ), m_height( 0 ), m_handle( nullptr ) 349 36 { 350 37 #if NV_PLATFORM == NV_WINDOWS 351 #if NV_SDL_VERSION == NV_SDL_20 352 m_context = new native_gl_context( m_device, dc ); 353 SDL_Window* window = SDL_CreateWindowFrom( handle ); 38 m_input = a_input; 39 40 m_handle = (void*)handle; 41 42 // TODO: error checking 43 HDC hdc = (HDC)dc; 44 45 const int wgl_attrib_list[] = 46 { 47 WGL_DRAW_TO_WINDOW_ARB, GL_TRUE, 48 WGL_SUPPORT_OPENGL_ARB, GL_TRUE, 49 WGL_ACCELERATION_ARB, WGL_FULL_ACCELERATION_ARB, 50 WGL_DOUBLE_BUFFER_ARB, GL_TRUE, 51 WGL_PIXEL_TYPE_ARB, WGL_TYPE_RGBA_ARB, 52 WGL_COLOR_BITS_ARB, 32, 53 WGL_DEPTH_BITS_ARB, 24, 54 WGL_STENCIL_BITS_ARB, 8, 55 0, 0 //End 56 }; 57 58 unsigned int num_formats; 59 int pixel_format; 60 PIXELFORMATDESCRIPTOR pfd; 61 62 if ( FALSE == wglChoosePixelFormatARB(hdc, wgl_attrib_list, NULL, 1, &pixel_format, &num_formats) ) 63 { 64 return; 65 } 66 67 if ( FALSE == SetPixelFormat(hdc, pixel_format, &pfd) ) 68 { 69 //int err = GetLastError(); 70 return; 71 } 72 73 int attribs[] = 74 { 75 WGL_CONTEXT_MAJOR_VERSION_ARB, 2, 76 WGL_CONTEXT_MINOR_VERSION_ARB, 1, 77 WGL_CONTEXT_PROFILE_MASK_ARB, WGL_CONTEXT_CORE_PROFILE_BIT_ARB, 78 //WGL_CONTEXT_PROFILE_MASK_ARB, WGL_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, 79 0, 0 //End 80 }; 81 82 HGLRC ctx_handle; 83 if ( 0 == (ctx_handle = wglCreateContextAttribsARB(hdc, 0, attribs) ) ) 84 { 85 return; 86 } 87 88 if ( FALSE == dynwglMakeCurrent( hdc, ctx_handle ) ) 89 { 90 return; 91 } 92 93 load_gl_library( NV_GL_PATH, true ); 94 load_wgl_library( NV_GL_PATH, true ); 95 96 m_context = new gl_context( m_device, ctx_handle ); 354 97 // Doesn't work :/ 355 98 // RECT rect; … … 357 100 // m_width = (uint16)rect.right; 358 101 // m_height = (uint16)rect.bottom; 359 m_handle = w indow;102 m_handle = wm->adopt_window( handle ); 360 103 m_hwnd = ::GetDC( (HWND)handle ); 361 104 m_context->set_viewport( nv::ivec4( 0, 0, m_width, m_height ) ); 362 #else363 NV_ASSERT( false, "Native GL context only working with SDL 2.0!" );364 #endif365 105 #else 366 106 NV_ASSERT( false, "Native GL context adoption not implemented for this platform!" ); 367 107 #endif 368 108 } 109
Note: See TracChangeset
for help on using the changeset viewer.