Safe Haskell | None |
---|---|
Language | Haskell2010 |
This module provides core definitions:
-
an opaque
Optic
type, which is parameterised over a type representing an optic kind (instantiated with tag types such asA_Lens
); -
the optic composition operator (
%
); -
the subtyping relation
Is
with an accompanyingcastOptic
function to convert an optic kind; -
the
JoinKinds
class used to find the optic kind resulting from a composition.
Each optic kind is identified by a "tag type" (such as
A_Lens
), which is an
empty data type. The type of the actual optics (such as
Lens
)
is obtained by applying
Optic
to the tag type.
See the
Optics
module in the main
optics
package for overview
documentation.
Synopsis
- type OpticKind = Type
- data Optic (k :: OpticKind ) (is :: IxList ) s t a b
- type Optic' k is s a = Optic k is s s a a
- castOptic :: forall destKind srcKind is s t a b. Is srcKind destKind => Optic srcKind is s t a b -> Optic destKind is s t a b
- class Is k l
- class JoinKinds k l m | k l -> m
- (%) :: forall k l m is js ks s t u v a b. ( JoinKinds k l m, AppendIndices is js ks) => Optic k is s t u v -> Optic l js u v a b -> Optic m ks s t a b
- (%%) :: forall k is js ks s t u v a b. AppendIndices is js ks => Optic k is s t u v -> Optic k js u v a b -> Optic k ks s t a b
- (%&) :: Optic k is s t a b -> ( Optic k is s t a b -> Optic l js s' t' a' b') -> Optic l js s' t' a' b'
- type IxList = [ Type ]
- type NoIx = '[] :: IxList
- type WithIx i = '[i] :: IxList
- class AppendIndices xs ys ks | xs ys -> ks
- class NonEmptyIndices (is :: IxList )
- class is ~ '[i] => HasSingleIndex (is :: IxList ) (i :: Type )
- class is ~ NoIx => AcceptsEmptyIndices (f :: Symbol ) (is :: IxList )
- type family Curry (xs :: IxList ) (y :: Type ) :: Type where ...
- class CurryCompose xs where
- (&) :: a -> (a -> b) -> b
- (<&>) :: Functor f => f a -> (a -> b) -> f b
Documentation
data Optic (k :: OpticKind ) (is :: IxList ) s t a b Source #
Wrapper newtype for the whole family of optics.
The first parameter
k
identifies the particular optic kind (e.g.
A_Lens
or
A_Traversal
).
The parameter
is
is a list of types available as indices. This will
typically be
NoIx
for unindexed optics, or
WithIx
for optics with a
single index. See the "Indexed optics" section of the overview documentation
in the
Optics
module of the main
optics
package for more details.
The parameters
s
and
t
represent the "big" structure,
whereas
a
and
b
represent the "small" structure.
type Optic' k is s a = Optic k is s s a a Source #
Common special case of
Optic
where source and target types are equal.
Here, we need only one "big" and one "small" type. For lenses, this means that in the restricted form we cannot do type-changing updates.
Subtyping
castOptic :: forall destKind srcKind is s t a b. Is srcKind destKind => Optic srcKind is s t a b -> Optic destKind is s t a b Source #
Subtyping relationship between kinds of optics.
An instance of
means that any
Is
k l
can be used
as an
Optic
k
. For example, we have an
Optic
l
instance, but not
Is
A_Lens
A_Traversal
.
Is
A_Traversal
A_Lens
This class needs instances for all possible combinations of tags.
Instances
class JoinKinds k l m | k l -> m Source #
Computes the least upper bound of two optics kinds.
In presence of a
JoinKinds k l m
constraint
Optic m
represents the least
upper bound of an
Optic k
and an
Optic l
. This means in particular that
composition of an
Optic k
and an
Optic k
will yield an
Optic m
.
Since: 0.4
Instances
Composition
The usual operator for composing optics is (
%
), which allows different
optic kinds to be composed, automatically calculating the resulting optic
kind using
JoinKinds
.
The (
.
) function composition operator cannot be used to compose optics,
because
optics are not functions
. The (
.
) operator
from
Control.Category
cannot be used either, because it would not support
type-changing optics or composing optics of different kinds.
(%) :: forall k l m is js ks s t u v a b. ( JoinKinds k l m, AppendIndices is js ks) => Optic k is s t u v -> Optic l js u v a b -> Optic m ks s t a b infixl 9 Source #
Compose two optics of compatible flavours.
Returns an optic of the appropriate supertype. If either or both optics are indexed, the composition preserves all the indices.
(%%) :: forall k is js ks s t u v a b. AppendIndices is js ks => Optic k is s t u v -> Optic k js u v a b -> Optic k ks s t a b infixl 9 Source #
Compose two optics of the same flavour.
Normally you can simply use (
%
) instead, but this may be useful to help
type inference if the type of one of the optics is otherwise
under-constrained.
(%&) :: Optic k is s t a b -> ( Optic k is s t a b -> Optic l js s' t' a' b') -> Optic l js s' t' a' b' infixl 9 Source #
Flipped function application, specialised to optics and binding tightly.
Useful for post-composing optics transformations:
>>>
toListOf (ifolded %& ifiltered (\i s -> length s <= i)) ["", "a","abc"]
["","a"]
Monoid structures
Fold
-like optics admit various monoid structures (e.g. see
Optics.Fold
). There is no
Semigroup
or
Monoid
instance for
Optic
, however, because there is not a unique choice of monoid to use,
and the (
<>
) operator could not be used to combine optics of different
kinds.
Indexed optics
See the "Indexed optics" section of the overview documentation in the
Optics
module of the main
optics
package for more details on indexed
optics.
class AppendIndices xs ys ks | xs ys -> ks Source #
In pseudo (dependent-)Haskell, provide a witness
foldr f (foldr f init xs) ys = foldr f init (ys ++ xs) where f = (->)
Since: 0.4
Instances
AppendIndices xs ('[] :: [ Type ]) xs Source # |
If the second list is empty, we can shortcircuit and pick the first list immediately. |
Defined in Optics.Internal.Optic.TypeLevel |
|
AppendIndices ('[] :: [ Type ]) ys ys Source # | |
Defined in Optics.Internal.Optic.TypeLevel |
|
AppendIndices xs ys ks => AppendIndices (x ': xs) ys (x ': ks) Source # | |
Defined in Optics.Internal.Optic.TypeLevel |
class NonEmptyIndices (is :: IxList ) Source #
Check whether a list of indices is not empty and generate sensible error message if it's not.
Instances
( TypeError (' Text "Indexed optic is expected") :: Constraint ) => NonEmptyIndices ('[] :: [ Type ]) Source # | |
Defined in Optics.Internal.Indexed |
|
NonEmptyIndices (x ': xs) Source # | |
Defined in Optics.Internal.Indexed |
class is ~ '[i] => HasSingleIndex (is :: IxList ) (i :: Type ) Source #
Generate sensible error messages in case a user tries to pass either an unindexed optic or indexed optic with unflattened indices where indexed optic with a single index is expected.
Instances
( TypeError (' Text "Indexed optic is expected") :: Constraint , ('[] :: [ Type ]) ~ '[i]) => HasSingleIndex ('[] :: [ Type ]) i Source # | |
Defined in Optics.Internal.Indexed |
|
( TypeError (' Text "Use icomposeN to flatten indices of type " :<>: ShowTypes is) :: Constraint , is ~ (i1 ': (i2 ': (i3 ': (i4 ': (i5 ': (i6 ': is')))))), is ~ '[i]) => HasSingleIndex (i1 ': (i2 ': (i3 ': (i4 ': (i5 ': (i6 ': is')))))) i Source # | |
Defined in Optics.Internal.Indexed |
|
( TypeError (' Text "Use icompose5 to flatten indices of type " :<>: ShowTypes is) :: Constraint , is ~ '[i1, i2, i3, i4, i5], is ~ '[i]) => HasSingleIndex '[i1, i2, i3, i4, i5] i Source # | |
Defined in Optics.Internal.Indexed |
|
( TypeError (' Text "Use icompose4 to combine indices of type " :<>: ShowTypes is) :: Constraint , is ~ '[i1, i2, i3, i4], is ~ '[i]) => HasSingleIndex '[i1, i2, i3, i4] i Source # | |
Defined in Optics.Internal.Indexed |
|
( TypeError (' Text "Use icompose3 to combine indices of type " :<>: ShowTypes is) :: Constraint , is ~ '[i1, i2, i3], is ~ '[i]) => HasSingleIndex '[i1, i2, i3] i Source # | |
Defined in Optics.Internal.Indexed |
|
( TypeError (' Text "Use (<%>) or icompose to combine indices of type " :<>: ShowTypes is) :: Constraint , is ~ '[i1, i2], is ~ '[i]) => HasSingleIndex '[i1, i2] i Source # | |
Defined in Optics.Internal.Indexed |
|
HasSingleIndex '[i] i Source # | |
Defined in Optics.Internal.Indexed |
class is ~ NoIx => AcceptsEmptyIndices (f :: Symbol ) (is :: IxList ) Source #
Show useful error message when a function expects optics without indices.
Instances
AcceptsEmptyIndices f ('[] :: [ Type ]) Source # | |
Defined in Optics.Internal.Indexed |
|
( TypeError ((' Text "\8216" :<>: ' Text f) :<>: ' Text "\8217 accepts only optics with no indices") :: Constraint , (x ': xs) ~ NoIx ) => AcceptsEmptyIndices f (x ': xs) Source # | |
Defined in Optics.Internal.Indexed |
class CurryCompose xs where Source #
Class that is inhabited by all type-level lists
xs
, providing the ability
to compose a function under
.
Curry
xs
Instances
CurryCompose ('[] :: [ Type ]) Source # | |
CurryCompose xs => CurryCompose (x ': xs) Source # | |