module KMonad.Keyboard.IO.Linux.Types
(
LinuxKeyEvent(..)
, linuxKeyEvent
, sync
, toLinuxKeyEvent
, fromLinuxKeyEvent
, module KMonad.Keyboard
, module KMonad.Keyboard.IO
)
where
import KMonad.Prelude
import Data.Time.Clock.System
import Foreign.C.Types (CInt)
import RIO.Partial (toEnum)
import KMonad.Keyboard
import KMonad.Keyboard.IO
import KMonad.Util
fi :: Integral a => a -> CInt
fi :: forall a. Integral a => a -> CInt
fi = a -> CInt
forall a b. (Integral a, Num b) => a -> b
fromIntegral
newtype LinuxKeyEvent = LinuxKeyEvent (CInt, CInt, CInt, CInt, CInt)
deriving Int -> LinuxKeyEvent -> ShowS
[LinuxKeyEvent] -> ShowS
LinuxKeyEvent -> String
(Int -> LinuxKeyEvent -> ShowS)
-> (LinuxKeyEvent -> String)
-> ([LinuxKeyEvent] -> ShowS)
-> Show LinuxKeyEvent
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> LinuxKeyEvent -> ShowS
showsPrec :: Int -> LinuxKeyEvent -> ShowS
$cshow :: LinuxKeyEvent -> String
show :: LinuxKeyEvent -> String
$cshowList :: [LinuxKeyEvent] -> ShowS
showList :: [LinuxKeyEvent] -> ShowS
Show
instance Display LinuxKeyEvent where
textDisplay :: LinuxKeyEvent -> Text
textDisplay (LinuxKeyEvent (CInt
s, CInt
ns, CInt
typ, CInt
c, CInt
val)) = [Text] -> Text
forall a. Monoid a => [a] -> a
mconcat
[ CInt -> Text
forall a. Show a => a -> Text
tshow CInt
s, Text
".", CInt -> Text
forall a. Show a => a -> Text
tshow CInt
ns, Text
": "
, Text
"type: ", CInt -> Text
forall a. Show a => a -> Text
tshow CInt
typ, Text
", "
, Text
"code: ", CInt -> Text
forall a. Show a => a -> Text
tshow CInt
c, Text
", "
, Text
"val: ", CInt -> Text
forall a. Show a => a -> Text
tshow CInt
val
]
linuxKeyEvent
:: (Integral a, Integral b, Integral c, Integral d, Integral e)
=> (a, b, c, d, e)
-> LinuxKeyEvent
linuxKeyEvent :: forall a b c d e.
(Integral a, Integral b, Integral c, Integral d, Integral e) =>
(a, b, c, d, e) -> LinuxKeyEvent
linuxKeyEvent (a
a, b
b, c
c, d
d, e
e) = (CInt, CInt, CInt, CInt, CInt) -> LinuxKeyEvent
LinuxKeyEvent (a -> CInt
forall a. Integral a => a -> CInt
f a
a, b -> CInt
forall a. Integral a => a -> CInt
f b
b, c -> CInt
forall a. Integral a => a -> CInt
f c
c, d -> CInt
forall a. Integral a => a -> CInt
f d
d, e -> CInt
forall a. Integral a => a -> CInt
f e
e)
where
f :: Integral a => a -> CInt
f :: forall a. Integral a => a -> CInt
f = a -> CInt
forall a b. (Integral a, Num b) => a -> b
fromIntegral
sync :: SystemTime -> LinuxKeyEvent
sync :: SystemTime -> LinuxKeyEvent
sync (MkSystemTime Int64
s Word32
ns) = (CInt, CInt, CInt, CInt, CInt) -> LinuxKeyEvent
LinuxKeyEvent (Int64 -> CInt
forall a. Integral a => a -> CInt
fi Int64
s, Word32 -> CInt
forall a. Integral a => a -> CInt
fi Word32
ns, CInt
0, CInt
0, CInt
0)
fromLinuxKeyEvent :: LinuxKeyEvent -> Maybe KeyEvent
fromLinuxKeyEvent :: LinuxKeyEvent -> Maybe KeyEvent
fromLinuxKeyEvent (LinuxKeyEvent (CInt
_, CInt
_, CInt
typ, CInt
c, CInt
val))
| CInt
c CInt -> CInt -> Bool
forall a. Ord a => a -> a -> Bool
>= CInt
0x2ff = Maybe KeyEvent
forall a. Maybe a
Nothing
| CInt
typ CInt -> CInt -> Bool
forall a. Eq a => a -> a -> Bool
== CInt
1 Bool -> Bool -> Bool
&& CInt
val CInt -> CInt -> Bool
forall a. Eq a => a -> a -> Bool
== CInt
0 = KeyEvent -> Maybe KeyEvent
forall a. a -> Maybe a
Just (KeyEvent -> Maybe KeyEvent)
-> (Keycode -> KeyEvent) -> Keycode -> Maybe KeyEvent
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Keycode -> KeyEvent
mkRelease (Keycode -> Maybe KeyEvent) -> Keycode -> Maybe KeyEvent
forall a b. (a -> b) -> a -> b
$ Keycode
kc
| CInt
typ CInt -> CInt -> Bool
forall a. Eq a => a -> a -> Bool
== CInt
1 Bool -> Bool -> Bool
&& CInt
val CInt -> CInt -> Bool
forall a. Eq a => a -> a -> Bool
== CInt
1 = KeyEvent -> Maybe KeyEvent
forall a. a -> Maybe a
Just (KeyEvent -> Maybe KeyEvent)
-> (Keycode -> KeyEvent) -> Keycode -> Maybe KeyEvent
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Keycode -> KeyEvent
mkPress (Keycode -> Maybe KeyEvent) -> Keycode -> Maybe KeyEvent
forall a b. (a -> b) -> a -> b
$ Keycode
kc
| Bool
otherwise = Maybe KeyEvent
forall a. Maybe a
Nothing
where
kc :: Keycode
kc = Int -> Keycode
forall a. Enum a => Int -> a
toEnum (Int -> Keycode) -> (CInt -> Int) -> CInt -> Keycode
forall b c a. (b -> c) -> (a -> b) -> a -> c
. CInt -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (CInt -> Keycode) -> CInt -> Keycode
forall a b. (a -> b) -> a -> b
$ CInt
c
toLinuxKeyEvent :: KeyEvent -> SystemTime -> LinuxKeyEvent
toLinuxKeyEvent :: KeyEvent -> SystemTime -> LinuxKeyEvent
toLinuxKeyEvent KeyEvent
e (MkSystemTime Int64
s Word32
ns)
= (CInt, CInt, CInt, CInt, CInt) -> LinuxKeyEvent
LinuxKeyEvent (Int64 -> CInt
forall a. Integral a => a -> CInt
fi Int64
s, Word32 -> CInt
forall a. Integral a => a -> CInt
fi Word32
ns, CInt
1, CInt
c, CInt
val)
where
c :: CInt
c = Int -> CInt
forall a. Integral a => a -> CInt
fi (Int -> CInt) -> (Keycode -> Int) -> Keycode -> CInt
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Keycode -> Int
forall a. Enum a => a -> Int
fromEnum (Keycode -> CInt) -> Keycode -> CInt
forall a b. (a -> b) -> a -> b
$ KeyEvent
eKeyEvent -> Getting Keycode KeyEvent Keycode -> Keycode
forall s a. s -> Getting a s a -> a
^.Getting Keycode KeyEvent Keycode
forall c. HasKeyEvent c => Lens' c Keycode
Lens' KeyEvent Keycode
keycode
val :: CInt
val = if KeyEvent
eKeyEvent -> Getting Switch KeyEvent Switch -> Switch
forall s a. s -> Getting a s a -> a
^.Getting Switch KeyEvent Switch
forall c. HasKeyEvent c => Lens' c Switch
Lens' KeyEvent Switch
switch Switch -> Switch -> Bool
forall a. Eq a => a -> a -> Bool
== Switch
Press then CInt
1 else CInt
0