{-# LANGUAGE DataKinds, ScopedTypeVariables, TypeOperators, GADTs, BangPatterns, FlexibleInstances, FlexibleContexts, KindSignatures, RankNTypes, ConstraintKinds #-} -- | Utilities for working with vertex buffer objects (VBOs) filled -- with vertices represented as vinyl records. module Graphics.VinylGL.Vertex (bufferVertices, bindVertices, reloadVertices, deleteVertices, enableVertices, enableVertices', BufferedVertices(..), fieldToVAD) where import Control.Applicative import Control.Arrow (second) import Data.Foldable (Foldable, foldMap) import Data.List (find) import qualified Data.Map as M import Data.Monoid (Sum(..)) import Data.Proxy (Proxy(..)) import qualified Data.Vector.Storable as V import Data.Vinyl ((:::)(..), PlainRec, Implicit(..), Elem) import Foreign.Ptr (plusPtr) import Foreign.Storable import GHC.TypeLits (Sing, SingI(..), fromSing) import Graphics.GLUtil hiding (Elem) import Graphics.Rendering.OpenGL (VertexArrayDescriptor(..), bindBuffer, ($=), BufferTarget(..)) import qualified Graphics.Rendering.OpenGL as GL import Data.Vinyl.Reflect import Graphics.VinylGL.Uniforms -- | Representation of a VBO whose type describes the vertices. newtype BufferedVertices (fields::[*]) = BufferedVertices { getVertexBuffer :: GL.BufferObject } -- | Load vertex data into a GPU-accessible buffer. bufferVertices :: (Storable (PlainRec rs), BufferSource (v (PlainRec rs))) => v (PlainRec rs) -> IO (BufferedVertices rs) bufferVertices = fmap BufferedVertices . fromSource ArrayBuffer -- | Reload 'BufferedVertices' with a 'V.Vector' of new vertex data. reloadVertices :: Storable (PlainRec rs) => BufferedVertices rs -> V.Vector (PlainRec rs) -> IO () reloadVertices b v = do bindBuffer ArrayBuffer $= Just (getVertexBuffer b) replaceVector ArrayBuffer v -- | Delete the object name associated with 'BufferedVertices'. deleteVertices :: BufferedVertices a -> IO () deleteVertices = GL.deleteObjectNames . (:[]) . getVertexBuffer -- | Bind previously-buffered vertex data. bindVertices :: BufferedVertices a -> IO () bindVertices = (bindBuffer ArrayBuffer $=) . Just . getVertexBuffer -- | Line up a shader's attribute inputs with a vertex record. This -- maps vertex fields to GLSL attributes on the basis of record field -- names on the Haskell side, and variable names on the GLSL side. enableVertices :: forall f rs. ViableVertex (PlainRec rs) => ShaderProgram -> f rs -> IO (Maybe String) enableVertices s _ = enableAttribs s (Proxy::Proxy (PlainRec rs)) -- | Behaves like 'enableVertices', but raises an exception if the -- supplied vertex record does not include a field required by the -- shader. enableVertices' :: forall f rs. ViableVertex (PlainRec rs) => ShaderProgram -> f rs -> IO () enableVertices' s _ = enableAttribs s (Proxy::Proxy (PlainRec rs)) >>= maybe (return ()) error -- | Produce a 'GL.VertexArrayDescriptor' for a particular field of a -- vertex record. fieldToVAD :: forall sy v a r rs. (r ~ (sy ::: v a), HasFieldNames (PlainRec rs), HasFieldSizes (PlainRec rs), HasGLType a, Storable (PlainRec rs), Num (v a), SingI sy, Foldable v, Implicit (Elem r rs)) => r -> Proxy (PlainRec rs) -> GL.VertexArrayDescriptor a fieldToVAD _ _ = GL.VertexArrayDescriptor dim (glType (undefined::a)) (fromIntegral $ sizeOf (undefined::PlainRec rs)) (offset0 `plusPtr` offset) where dim = getSum $ foldMap (const (Sum 1)) (0::v a) Just offset = lookup n $ namesAndOffsets (undefined::PlainRec rs) n = fromSing (sing::Sing sy) -- Constraint alias capturing the requirements of a vertex type. type ViableVertex t = (HasFieldNames t, HasFieldSizes t, HasFieldDims t, HasFieldGLTypes t, Storable t) enableAttribs :: forall v. ViableVertex v => ShaderProgram -> Proxy v -> IO (Maybe String) enableAttribs s p = go (map (second snd) $ M.assocs (attribs s)) where go [] = return Nothing go ((n,t):ns) = case find ((== n) . fieldName) fs of Nothing -> return (Just $ "GLSL expecting "++n) Just fd | fieldType fd == t -> do enableAttrib s n setAttrib s n GL.ToFloat $ descriptorVAD p fd go ns | otherwise -> return . Just $ "Type mismatch in "++n fs = fieldDescriptors (undefined::v) namesAndOffsets :: (HasFieldNames t, HasFieldSizes t) => t -> [(String,Int)] namesAndOffsets x = zip (fieldNames x) (scanl (+) 0 $ fieldSizes x) data FieldDescriptor = FieldDescriptor { fieldName :: String , fieldOffset :: Int , fieldDim :: Int , fieldType :: GL.VariableType } deriving Show fieldDescriptors :: ViableVertex t => t -> [FieldDescriptor] fieldDescriptors x = getZipList $ FieldDescriptor <$> zl (fieldNames x) <*> zl (scanl (+) 0 $ fieldSizes x) <*> zl (fieldDims x) <*> zl (fieldGLTypes x) where zl = ZipList descriptorVAD :: forall t a. Storable t => Proxy t -> FieldDescriptor -> VertexArrayDescriptor a descriptorVAD _ fd = VertexArrayDescriptor (fromIntegral $ fieldDim fd) (variableDataType $ fieldType fd) (fromIntegral $ sizeOf (undefined::t)) (offset0 `plusPtr` fieldOffset fd)