{-# LANGUAGE BangPatterns #-}
{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeFamilies #-}
module Graphics.ColorSpace.YCbCr (
YCbCr(..), YCbCrA(..), Pixel(..)
) where
import Data.Foldable
import Data.Typeable (Typeable)
import Foreign.Ptr
import Foreign.Storable
import Graphics.ColorSpace.Internal
import Prelude hiding (map)
data YCbCr = LumaYCbCr
| CBlueYCbCr
| CRedYCbCr
deriving (Eq, Enum, Show, Bounded, Typeable)
data instance Pixel YCbCr e = PixelYCbCr !e !e !e deriving (Eq, Ord)
instance Show e => Show (Pixel YCbCr e) where
show (PixelYCbCr y b r) = "<YCbCr:("++show y++"|"++show b++"|"++show r++")>"
instance Elevator e => ColorSpace YCbCr e where
type Components YCbCr e = (e, e, e)
fromComponents !(y, b, r) = PixelYCbCr y b r
{-# INLINE fromComponents #-}
toComponents (PixelYCbCr y b r) = (y, b, r)
{-# INLINE toComponents #-}
getPxC (PixelYCbCr y _ _) LumaYCbCr = y
getPxC (PixelYCbCr _ b _) CBlueYCbCr = b
getPxC (PixelYCbCr _ _ r) CRedYCbCr = r
{-# INLINE getPxC #-}
setPxC (PixelYCbCr _ b r) LumaYCbCr y = PixelYCbCr y b r
setPxC (PixelYCbCr y _ r) CBlueYCbCr b = PixelYCbCr y b r
setPxC (PixelYCbCr y b _) CRedYCbCr r = PixelYCbCr y b r
{-# INLINE setPxC #-}
mapPxC f (PixelYCbCr y b r) = PixelYCbCr (f LumaYCbCr y) (f CBlueYCbCr b) (f CRedYCbCr r)
{-# INLINE mapPxC #-}
foldlPx2 f !z (PixelYCbCr y1 b1 r1) (PixelYCbCr y2 b2 r2) =
f (f (f z y1 y2) b1 b2) r1 r2
{-# INLINE foldlPx2 #-}
instance Functor (Pixel YCbCr) where
fmap f (PixelYCbCr y b r) = PixelYCbCr (f y) (f b) (f r)
{-# INLINE fmap #-}
instance Applicative (Pixel YCbCr) where
pure !e = PixelYCbCr e e e
{-# INLINE pure #-}
(PixelYCbCr fy fb fr) <*> (PixelYCbCr y b r) = PixelYCbCr (fy y) (fb b) (fr r)
{-# INLINE (<*>) #-}
instance Foldable (Pixel YCbCr) where
foldr f !z (PixelYCbCr y b r) = f y (f b (f r z))
{-# INLINE foldr #-}
instance Storable e => Storable (Pixel YCbCr e) where
sizeOf _ = 3 * sizeOf (undefined :: e)
{-# INLINE sizeOf #-}
alignment _ = alignment (undefined :: e)
{-# INLINE alignment #-}
peek !p = do
q <- return $ castPtr p
y <- peek q
b <- peekElemOff q 1
r <- peekElemOff q 2
return (PixelYCbCr y b r)
{-# INLINE poke #-}
poke !p (PixelYCbCr y b r) = do
q <- return $ castPtr p
pokeElemOff q 0 y
pokeElemOff q 1 b
pokeElemOff q 2 r
{-# INLINE peek #-}
data YCbCrA = LumaYCbCrA
| CBlueYCbCrA
| CRedYCbCrA
| AlphaYCbCrA
deriving (Eq, Enum, Show, Bounded, Typeable)
data instance Pixel YCbCrA e = PixelYCbCrA !e !e !e !e deriving (Eq, Ord)
instance Show e => Show (Pixel YCbCrA e) where
show (PixelYCbCrA y b r a) = "<YCbCrA:("++show y++"|"++show b++"|"++show r++"|"++show a++")>"
instance Elevator e => ColorSpace YCbCrA e where
type Components YCbCrA e = (e, e, e, e)
fromComponents !(y, b, r, a) = PixelYCbCrA y b r a
{-# INLINE fromComponents #-}
toComponents (PixelYCbCrA y b r a) = (y, b, r, a)
{-# INLINE toComponents #-}
getPxC (PixelYCbCrA y _ _ _) LumaYCbCrA = y
getPxC (PixelYCbCrA _ b _ _) CBlueYCbCrA = b
getPxC (PixelYCbCrA _ _ r _) CRedYCbCrA = r
getPxC (PixelYCbCrA _ _ _ a) AlphaYCbCrA = a
{-# INLINE getPxC #-}
setPxC (PixelYCbCrA _ b r a) LumaYCbCrA y = PixelYCbCrA y b r a
setPxC (PixelYCbCrA y _ r a) CBlueYCbCrA b = PixelYCbCrA y b r a
setPxC (PixelYCbCrA y b _ a) CRedYCbCrA r = PixelYCbCrA y b r a
setPxC (PixelYCbCrA y b r _) AlphaYCbCrA a = PixelYCbCrA y b r a
{-# INLINE setPxC #-}
mapPxC f (PixelYCbCrA y b r a) =
PixelYCbCrA (f LumaYCbCrA y) (f CBlueYCbCrA b) (f CRedYCbCrA r) (f AlphaYCbCrA a)
{-# INLINE mapPxC #-}
foldlPx2 f !z (PixelYCbCrA y1 b1 r1 a1) (PixelYCbCrA y2 b2 r2 a2) =
f (f (f (f z y1 y2) b1 b2) r1 r2) a1 a2
{-# INLINE foldlPx2 #-}
instance Elevator e => AlphaSpace YCbCrA e where
type Opaque YCbCrA = YCbCr
getAlpha (PixelYCbCrA _ _ _ a) = a
{-# INLINE getAlpha #-}
addAlpha !a (PixelYCbCr y b r) = PixelYCbCrA y b r a
{-# INLINE addAlpha #-}
dropAlpha (PixelYCbCrA y b r _) = PixelYCbCr y b r
{-# INLINE dropAlpha #-}
instance Functor (Pixel YCbCrA) where
fmap f (PixelYCbCrA y b r a) = PixelYCbCrA (f y) (f b) (f r) (f a)
{-# INLINE fmap #-}
instance Applicative (Pixel YCbCrA) where
pure !e = PixelYCbCrA e e e e
{-# INLINE pure #-}
(PixelYCbCrA fy fb fr fa) <*> (PixelYCbCrA y b r a) = PixelYCbCrA (fy y) (fb b) (fr r) (fa a)
{-# INLINE (<*>) #-}
instance Foldable (Pixel YCbCrA) where
foldr f !z (PixelYCbCrA y b r a) = f y (f b (f r (f a z)))
{-# INLINE foldr #-}
instance Storable e => Storable (Pixel YCbCrA e) where
sizeOf _ = 4 * sizeOf (undefined :: e)
{-# INLINE sizeOf #-}
alignment _ = alignment (undefined :: e)
{-# INLINE alignment #-}
peek !p = do
q <- return $ castPtr p
y <- peekElemOff q 0
b <- peekElemOff q 1
r <- peekElemOff q 2
a <- peekElemOff q 3
return (PixelYCbCrA y b r a)
{-# INLINE peek #-}
poke !p (PixelYCbCrA y b r a) = do
q <- return $ castPtr p
poke q y
pokeElemOff q 1 b
pokeElemOff q 2 r
pokeElemOff q 3 a
{-# INLINE poke #-}