This repository has been archived on 2025-03-03. You can view files and clone it, but cannot push or open issues or pull requests.
htanks/GLX.hs

197 lines
8.1 KiB
Haskell
Raw Normal View History

2010-02-22 16:50:42 +01:00
module GLX ( glxDriver
) where
import GLDriver
import Bindings.GLX
import Control.Monad (when, unless)
import Data.Bits ((.|.))
2010-02-22 22:25:06 +01:00
import Data.Maybe (isJust)
2010-03-05 04:38:31 +01:00
import Data.Ratio
2010-02-22 16:50:42 +01:00
2010-03-08 19:31:48 +01:00
import Graphics.Rendering.OpenGL.GL (($=), GLdouble, GLfloat, Vector3(..), Capability(..))
import Graphics.Rendering.OpenGL.GL.CoordTrans (matrixMode, MatrixMode(..), viewport, Position(..), Size(..), loadIdentity, ortho, translate)
2010-02-22 16:50:42 +01:00
import Graphics.X11.Types
2010-02-22 22:25:06 +01:00
import Graphics.X11.Xlib.Atom (internAtom)
2010-02-22 16:50:42 +01:00
import Graphics.X11.Xlib.Display (defaultScreen, openDisplay, rootWindow)
2010-02-22 22:25:06 +01:00
import Graphics.X11.Xlib.Event (XEventPtr, allocaXEvent, nextEvent, get_Window, get_EventType, pending)
2010-03-05 04:38:31 +01:00
import Graphics.X11.Xlib.Extras (ClassHint(..), getEvent, ev_event_type, ev_width, ev_height, ev_keycode, ev_data, ev_x, ev_y, getWindowAttributes, wa_width, wa_height)
2010-02-22 22:25:06 +01:00
import Graphics.X11.Xlib.Misc (keycodeToKeysym, setWMProtocols)
2010-02-22 16:50:42 +01:00
import Graphics.X11.Xlib.Types
import Graphics.X11.Xlib.Window (destroyWindow, mapWindow, storeName)
2010-02-22 16:50:42 +01:00
import Foreign.Marshal.Utils (with)
import Foreign.Ptr
import Foreign.Storable
2010-02-22 22:25:06 +01:00
data GLX = GLX
2010-03-08 19:31:48 +01:00
{ glxDisplay :: !Display
, glxWindow :: !Window
, glxContext :: !Context
2010-02-22 22:25:06 +01:00
, glxDeleteWindow :: !Atom
2010-03-08 19:31:48 +01:00
, glxScale :: !Rational
, glxLevelWidth :: !Int
, glxLevelHeight :: !Int
2010-02-22 22:25:06 +01:00
}
2010-02-22 16:50:42 +01:00
2010-03-08 19:31:48 +01:00
glxDriver :: Int -> Int -> GLX
glxDriver w h = GLX
2010-02-22 22:25:06 +01:00
{ glxDisplay = Display nullPtr
, glxWindow = 0
, glxContext = Context nullPtr
, glxDeleteWindow = 0
, glxScale = 1
2010-03-08 19:31:48 +01:00
, glxLevelWidth = w
, glxLevelHeight = h
2010-02-22 22:25:06 +01:00
}
2010-02-22 16:50:42 +01:00
2010-02-22 18:27:18 +01:00
instance Driver GLX where
initialized glx = ((glxContext glx) /= (Context nullPtr))
2010-02-22 16:50:42 +01:00
2010-02-22 22:25:06 +01:00
initGL glx = do
when (initialized glx) $ fail "GLX already initialized"
2010-02-22 16:50:42 +01:00
disp <- openDisplay ""
2010-02-22 22:25:06 +01:00
delwnd <- internAtom disp "WM_DELETE_WINDOW" False
2010-02-22 16:50:42 +01:00
fbconfigs <- chooseFBConfig disp (fromIntegral . defaultScreen $ disp)
2010-03-02 03:48:03 +01:00
[ (renderType, rgbaBit)
, (drawableType, windowBit)
, (doublebuffer, 1)
, (xRenderable, 1)
, (depthSize, 1)
, (stencilSize, 1)
]
2010-02-22 16:50:42 +01:00
visualinfo <- getVisualFromFBConfig disp (head fbconfigs)
2010-02-22 18:27:18 +01:00
rootwindow <- rootWindow disp (fromIntegral $ viScreen visualinfo)
cmap <- createColormap disp rootwindow (viVisual visualinfo) allocNone
2010-02-22 16:50:42 +01:00
let swa = nullSetWindowAttributes {swaColormap = cmap, swaEventMask = structureNotifyMask .|. keyPressMask .|. keyReleaseMask .|. pointerMotionMask}
2010-02-22 16:50:42 +01:00
2010-02-22 22:25:06 +01:00
wnd <- with swa $ \swaptr ->
createWindow disp rootwindow 0 0 800 600 0 (fromIntegral $ viDepth visualinfo) inputOutput (viVisual visualinfo) (cWBorderPixel.|.cWColormap.|.cWEventMask) swaptr
setClassHint disp wnd (ClassHint "HTanks" "htanks")
2010-02-22 22:25:06 +01:00
setWMProtocols disp wnd [delwnd]
storeName disp wnd "HTanks"
2010-02-22 16:50:42 +01:00
mapWindow disp wnd
waitForMapNotify disp wnd
2010-02-22 18:27:18 +01:00
ctx <- with visualinfo $ \vi -> createContext disp vi (Context nullPtr) True
makeCurrent disp wnd ctx
2010-02-22 16:50:42 +01:00
wa <- getWindowAttributes disp wnd
2010-03-08 19:31:48 +01:00
s <- resize (glxLevelWidth glx) (glxLevelHeight glx) (fromIntegral . wa_width $ wa) (fromIntegral . wa_height $ wa)
2010-03-08 19:31:48 +01:00
return glx
2010-02-22 22:25:06 +01:00
{ glxDisplay = disp
, glxWindow = wnd
, glxContext = ctx
, glxDeleteWindow = delwnd
2010-03-05 04:38:31 +01:00
, glxScale = s
2010-02-22 22:25:06 +01:00
}
deinitGL glx = do
destroyWindow (glxDisplay glx) (glxWindow glx)
destroyContext (glxDisplay glx) (glxContext glx)
2010-02-22 16:50:42 +01:00
swapBuffers glx = Bindings.GLX.swapBuffers (glxDisplay glx) (glxWindow glx)
nextEvent glx = allocaXEvent $ nextEvent' glx
2010-02-22 22:25:06 +01:00
2010-03-05 04:38:31 +01:00
nextEvent' :: GLX -> XEventPtr -> IO (GLX, Maybe SomeEvent)
nextEvent' glx xevent = do
p <- pending $ glxDisplay glx
2010-02-22 22:25:06 +01:00
if (p > 0) then do
Graphics.X11.Xlib.Event.nextEvent (glxDisplay glx) xevent
2010-03-05 04:38:31 +01:00
(newglx, ev) <- handleEvent glx xevent
2010-02-22 22:25:06 +01:00
if isJust ev then
2010-03-05 04:38:31 +01:00
return (newglx, ev)
2010-02-22 22:25:06 +01:00
else
2010-03-05 04:38:31 +01:00
nextEvent' newglx xevent
2010-02-22 22:25:06 +01:00
else
2010-03-05 04:38:31 +01:00
return (glx, Nothing)
2010-02-22 22:25:06 +01:00
2010-03-05 04:38:31 +01:00
handleEvent :: GLX -> XEventPtr -> IO (GLX, Maybe SomeEvent)
handleEvent glx xevent = do
2010-02-22 22:25:06 +01:00
event <- getEvent xevent
let evtype = ev_event_type event
case () of
2010-02-23 20:51:30 +01:00
_ | evtype == configureNotify -> do
2010-03-08 19:31:48 +01:00
s <- resize (glxLevelWidth glx) (glxLevelHeight glx) (fromIntegral . ev_width $ event) (fromIntegral . ev_height $ event)
2010-03-05 04:38:31 +01:00
return (glx {glxScale = s}, Nothing)
2010-02-23 20:51:30 +01:00
| evtype == keyPress -> do
2010-03-05 04:38:31 +01:00
keysym <- keycodeToKeysym (glxDisplay glx) (ev_keycode event) 0
case () of
_ | keysym == xK_Escape -> return (glx, Just $ SomeEvent QuitEvent)
| keysym == xK_Up -> return (glx, Just $ SomeEvent $ KeyPressEvent KeyUp)
| keysym == xK_Down -> return (glx, Just $ SomeEvent $ KeyPressEvent KeyDown)
| keysym == xK_Left -> return (glx, Just $ SomeEvent $ KeyPressEvent KeyLeft)
| keysym == xK_Right -> return (glx, Just $ SomeEvent $ KeyPressEvent KeyRight)
| otherwise -> return (glx, Nothing)
2010-02-24 03:40:06 +01:00
| evtype == keyRelease -> do
2010-03-05 04:38:31 +01:00
keysym <- keycodeToKeysym (glxDisplay glx) (ev_keycode event) 0
case () of
_ | keysym == xK_Up -> return (glx, Just $ SomeEvent $ KeyReleaseEvent KeyUp)
| keysym == xK_Down -> return (glx, Just $ SomeEvent $ KeyReleaseEvent KeyDown)
| keysym == xK_Left -> return (glx, Just $ SomeEvent $ KeyReleaseEvent KeyLeft)
| keysym == xK_Right -> return (glx, Just $ SomeEvent $ KeyReleaseEvent KeyRight)
| otherwise -> return (glx, Nothing)
2010-02-23 20:51:30 +01:00
| evtype == clientMessage -> do
2010-03-05 04:38:31 +01:00
if ((glxDeleteWindow glx) == (fromIntegral . head . ev_data $ event))
then
return (glx, Just $ SomeEvent QuitEvent)
else
return (glx, Nothing)
| evtype == motionNotify -> do
wa <- getWindowAttributes (glxDisplay glx) (glxWindow glx)
let x = fromIntegral . ev_x $ event
y = fromIntegral . ev_y $ event
w = fromIntegral . wa_width $ wa
2010-03-05 04:38:31 +01:00
h = fromIntegral . wa_height $ wa
s = fromRational . glxScale $ glx
lw = fromIntegral . glxLevelWidth $ glx
lh = fromIntegral . glxLevelHeight $ glx
return (glx, Just $ SomeEvent $ MouseMotionEvent ((w/2+x)/s - lw/2) ((h/2-y)/s + lh/2))
2010-03-05 04:38:31 +01:00
| otherwise -> return (glx, Nothing)
2010-03-08 19:31:48 +01:00
resize :: Int -> Int -> Int -> Int -> IO Rational
resize lw lh w h = do
let aspect = (fromIntegral w)%(fromIntegral h)
s = (max ((fromIntegral lw)/aspect) (fromIntegral lh))/2
2010-03-05 04:38:31 +01:00
sf = fromRational s
aspectf = fromRational aspect
matrixMode $= Projection
loadIdentity
2010-03-05 04:38:31 +01:00
ortho (-sf*aspectf) (sf*aspectf) (-sf) sf (-1) 1
2010-03-08 19:31:48 +01:00
translate $ Vector3 (-(fromIntegral lw)/2) (-(fromIntegral lh)/2) (0 :: GLfloat)
matrixMode $= Modelview 0
viewport $= ((Position 0 0), (Size (fromIntegral w) (fromIntegral h)))
2010-03-05 04:38:31 +01:00
2010-03-08 19:31:48 +01:00
return $ (fromIntegral h)/(2*s)
2010-02-22 16:50:42 +01:00
waitForMapNotify :: Display -> Window -> IO ()
waitForMapNotify disp wnd = allocaXEvent waitForMapNotify'
where
waitForMapNotify' event = do
Graphics.X11.Xlib.Event.nextEvent disp event
window <- get_Window event
eventType <- get_EventType event
unless (window == wnd && eventType == mapNotify) $
waitForMapNotify' event