{-# LANGUAGE AllowAmbiguousTypes #-} {-# LANGUAGE DataKinds #-} {-# LANGUAGE FlexibleInstances #-} {-# LANGUAGE FunctionalDependencies #-} {-# LANGUAGE KindSignatures #-} {-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE ScopedTypeVariables #-} {-# LANGUAGE TypeApplications #-} {-# LANGUAGE TypeFamilies #-} {-# LANGUAGE TypeOperators #-} {-# LANGUAGE UndecidableInstances #-} ----------------------------------------------------------------------------- -- | -- Module : Data.Generics.Product.Fields -- Copyright : (C) 2017 Csongor Kiss -- License : BSD3 -- Maintainer : Csongor Kiss <kiss.csongor.kiss@gmail.com> -- Stability : experimental -- Portability : non-portable -- -- Derive record field getters and setters generically. -- ----------------------------------------------------------------------------- module Data.Generics.Product.Fields ( -- *Lenses -- $example HasField (..) ) where import Data.Generics.Internal.Families import Data.Generics.Internal.Lens import Data.Generics.Product.Internal.Fields import Data.Kind (Constraint, Type) import GHC.Generics import GHC.TypeLits (Symbol, ErrorMessage(..), TypeError) -- $example -- @ -- module Example where -- -- import Data.Generics.Product -- import GHC.Generics -- -- data Human = Human -- { name :: String -- , age :: Int -- , address :: String -- } -- deriving (Generic, Show) -- -- human :: Human -- human = Human \"Tunyasz\" 50 \"London\" -- @ -- |Records that have a field with a given name. class HasField (field :: Symbol) a s | s field -> a where -- |A lens that focuses on a field with a given name. Compatible with the -- lens package's 'Control.Lens.Lens' type. -- -- >>> human ^. field @"age" -- 50 -- >>> human & field @"name" .~ "Tamas" -- Human {name = "Tamas", age = 50, address = "London"} field :: Lens' s a field f s = fmap (flip (setField @field) s) (f (getField @field s)) -- |Get 'field' -- -- >>> getField @"name" human -- "Tunyasz" getField :: s -> a getField s = s ^. field @field -- |Set 'field' -- -- >>> setField @"age" (setField @"name" "Tamas" human) 30 -- Human {name = "Tamas", age = 30, address = "London"} setField :: a -> s -> s setField = set (field @field) {-# MINIMAL field | setField, getField #-} instance ( Generic s , ErrorUnless field s (HasTotalFieldP field (Rep s)) , GHasField field (Rep s) a ) => HasField field a s where field = ravel (repLens . gfield @field) type family ErrorUnless (field :: Symbol) (s :: Type) (contains :: Bool) :: Constraint where ErrorUnless field s 'False = TypeError ( 'Text "The type " ':<>: 'ShowType s ':<>: 'Text " does not contain a field named " ':<>: 'ShowType field ) ErrorUnless _ _ 'True = ()