{-# OPTIONS_HADDOCK hide #-} -- This makes a kind of tri. Has space efficiency issues with large -- input blocks. Likely building a parser and just applying that would -- be better. module Graphics.Vty.Input.Classify ( classify , KClass(..) ) where import Graphics.Vty.Input.Events import Graphics.Vty.Input.Mouse import Graphics.Vty.Input.Focus import Graphics.Vty.Input.Paste import Graphics.Vty.Input.Classify.Types import Codec.Binary.UTF8.Generic (decode) import Data.List (inits) import qualified Data.Map as M( fromList, lookup ) import Data.Maybe ( mapMaybe ) import qualified Data.Set as S( fromList, member ) import Data.Char import Data.Word compile :: ClassifyMap -> [Char] -> KClass compile table = cl' where -- take all prefixes and create a set of these prefixSet = S.fromList $ concatMap (init . inits . fst) $ table maxValidInputLength = maximum (map (length . fst) table) eventForInput = M.fromList table cl' [] = Prefix cl' inputBlock = case M.lookup inputBlock eventForInput of -- if the inputBlock is exactly what is expected for an -- event then consume the whole block and return the event Just e -> Valid e [] Nothing -> case S.member inputBlock prefixSet of True -> Prefix -- look up progressively smaller tails of the input -- block until an event is found The assumption is that -- the event that consumes the most input bytes should -- be produced. -- The test verifyFullSynInputToEvent2x verifies this. -- H: There will always be one match. The prefixSet -- contains, by definition, all prefixes of an event. False -> let inputPrefixes = reverse $ take maxValidInputLength $ tail $ inits inputBlock in case mapMaybe (\s -> (,) s `fmap` M.lookup s eventForInput) inputPrefixes of (s,e) : _ -> Valid e (drop (length s) inputBlock) -- neither a prefix or a full event. [] -> Invalid classify :: ClassifyMap -> [Char] -> KClass classify table = let standardClassifier = compile table in \s -> case s of _ | bracketedPasteStarted s -> if bracketedPasteFinished s then parseBracketedPaste s else Prefix _ | isMouseEvent s -> classifyMouseEvent s _ | isFocusEvent s -> classifyFocusEvent s c:cs | ord c >= 0xC2 -> classifyUtf8 c cs _ -> standardClassifier s classifyUtf8 :: Char -> [Char] -> KClass classifyUtf8 c cs = let n = utf8Length (ord c) (codepoint,rest) = splitAt n (c:cs) codepoint8 :: [Word8] codepoint8 = map (fromIntegral . ord) codepoint in case decode codepoint8 of _ | n < length codepoint -> Prefix Just (unicodeChar, _) -> Valid (EvKey (KChar unicodeChar) []) rest -- something bad happened; just ignore and continue. Nothing -> Invalid utf8Length :: (Num t, Ord a, Num a) => a -> t utf8Length c | c < 0x80 = 1 | c < 0xE0 = 2 | c < 0xF0 = 3 | otherwise = 4