-- | -- -- /Case/ is a property of letters. /A-Z/ are /upper case/ letters, and -- /a-z/ are /lower case/ letters. No other ASCII characters have case. module ASCII.Case (Case (..), letterCase, isCase, toCase, opposite) where import ASCII.Char (Char (..)) import ASCII.Char qualified as Char import Data.Bool (Bool, otherwise) import Data.Bool qualified as Bool import Data.Data (Data) import Data.Eq (Eq) import Data.Function ((.)) import Data.Hashable (Hashable) import Data.Maybe (Maybe (..)) import Data.Ord (Ord, (<=), (>=)) import GHC.Generics (Generic) import Text.Show (Show) import Prelude (Bounded, Enum, Int, (+), (-)) data Case = -- | The letters from 'CapitalLetterA' to 'CapitalLetterZ'. UpperCase | -- | The letters from 'SmallLetterA' to 'SmallLetterZ'. LowerCase deriving stock instance Eq Case deriving stock instance Ord Case deriving stock instance Enum Case deriving stock instance Bounded Case deriving stock instance Show Case deriving stock instance Data Case deriving stock instance Generic Case deriving anyclass instance Hashable Case -- | Determines whether a character is a letter, and if so, whether it is upper or lower case. -- -- >>> map letterCase [CapitalLetterR, SmallLetterR, DollarSign] -- [Just UpperCase,Just LowerCase,Nothing] letterCase :: Char -> Maybe Case letterCase x | isCase UpperCase x = Just UpperCase | isCase LowerCase x = Just LowerCase | otherwise = Nothing -- | Determines whether a character is a letter of a particular case. -- -- >>> map (isCase UpperCase) [CapitalLetterR,SmallLetterR,DollarSign] -- [True,False,False] isCase :: Case -> Char -> Bool isCase c x = (Bool.&&) (x >= a) (x <= z) where (a, z) = az c az :: Case -> (Char, Char) az = \case UpperCase -> (CapitalLetterA, CapitalLetterZ) LowerCase -> (SmallLetterA, SmallLetterZ) -- | Maps a letter character to its upper/lower case equivalent. -- -- >>> toCase UpperCase SmallLetterX -- CapitalLetterX -- -- >>> toCase LowerCase CapitalLetterF -- SmallLetterF -- -- Characters that are already in the requested case are unmodified by this transformation. -- -- >>> toCase UpperCase CapitalLetterA -- CapitalLetterA -- -- Characters that are not letters, such as exclamation mark, are unmodified by this transformation. -- -- >>> toCase UpperCase ExclamationMark -- ExclamationMark toCase :: Case -> Char -> Char toCase c x = if isCase (opposite c) x then changeCaseUnsafe c x else x -- | Change a letter to the given case, assuming that the input character is a letter of the opposite case. changeCaseUnsafe :: Case -> Char -> Char changeCaseUnsafe c = charAsIntUnsafe (changeCaseInt c) changeCaseInt :: Case -> Int -> Int changeCaseInt c i = case c of LowerCase -> i + 32 UpperCase -> i - 32 opposite :: Case -> Case opposite = \case UpperCase -> LowerCase LowerCase -> UpperCase charAsIntUnsafe :: (Int -> Int) -> (Char -> Char) charAsIntUnsafe f = Char.fromIntUnsafe . f . Char.toInt