Copyright | (C) 2013-2016 Edward Kmett 2018 Monadfix |
---|---|
License | BSD-style (see the file LICENSE) |
Safe Haskell | Safe-Inferred |
Language | Haskell2010 |
This module is home to lens definitions that require
profunctors, most notably
Iso
and Prism
. Depending on profunctors
is quite the to bear — one
that includes all dependencies of microlens-platform
. For this reason,
microlens-pro
re-exports the entirety of Lens.Micro.Platform, but
with the profunctor-less definitions hidden and overridden with profunctor'd
definitions from this module.
Synopsis
- type Iso s t a b = forall p f. (Profunctor p, Functor f) => p a (f b) -> p s (f t)
- type Iso' s a = Iso s s a a
- iso :: (s -> a) -> (b -> t) -> Iso s t a b
- from :: AnIso s t a b -> Iso b a t s
- under :: AnIso s t a b -> (t -> s) -> b -> a
- non :: Eq a => a -> Iso' (Maybe a) a
- non' :: APrism' a () -> Iso' (Maybe a) a
- _Show :: (Read a, Show a) => Iso' String a
- strict :: Strict lazy strict => Iso' lazy strict
- lazy :: Strict lazy strict => Iso' strict lazy
- enum :: Enum a => Iso' Int a
- coerced :: forall s t a b. (Coercible s a, Coercible t b) => Iso s t a b
- mapping :: (Functor f, Functor g) => AnIso s t a b -> Iso (f s) (g t) (f a) (g b)
- packed :: IsText t => Iso' String t
- unpacked :: IsText t => Iso' t String
- type AnIso s t a b = Exchange a b a (Identity b) -> Exchange a b s (Identity t)
- type AnIso' s a = AnIso s s a a
- cloneIso :: AnIso s t a b -> Iso s t a b
- withIso :: forall s t a b rep (r :: TYPE rep). AnIso s t a b -> ((s -> a) -> (b -> t) -> r) -> r
- type Prism s t a b = forall p f. (Choice p, Applicative f) => p a (f b) -> p s (f t)
- type Prism' s a = Prism s s a a
- prism :: (b -> t) -> (s -> Either t a) -> Prism s t a b
- prism' :: (b -> s) -> (s -> Maybe a) -> Prism s s a b
- nearly :: a -> (a -> Bool) -> Prism' a ()
- only :: Eq a => a -> Prism' a ()
- _Left :: Prism (Either a c) (Either b c) a b
- _Right :: Prism (Either c a) (Either c b) a b
- _Just :: Prism (Maybe a) (Maybe b) a b
- _Nothing :: Prism' (Maybe a) ()
- _Empty :: AsEmpty a => Prism' a ()
- type APrism s t a b = Market a b a (Identity b) -> Market a b s (Identity t)
- type APrism' s a = Market a a a (Identity a) -> Market a a s (Identity s)
- clonePrism :: APrism s t a b -> Prism s t a b
- withPrism :: APrism s t a b -> ((b -> t) -> (s -> Either t a) -> r) -> r
- type AReview t b = Tagged b (Identity b) -> Tagged t (Identity t)
- type SimpleReview t b = forall p. (Choice p, Bifunctor p) => p b (Identity b) -> p t (Identity t)
- re :: AReview t b -> Getter b t
- review :: MonadReader b m => AReview t b -> m t
- (#) :: AReview t b -> b -> t
- unto :: (Profunctor p, Bifunctor p, Functor f) => (b -> t) -> p a (f b) -> p s (f t)
- type LensLike' (f :: Type -> Type) s a = LensLike f s s a a
- type LensLike (f :: Type -> Type) s t a b = (a -> f b) -> s -> f t
- type Traversal' s a = Traversal s s a a
- type Traversal s t a b = forall (f :: Type -> Type). Applicative f => (a -> f b) -> s -> f t
- type Lens' s a = Lens s s a a
- type Lens s t a b = forall (f :: Type -> Type). Functor f => (a -> f b) -> s -> f t
- type SimpleFold s a = forall r. Monoid r => Getting r s a
- type Getting r s a = (a -> Const r a) -> s -> Const r s
- type SimpleGetter s a = forall r. Getting r s a
- type ASetter' s a = ASetter s s a a
- type ASetter s t a b = (a -> Identity b) -> s -> Identity t
- data DefName
- data LensRules
- (<&>) :: Functor f => f a -> (a -> b) -> f b
- (&) :: a -> (a -> b) -> b
- to :: (s -> a) -> SimpleGetter s a
- traverseOf_ :: Functor f => Getting (Traversed r f) s a -> (a -> f r) -> s -> f ()
- (<<%~) :: LensLike ((,) a) s t a b -> (a -> b) -> s -> (a, t)
- (^.) :: s -> Getting a s a -> a
- (^..) :: s -> Getting (Endo [a]) s a -> [a]
- (^?) :: s -> Getting (First a) s a -> Maybe a
- view :: MonadReader s m => Getting a s a -> m a
- (.~) :: ASetter s t a b -> b -> s -> t
- (?~) :: ASetter s t a (Maybe b) -> b -> s -> t
- preview :: MonadReader s m => Getting (First a) s a -> m (Maybe a)
- _5 :: Field5 s t a b => Lens s t a b
- _4 :: Field4 s t a b => Lens s t a b
- _3 :: Field3 s t a b => Lens s t a b
- _2 :: Field2 s t a b => Lens s t a b
- _1 :: Field1 s t a b => Lens s t a b
- at :: At m => Index m -> Lens' m (Maybe (IxValue m))
- ix :: Ixed m => Index m -> Traversal' m (IxValue m)
- each :: Each s t a b => Traversal s t a b
- traversed :: forall (f :: Type -> Type) a b. Traversable f => Traversal (f a) (f b) a b
- folded :: forall (f :: Type -> Type) a. Foldable f => SimpleFold (f a) a
- sets :: ((a -> b) -> s -> t) -> ASetter s t a b
- (%~) :: ASetter s t a b -> (a -> b) -> s -> t
- over :: ASetter s t a b -> (a -> b) -> s -> t
- (+~) :: Num a => ASetter s t a a -> a -> s -> t
- (-~) :: Num a => ASetter s t a a -> a -> s -> t
- (<>~) :: Monoid a => ASetter s t a a -> a -> s -> t
- set :: ASetter s t a b -> b -> s -> t
- mapped :: Functor f => ASetter (f a) (f b) a b
- (<%~) :: LensLike ((,) b) s t a b -> (a -> b) -> s -> (b, t)
- (<<.~) :: LensLike ((,) a) s t a b -> b -> s -> (a, t)
- rewriteOf :: ASetter a b a b -> (b -> Maybe a) -> a -> b
- transformOf :: ASetter a b a b -> (b -> b) -> a -> b
- toListOf :: Getting (Endo [a]) s a -> s -> [a]
- (^?!) :: HasCallStack => s -> Getting (Endo a) s a -> a
- forOf_ :: Functor f => Getting (Traversed r f) s a -> s -> (a -> f r) -> f ()
- has :: Getting Any s a -> s -> Bool
- folding :: Foldable f => (s -> f a) -> SimpleFold s a
- lens :: (s -> a) -> (s -> b -> t) -> Lens s t a b
- traverseOf :: LensLike f s t a b -> (a -> f b) -> s -> f t
- forOf :: LensLike f s t a b -> s -> (a -> f b) -> f t
- singular :: HasCallStack => Traversal s t a a -> Lens s t a a
- failing :: Traversal s t a b -> Traversal s t a b -> Traversal s t a b
- filtered :: (a -> Bool) -> Traversal' a a
- both :: forall a b f. Applicative f => (a -> f b) -> (a, a) -> f (b, b)
- _head :: Cons s s a a => Traversal' s a
- _tail :: Cons s s a a => Traversal' s s
- _init :: Snoc s s a a => Traversal' s s
- _last :: Snoc s s a a => Traversal' s a
- mapAccumLOf :: LensLike (State acc) s t a b -> (acc -> a -> (acc, b)) -> acc -> s -> (acc, t)
- worded :: Traversal' String String
- lined :: Traversal' String String
- packedBytes :: IsByteString t => Lens' [Word8] t
- unpackedBytes :: IsByteString t => Lens' t [Word8]
- packedChars :: IsByteString t => Lens' String t
- unpackedChars :: IsByteString t => Lens' t String
- chars :: IsByteString t => Traversal' t Char
- (+=) :: (MonadState s m, Num a) => ASetter s s a a -> a -> m ()
- use :: MonadState s m => Getting a s a -> m a
- (.=) :: MonadState s m => ASetter s s a b -> b -> m ()
- magnify :: Magnify m n b a => LensLike' (Magnified m c) a b -> m c -> n c
- zoom :: Zoom m n s t => LensLike' (Zoomed m c) t s -> m c -> n c
- preuse :: MonadState s m => Getting (First a) s a -> m (Maybe a)
- (&~) :: s -> State s a -> s
- assign :: MonadState s m => ASetter s s a b -> b -> m ()
- (?=) :: MonadState s m => ASetter s s a (Maybe b) -> b -> m ()
- (<~) :: MonadState s m => ASetter s s a b -> m b -> m ()
- (%=) :: MonadState s m => ASetter s s a b -> (a -> b) -> m ()
- modifying :: MonadState s m => ASetter s s a b -> (a -> b) -> m ()
- (-=) :: (MonadState s m, Num a) => ASetter s s a a -> a -> m ()
- (*=) :: (MonadState s m, Num a) => ASetter s s a a -> a -> m ()
- (//=) :: (MonadState s m, Fractional a) => ASetter s s a a -> a -> m ()
- (<%=) :: MonadState s m => LensLike ((,) b) s s a b -> (a -> b) -> m b
- (<<%=) :: MonadState s m => LensLike ((,) a) s s a b -> (a -> b) -> m a
- (<<.=) :: MonadState s m => LensLike ((,) a) s s a b -> b -> m a
- (<.=) :: MonadState s m => LensLike ((,) b) s s a b -> b -> m b
- (<?=) :: MonadState s m => LensLike ((,) b) s s a (Maybe b) -> b -> m b
- makeLenses :: Name -> DecsQ
- makeLensesFor :: [(String, String)] -> Name -> DecsQ
- makeLensesWith :: LensRules -> Name -> DecsQ
- makeFields :: Name -> DecsQ
- makeClassy :: Name -> DecsQ
- simpleLenses :: Lens' LensRules Bool
- generateSignatures :: Lens' LensRules Bool
- generateUpdateableOptics :: Lens' LensRules Bool
- generateLazyPatterns :: Lens' LensRules Bool
- lensField :: Lens' LensRules (Name -> [Name] -> Name -> [DefName])
- lensClass :: Lens' LensRules (Name -> Maybe (Name, Name))
- createClass :: Lens' LensRules Bool
- lensRules :: LensRules
- lensRulesFor :: [(String, String)] -> LensRules
- camelCaseFields :: LensRules
- abbreviatedFields :: LensRules
- classyRules :: LensRules
Iso: Losslessly convert between types
Isos (or isomorphisms) are lenses that convert a value instead of targeting a
part of it; in other words, inside of every list lives a reversed list, inside
of every strict Text
lives a lazy Text
, and inside of every (a, b)
lives a
(b, a)
. Since an isomorphism doesn't lose any information, it's possible to
reverse it and use it in the opposite direction by using from
:
from :: Iso' s a -> Iso' a s from :: Iso s t a b -> Iso t s b a
Isos are constructed from a pair of inverse functions. For example, assume
lawful instances of Show
and Read
:
show . read = id read . show = id
The isomorphisms defined in this module are true lens-compatible isos. Many of them share names with the lens-incompatible definitions from Lens.Micro and Lens.Micro.Platform. For convenience, we re-export Lens.Micro.Platform, but with non-lens-compatible isos hidden and replaced with lens-compatbile ones.
type Iso s t a b = forall p f. (Profunctor p, Functor f) => p a (f b) -> p s (f t) Source #
The type signature of iso
provides a nice interpretation of
Iso
. If you want to apply a function a -> b
to a type s
, you'd have to
convert with s -> a
, apply your function a -> b
, and convert back with
b -> t
.
iso
:: (s -> a) -> (b -> t) -> Iso s t a b -- or, put monomorphicallyiso
:: (s -> a) -> (a -> s) -> Iso' s a
type Iso' s a = Iso s s a a Source #
The type of monomorphic isomorphisms, i.e. isos that change neither the outer type
s
nor the inner type a
.
Constructing Isos
Iso Combinators
non :: Eq a => a -> Iso' (Maybe a) a Source #
non
lets you “relabel” a Maybe
by equating Nothing
to an arbitrary value
(which you can choose):
>>>
Just 1 ^. non 0 1
>>>
Nothing ^. non 0 0
The most useful thing about non
is that relabeling also works in other
direction. If you try to set
the “forbidden” value, it'll be turned to
Nothing
:
>>>
Just 1 & non 0 .~ 0 Nothing
Setting anything else works just fine:
>>>
Just 1 & non 0 .~ 5 Just 5
Same happens if you try to modify a value:
>>>
Just 1 & non 0 %~ subtract 1 Nothing
>>>
Just 1 & non 0 %~ (+ 1) Just 2
non
is often useful when combined with at
. For instance, if you have a map
of songs and their playcounts, it makes sense not to store songs with 0 plays in
the map; non
can act as a filter that wouldn't pass such entries.
Decrease playcount of a song to 0, and it'll be gone:
>>>
fromList [("Soon",1),("Yesterday",3)] & at "Soon" . non 0 %~ subtract 1
fromList [("Yesterday",3)]
Try to add a song with 0 plays, and it won't be added:
>>>
fromList [("Yesterday",3)] & at "Soon" . non 0 .~ 0
fromList [("Yesterday",3)]
But it will be added if you set any other number:
>>>
fromList [("Yesterday",3)] & at "Soon" . non 0 .~ 1
fromList [("Soon",1),("Yesterday",3)]
non
is also useful when working with nested maps. Here a nested map is created
when it's missing:
>>>
Map.empty & at "Dez Mona" . non Map.empty . at "Soon" .~ Just 1
fromList [("Dez Mona",fromList [("Soon",1)])]
and here it is deleted when its last entry is deleted (notice that non
is used
twice here):
>>>
fromList [("Dez Mona",fromList [("Soon",1)])] & at "Dez Mona" . non Map.empty . at "Soon" . non 0 %~ subtract 1
fromList []
To understand the last example better, observe the flow of values in it:
- the map goes into
at "Dez Mona"
* the nested map (wrapped intoJust
) goes intonon Map.empty
*Just
is unwrapped and the nested map goes intoat "Soon"
*Just 1
is unwrapped bynon 0
Then the final value – i.e. 1 – is modified by subtract 1
and the result
(which is 0) starts flowing backwards:
non 0
sees the 0 and produces aNothing
at "Soon"
seesNothing
and deletes the corresponding value from the map- the resulting empty map is passed to
non Map.empty
, which sees that it's empty and thus producesNothing
at "Dez Mona"
seesNothing
and removes the key from the map
Common Isos
enum :: Enum a => Iso' Int a Source #
enum
is a questionable inclusion, as many (most) Enum
instances throw
errors for out-of-bounds integers, but it is occasionally useful when used with
that information in mind. Handle with care!
>>>
97 ^. enum :: Char
'a'>>>
(-1) ^. enum :: Char
*** Exception: Prelude.chr: bad argument: (-1)>>>
[True,False] ^. mapping (from enum)
[1,0]
coerced :: forall s t a b. (Coercible s a, Coercible t b) => Iso s t a b Source #
Coercible types have the same runtime representation, i.e. they are isomorphic.
>>>
(Sum 123 :: Sum Int) ^. coerced :: Int
123
mapping :: (Functor f, Functor g) => AnIso s t a b -> Iso (f s) (g t) (f a) (g b) Source #
An isomorphism holds when lifted into a functor. For example, if a list contains
a bunch of a
's which are each isomorphic to a b
, the whole list of a
's is
isomorphic to a list of b
's.
>>>
["1","2","3"] ^. mapping _Show :: [Int]
[1,2,3]>>>
([1,2,3] :: [Int]) ^. from (mapping _Show)
["1","2","3"]
This also hold across different functors:
>>>
let l = mapping @[] @Maybe _Show
>>>
:t l
l :: (Read b, Show b) => Iso [String] (Maybe String) [b] (Maybe b)>>>
["1","2","3"] & l %~ Just . sum
Just "6"
Miscellaneous
withIso :: forall s t a b rep (r :: TYPE rep). AnIso s t a b -> ((s -> a) -> (b -> t) -> r) -> r Source #
Extract the two functions, s -> a
and one b -> t
that characterize an
Iso
.
Prism: A traversal with zero or one targets
If a Lens
views and updates individual components of product types, a
Prism
views and updates individual components of sum types. For example, you
may want to update the Left
field of an Either
:
>>>
Left "salmon" & _Left .~ "orb"
Left "orb">>>
Right "pudding" & _Left .~ "orb"
Right "pudding"
Also similarly to a Lens
, you might want to view the Left
field. However, it
might not always be there, so we treat it as a traversal with either one or zero
results.
>>>
Right "bass" ^? _Left
Nothing>>>
Left "bubbles" ^? _Left
Just "bubbles"
A unique feature of Prism
s is that they may be flipped around using re
to
construct the larger structure. Maintaining our example of Either
, remember
that you can construct the entire Either
via the constructor Left
.
>>>
:t re _Left
re _Left :: Getter b (Either b c)>>>
view (re _Left) "bungo"
Left "bungo"
This
idiom isn't the prettiest, so we define view
(re
l)
as shorthand. review
=
view
. re
review
also has an infix synonym, (#)
.
>>>
:t _Just
_Just :: Prism (Maybe a) (Maybe b) a b>>>
review _Just "bilbo"
Just "bilbo">>>
_Just # "bilbo"
Just "bilbo"
As is the whole point of optics, prisms may of course be composed with other optics:
type Thing = Either (Maybe String) (Maybe (Either [Bool] Int)) thing :: Thing thing = Right (Just (Left [True,False]))
>>>
thing & _Right . _Just . _Left . each %~ not
Right (Just (Left [False,True]))
type Prism s t a b = forall p f. (Choice p, Applicative f) => p a (f b) -> p s (f t) Source #
s
is the type of the whole structuret
is the type of the reconstructed structurea
is the type of the targetb
is the type of the value used for reconstruction
type Prism' s a = Prism s s a a Source #
The type of monomorphic prisms, i.e. prisms that change neither the outer type
s
nor the inner type a
.
Constructing Prisms
prism' :: (b -> s) -> (s -> Maybe a) -> Prism s s a b Source #
Generate a Prism
out of a constructor and a selector.
_Nothing = prism Left $ either Right (Left . Right)
Prism Combinators
nearly :: a -> (a -> Bool) -> Prism' a () Source #
is a prism that matches "loose equality" with nearly
a pa
by assuming p
x
is true iff x ≡ a
.
>>>
nearly [] null # ()
[]>>>
[1,2,3,4] ^? nearly [] null
Nothing
only :: Eq a => a -> Prism' a () Source #
A prism that matches equality with a value:
>>>
1 ^? only 2
Nothing>>>
1 ^? only 1
Just 1
Common Prisms
_Empty :: AsEmpty a => Prism' a () Source #
A prism that matches the empty structure.
>>>
has _Empty []
True
Miscellaneous
type APrism' s a = Market a a a (Identity a) -> Market a a s (Identity s) Source #
Monomorphic APrism
.
clonePrism :: APrism s t a b -> Prism s t a b Source #
Review
type SimpleReview t b = forall p. (Choice p, Bifunctor p) => p b (Identity b) -> p t (Identity t) Source #
Review
,
from lens, is limited form of Prism
that can only be used for re
operations.
Similarly to SimpleGetter
from microlens, microlens-pro does not define Review
and opts for
a less general SimpleReview
in order to avoid a
distributive
dependency.
review :: MonadReader b m => AReview t b -> m t Source #
unto :: (Profunctor p, Bifunctor p, Functor f) => (b -> t) -> p a (f b) -> p s (f t) Source #
Construct a Review
out of a constructor. Consider this more pleasant type
signature:
unto :: (b -> t) -> Review' t b
Pardon the actual type signature — microlens defines neither Optic
(used in
lens'
unto
) nor Review'
. Here we simply expand the definition of Optic
.
type LensLike' (f :: Type -> Type) s a = LensLike f s s a a #
A type alias for monomorphic LensLike
s.
type LensLike (f :: Type -> Type) s t a b = (a -> f b) -> s -> f t #
LensLike
is a type that is often used to make combinators as general as possible. For instance, take (<<%~
), which only requires the passed lens to be able to work with the (,) a
functor (lenses and traversals can do that). The fully expanded type is as follows:
(<<%~
) :: ((a -> (a, b)) -> s -> (a, t)) -> (a -> b) -> s -> (a, t)
With LensLike
, the intent to use the (,) a
functor can be made a bit clearer:
(<<%~
) :: LensLike ((,) a) s t a b -> (a -> b) -> s -> (a, t)
type Traversal' s a = Traversal s s a a #
This is a type alias for monomorphic traversals which don't change the type of the container (or of the values inside).
type Traversal s t a b = forall (f :: Type -> Type). Applicative f => (a -> f b) -> s -> f t #
Traversal s t a b
is a generalisation of Lens
which allows many targets (possibly 0). It's achieved by changing the constraint to Applicative
instead of Functor
– indeed, the point of Applicative
is that you can combine effects, which is just what we need to have many targets.
Ultimately, traversals should follow 2 laws:
t pure ≡ pure fmap (t f) . t g ≡ getCompose . t (Compose . fmap f . g)
The 1st law states that you can't change the shape of the structure or do anything funny with elements (traverse elements which aren't in the structure, create new elements out of thin air, etc.). The 2nd law states that you should be able to fuse 2 identical traversals into one. For a more detailed explanation of the laws, see this blog post (if you prefer rambling blog posts), or The Essence Of The Iterator Pattern (if you prefer papers).
Traversing any value twice is a violation of traversal laws. You can, however, traverse values in any order.
type Lens' s a = Lens s s a a #
This is a type alias for monomorphic lenses which don't change the type of the container (or of the value inside).
type Lens s t a b = forall (f :: Type -> Type). Functor f => (a -> f b) -> s -> f t #
Lens s t a b
is the lowest common denominator of a setter and a getter, something that has the power of both; it has a Functor
constraint, and since both Const
and Identity
are functors, it can be used whenever a getter or a setter is needed.
a
is the type of the value inside of structureb
is the type of the replaced values
is the type of the whole structuret
is the type of the structure after replacinga
in it withb
type SimpleFold s a = forall r. Monoid r => Getting r s a #
A SimpleFold s a
extracts several a
s from s
; so, it's pretty much the same thing as (s -> [a])
, but you can use it with lens operators.
The actual Fold
from lens is more general:
type Fold s a = forall f. (Contravariant f, Applicative f) => (a -> f a) -> s -> f s
There are several functions in lens that accept lens's Fold
but won't accept SimpleFold
; I'm aware of
takingWhile
,
droppingWhile
,
backwards
,
foldByOf
,
foldMapByOf
.
For this reason, try not to export SimpleFold
s if at all possible. microlens-contra provides a fully lens-compatible Fold
.
Lens users: you can convert a SimpleFold
to Fold
by applying folded . toListOf
to it.
type Getting r s a = (a -> Const r a) -> s -> Const r s #
Functions that operate on getters and folds – such as (^.
), (^..
), (^?
) – use Getter r s a
(with different values of r
) to describe what kind of result they need. For instance, (^.
) needs the getter to be able to return a single value, and so it accepts a getter of type Getting a s a
. (^..
) wants the getter to gather values together, so it uses Getting (Endo [a]) s a
(it could've used Getting [a] s a
instead, but it's faster with Endo
). The choice of r
depends on what you want to do with elements you're extracting from s
.
type SimpleGetter s a = forall r. Getting r s a #
A SimpleGetter s a
extracts a
from s
; so, it's the same thing as (s -> a)
, but you can use it in lens chains because its type looks like this:
type SimpleGetter s a = forall r. (a -> Const r a) -> s -> Const r s
Since Const r
is a functor, SimpleGetter
has the same shape as other lens types and can be composed with them. To get (s -> a)
out of a SimpleGetter
, choose r ~ a
and feed Const :: a -> Const a a
to the getter:
-- the actual signature is more permissive: --view
::Getting
a s a -> s -> aview
::SimpleGetter
s a -> s -> aview
getter =getConst
. getterConst
The actual Getter
from lens is more general:
type Getter s a = forall f. (Contravariant f, Functor f) => (a -> f a) -> s -> f s
I'm not currently aware of any functions that take lens's Getter
but won't accept SimpleGetter
, but you should try to avoid exporting SimpleGetter
s anyway to minimise confusion. Alternatively, look at microlens-contra, which provides a fully lens-compatible Getter
.
Lens users: you can convert a SimpleGetter
to Getter
by applying to . view
to it.
type ASetter s t a b = (a -> Identity b) -> s -> Identity t #
ASetter s t a b
is something that turns a function modifying a value into a function modifying a structure. If you ignore Identity
(as Identity a
is the same thing as a
), the type is:
type ASetter s t a b = (a -> b) -> s -> t
The reason Identity
is used here is for ASetter
to be composable with other types, such as Lens
.
Technically, if you're writing a library, you shouldn't use this type for setters you are exporting from your library; the right type to use is Setter
, but it is not provided by this package (because then it'd have to depend on distributive). It's completely alright, however, to export functions which take an ASetter
as an argument.
Name to give to a generated lens (used in lensField
).
TopName Name | Simple top-level definiton name |
MethodName Name Name |
|
Rules used to generate lenses. The fields are intentionally not exported; to create your own rules, see lenses in the “Configuring lens rules” section. You'd have to customise one of the existing rulesets; for an example of doing that, see makeLensesWith
.
to :: (s -> a) -> SimpleGetter s a #
to
creates a getter from any function:
a^.
to
f = f a
It's most useful in chains, because it lets you mix lenses and ordinary functions. Suppose you have a record which comes from some third-party library and doesn't have any lens accessors. You want to do something like this:
value ^. _1 . field . at 2
However, field
isn't a getter, and you have to do this instead:
field (value ^. _1) ^. at 2
but now value
is in the middle and it's hard to read the resulting code. A variant with to
is prettier and more readable:
value ^. _1 . to field . at 2
traverseOf_ :: Functor f => Getting (Traversed r f) s a -> (a -> f r) -> s -> f () #
Apply an action to all targets and discard the result (like mapM_
or traverse_
):
>>>
traverseOf_ both putStrLn ("hello", "world")
hello world
Works with anything that allows getting, including lenses and getters (so, anything except for ASetter
). Should be faster than traverseOf
when you don't need the result.
(^.) :: s -> Getting a s a -> a infixl 8 #
(^.
) applies a getter to a value; in other words, it gets a value out of a structure using a getter (which can be a lens, traversal, fold, etc.).
Getting 1st field of a tuple:
(^.
_1
) :: (a, b) -> a (^.
_1
) =fst
When (^.
) is used with a traversal, it combines all results using the Monoid
instance for the resulting type. For instance, for lists it would be simple concatenation:
>>>
("str","ing") ^. each
"string"
The reason for this is that traversals use Applicative
, and the Applicative
instance for Const
uses monoid concatenation to combine “effects” of Const
.
A non-operator version of (^.
) is called view
, and it's a bit more general than (^.
) (it works in MonadReader
). If you need the general version, you can get it from microlens-mtl; otherwise there's view
available in Lens.Micro.Extras.
(^..) :: s -> Getting (Endo [a]) s a -> [a] infixl 8 #
s ^.. t
returns the list of all values that t
gets from s
.
A Maybe
contains either 0 or 1 values:
>>>
Just 3 ^.. _Just
[3]
Gathering all values in a list of tuples:
>>>
[(1,2),(3,4)] ^.. each.each
[1,2,3,4]
(^?) :: s -> Getting (First a) s a -> Maybe a infixl 8 #
s ^? t
returns the 1st element t
returns, or Nothing
if t
doesn't return anything. It's trivially implemented by passing the First
monoid to the getter.
Safe head
:
>>>
[] ^? each
Nothing
>>>
[1..3] ^? each
Just 1
>>>
Left 1 ^? _Right
Nothing
>>>
Right 1 ^? _Right
Just 1
A non-operator version of (^?
) is called preview
, and – like view
– it's a bit more general than (^?
) (it works in MonadReader
). If you need the general version, you can get it from microlens-mtl; otherwise there's preview
available in Lens.Micro.Extras.
view :: MonadReader s m => Getting a s a -> m a #
view
is a synonym for (^.
), generalised for MonadReader
(we are able to use it instead of (^.
) since functions are instances of the MonadReader
class):
>>>
view _1 (1, 2)
1
When you're using Reader
for config and your config type has lenses generated for it, most of the time you'll be using view
instead of asks
:
doSomething :: (MonadReader
Config m) => m Int doSomething = do thingy <-view
setting1 -- same as “asks
(^.
setting1)” anotherThingy <-view
setting2 ...
preview :: MonadReader s m => Getting (First a) s a -> m (Maybe a) #
preview
is a synonym for (^?
), generalised for MonadReader
(just like view
, which is a synonym for (^.
)).
>>>
preview each [1..5]
Just 1
_1 :: Field1 s t a b => Lens s t a b #
Gives access to the 1st field of a tuple (up to 5-tuples).
Getting the 1st component:
>>>
(1,2,3,4,5) ^. _1
1
Setting the 1st component:
>>>
(1,2,3) & _1 .~ 10
(10,2,3)
Note that this lens is lazy, and can set fields even of undefined
:
>>>
set _1 10 undefined :: (Int, Int)
(10,*** Exception: Prelude.undefined
This is done to avoid violating a lens law stating that you can get back what you put:
>>>
view _1 . set _1 10 $ (undefined :: (Int, Int))
10
The implementation (for 2-tuples) is:
_1
f t = (,)<$>
f (fst
t)<*>
pure
(snd
t)
or, alternatively,
_1
f ~(a,b) = (\a' -> (a',b))<$>
f a
(where ~
means a lazy pattern).
at :: At m => Index m -> Lens' m (Maybe (IxValue m)) #
This lens lets you read, write, or delete elements in Map
-like structures. It returns Nothing
when the value isn't found, just like lookup
:
Data.Map.lookup k m = m ^.
at k
However, it also lets you insert and delete values by setting the value to
or Just
valueNothing
:
Data.Map.insert k a m = m&
at k.~
Just a Data.Map.delete k m = m&
at k.~
Nothing
Or you could use (?~
) instead of (.~
):
Data.Map.insert k a m = m&
at k?~
a
Note that at
doesn't work for arrays or lists. You can't delete an arbitrary element from an array (what would be left in its place?), and you can't set an arbitrary element in a list because if the index is out of list's bounds, you'd have to somehow fill the stretch between the last element and the element you just inserted (i.e. [1,2,3] & at 10 .~ 5
is undefined). If you want to modify an already existing value in an array or list, you should use ix
instead.
at
is often used with non
. See the documentation of non
for examples.
Note that at
isn't strict for Map
, even if you're using Data.Map.Strict
:
>>>
Data.Map.Strict.size (Data.Map.Strict.empty & at 1 .~ Just undefined)
1
The reason for such behavior is that there's actually no “strict Map
” type; Data.Map.Strict
just provides some strict functions for ordinary Map
s.
This package doesn't actually provide any instances for at
, but there are instances for Map
and IntMap
in microlens-ghc and an instance for HashMap
in microlens-platform.
ix :: Ixed m => Index m -> Traversal' m (IxValue m) #
This traversal lets you access (and update) an arbitrary element in a list, array, Map
, etc. (If you want to insert or delete elements as well, look at at
.)
An example for lists:
>>>
[0..5] & ix 3 .~ 10
[0,1,2,10,4,5]
You can use it for getting, too:
>>>
[0..5] ^? ix 3
Just 3
Of course, the element may not be present (which means that you can use ix
as a safe variant of (!!
)):
>>>
[0..5] ^? ix 10
Nothing
Another useful instance is the one for functions – it lets you modify their outputs for specific inputs. For instance, here's maximum
that returns 0 when the list is empty (instead of throwing an exception):
maximum0 =maximum
&
ix
[].~
0
The following instances are provided in this package:
ix
::Int
->Traversal'
[a] aix
::Int
->Traversal'
(NonEmpty a) aix
:: (Eq
e) => e ->Traversal'
(e -> a) a
You can also use ix
with types from array, bytestring, and containers by using microlens-ghc, or additionally with types from vector, text, and unordered-containers by using microlens-platform.
each :: Each s t a b => Traversal s t a b #
each
tries to be a universal Traversal
– it behaves like traversed
in most situations, but also adds support for e.g. tuples with same-typed values:
>>>
(1,2) & each %~ succ
(2,3)
>>>
["x", "y", "z"] ^. each
"xyz"
However, note that each
doesn't work on every instance of Traversable
. If you have a Traversable
which isn't supported by each
, you can use traversed
instead. Personally, I like using each
instead of traversed
whenever possible – it's shorter and more descriptive.
You can use each
with these things:
each
::Traversal
[a] [b] a beach
::Traversal
(Maybe
a) (Maybe
b) a beach
::Traversal
(Either
a a) (Either
b b) a b -- since 0.4.11each
::Traversal
(a,a) (b,b) a beach
::Traversal
(a,a,a) (b,b,b) a beach
::Traversal
(a,a,a,a) (b,b,b,b) a beach
::Traversal
(a,a,a,a,a) (b,b,b,b,b) a beach
:: (RealFloat
a,RealFloat
b) =>Traversal
(Complex
a) (Complex
b) a b
You can also use each
with types from array, bytestring, and containers by using microlens-ghc, or additionally with types from vector, text, and unordered-containers by using microlens-platform.
traversed :: forall (f :: Type -> Type) a b. Traversable f => Traversal (f a) (f b) a b #
traversed
traverses any Traversable
container (list, vector, Map
, Maybe
, you name it):
>>>
Just 1 ^.. traversed
[1]
traversed
is the same as traverse
, but can be faster thanks to magic rewrite rules.
(%~) :: ASetter s t a b -> (a -> b) -> s -> t infixr 4 #
(%~
) applies a function to the target; an alternative explanation is that it is an inverse of sets
, which turns a setter into an ordinary function.
is the same thing as mapped
%~
reverse
.fmap
reverse
See over
if you want a non-operator synonym.
Negating the 1st element of a pair:
>>>
(1,2) & _1 %~ negate
(-1,2)
Turning all Left
s in a list to upper case:
>>>
(mapped._Left.mapped %~ toUpper) [Left "foo", Right "bar"]
[Left "FOO",Right "bar"]
over :: ASetter s t a b -> (a -> b) -> s -> t #
Getting fmap
in a roundabout way:
over
mapped
::Functor
f => (a -> b) -> f a -> f bover
mapped
=fmap
Applying a function to both components of a pair:
over
both
:: (a -> b) -> (a, a) -> (b, b)over
both
= \f t -> (f (fst t), f (snd t))
Using
as a replacement for over
_2
second
:
>>>
over _2 show (10,20)
(10,"20")
(+~) :: Num a => ASetter s t a a -> a -> s -> t infixr 4 #
Increment the target(s) of a numerically valued Lens
or Traversal
.
>>>
(a,b) & _1 +~ c
(a + c,b)
>>>
(a,b) & both +~ c
(a + c,b + c)
>>>
(1,2) & _2 +~ 1
(1,3)
>>>
[(a,b),(c,d)] & traverse.both +~ e
[(a + e,b + e),(c + e,d + e)]
(+~
) ::Num
a =>Lens'
s a -> a -> s -> s (+~
) ::Num
a =>Traversal'
s a -> a -> s -> s
Since: microlens-0.4.10
(-~) :: Num a => ASetter s t a a -> a -> s -> t infixr 4 #
Decrement the target(s) of a numerically valued Lens
, or Traversal
.
>>>
(a,b) & _1 -~ c
(a - c,b)
>>>
(a,b) & both -~ c
(a - c,b - c)
>>>
_1 -~ 2 $ (1,2)
(-1,2)
>>>
mapped.mapped -~ 1 $ [[4,5],[6,7]]
[[3,4],[5,6]]
(-~
) ::Num
a =>Lens'
s a -> a -> s -> s (-~
) ::Num
a =>Traversal'
s a -> a -> s -> s
Since: microlens-0.4.10
(<>~) :: Monoid a => ASetter s t a a -> a -> s -> t infixr 4 #
(<>~
) appends a value monoidally to the target.
>>>
("hello", "goodbye") & both <>~ " world!"
("hello world!", "goodbye world!")
Since: microlens-0.4.9
mapped :: Functor f => ASetter (f a) (f b) a b #
mapped
is a setter for everything contained in a functor. You can use it to map over lists, Maybe
, or even IO
(which is something you can't do with traversed
or each
).
Here mapped
is used to turn a value to all non-Nothing
values in a list:
>>>
[Just 3,Nothing,Just 5] & mapped.mapped .~ 0
[Just 0,Nothing,Just 0]
Keep in mind that while mapped
is a more powerful setter than each
, it can't be used as a getter! This won't work (and will fail with a type error):
[(1,2),(3,4),(5,6)]^..
mapped
.both
(<%~) :: LensLike ((,) b) s t a b -> (a -> b) -> s -> (b, t) infixr 4 #
This is a version of (%~
) which modifies the structure and returns it along with the new value:
>>>
(1, 2) & _1 <%~ negate
(-1, (-1, 2))
Simpler type signatures:
(<%~
) ::Lens
s t a b -> (a -> b) -> s -> (b, t) (<%~
) ::Monoid
b =>Traversal
s t a b -> (a -> b) -> s -> (b, t)
Since it does getting in addition to setting, you can't use it with ASetter
(but you can use it with lens and traversals).
rewriteOf :: ASetter a b a b -> (b -> Maybe a) -> a -> b #
→ See an example on GitHub.
Rewrite by applying a rule everywhere you can. Ensures that the rule cannot be applied anywhere in the result.
Usually transformOf
is more appropriate, but rewriteOf
can give better compositionality. Given two single transformations f
and g
, you can construct \a -> f a
which performs both rewrites until a fixed point.<|>
g a
Since: microlens-0.4.11
transformOf :: ASetter a b a b -> (b -> b) -> a -> b #
Transform every element by recursively applying a given ASetter
in a bottom-up manner.
Since: microlens-0.4.11
(^?!) :: HasCallStack => s -> Getting (Endo a) s a -> a infixl 8 #
forOf_ :: Functor f => Getting (Traversed r f) s a -> s -> (a -> f r) -> f () #
traverseOf_
with flipped arguments. Useful if the “loop body” is a lambda
or a do
block, or in some other cases – for instance, you can avoid
accidentally using for_
on a tuple or Either
by switching
to
. Or you can write custom loops like these:forOf_
each
forOf_
both
(a, b) $ \x -> ...forOf_
each
[1..10] $ \x -> ...forOf_
(each
._Right
) $ \x -> ...
has :: Getting Any s a -> s -> Bool #
has
checks whether a getter (any getter, including lenses, traversals, and folds) returns at least 1 value.
Checking whether a list is non-empty:
>>>
has each []
False
You can also use it with e.g. _Left
(and other 0-or-1 traversals) as a replacement for isNothing
, isJust
and other isConstructorName
functions:
>>>
has _Left (Left 1)
True
folding :: Foldable f => (s -> f a) -> SimpleFold s a #
lens :: (s -> a) -> (s -> b -> t) -> Lens s t a b #
lens
creates a Lens
from a getter and a setter. The resulting lens isn't the most effective one (because of having to traverse the structure twice when modifying), but it shouldn't matter much.
A (partial) lens for list indexing:
ix :: Int ->Lens'
[a] a ix i =lens
(!!
i) -- getter (\s b -> take i s ++ b : drop (i+1) s) -- setter
Usage:
>>> [1..9]^.
ix 3 4 >>> [1..9] & ix 3%~
negate [1,2,3,-4,5,6,7,8,9]
When getting, the setter is completely unused; when setting, the getter is unused. Both are used only when the value is being modified. For instance, here we define a lens for the 1st element of a list, but instead of a legitimate getter we use undefined
. Then we use the resulting lens for setting and it works, which proves that the getter wasn't used:
>>>
[1,2,3] & lens undefined (\s b -> b : tail s) .~ 10
[10,2,3]
traverseOf :: LensLike f s t a b -> (a -> f b) -> s -> f t #
Apply an action to all targets (like mapM
or traverse
):
>>>
traverseOf both readFile ("file1", "file2")
(<contents of file1>, <contents of file2>)
>>>
traverseOf _1 id (Just 1, 2)
Just (1, 2)>>>
traverseOf _1 id (Nothing, 2)
Nothing
You can also just apply the lens/traversal directly (but traverseOf
might be more readable):
>>>
both readFile ("file1", "file2")
(<contents of file1>, <contents of file2>)
If you don't need the result, use traverseOf_
.
forOf :: LensLike f s t a b -> s -> (a -> f b) -> f t #
traverseOf
with flipped arguments. Useful if the “loop body” is a lambda or
a do
block.
singular :: HasCallStack => Traversal s t a a -> Lens s t a a #
singular
turns a traversal into a lens that behaves like a single-element traversal:
>>>
[1,2,3] ^. singular each
1
>>>
[1,2,3] & singular each %~ negate
[-1,2,3]
If there is nothing to return, it'll throw an error:
>>>
[] ^. singular each
*** Exception: Lens.Micro.singular: empty traversal
However, it won't fail if you are merely setting the value:
>>>
[] & singular each %~ negate
failing :: Traversal s t a b -> Traversal s t a b -> Traversal s t a b infixl 5 #
failing
lets you chain traversals together; if the 1st traversal fails, the 2nd traversal will be used.
>>>
([1,2],[3]) & failing (_1.each) (_2.each) .~ 0
([0,0],[3])
>>>
([],[3]) & failing (_1.each) (_2.each) .~ 0
([],[0])
Note that the resulting traversal won't be valid unless either both traversals don't touch each others' elements, or both traversals return exactly the same results. To see an example of how failing
can generate invalid traversals, see this Stackoverflow question.
filtered :: (a -> Bool) -> Traversal' a a #
filtered
is a traversal that filters elements “passing” through it:
>>>
(1,2,3,4) ^.. each
[1,2,3,4]
>>>
(1,2,3,4) ^.. each . filtered even
[2,4]
It also can be used to modify elements selectively:
>>>
(1,2,3,4) & each . filtered even %~ (*100)
(1,200,3,400)
The implementation of filtered
is very simple. Consider this traversal, which always “traverses” just the value it's given:
id :: Traversal'
a a
id f s = f s
And this traversal, which traverses nothing (in other words, doesn't traverse the value it's given):
ignored ::Traversal'
a a ignored f s =pure
s
And now combine them into a traversal that conditionally traverses the value it's given, and you get filtered
:
filtered :: (a -> Bool) ->Traversal'
a a filtered p f s = if p s then f s elsepure
s
By the way, note that filtered
can generate illegal traversals – sometimes this can bite you. In particular, an optimisation that should be safe becomes unsafe. (To the best of my knowledge, this optimisation never happens automatically. If you just use filtered
to modify/view something, you're safe. If you don't define any traversals that use filtered
, you're safe too.)
Let's use evens
as an example:
evens =filtered
even
If evens
was a legal traversal, you'd be able to fuse several applications of evens
like this:
over
evens f.
over
evens g =over
evens (f.
g)
Unfortunately, in case of evens
this isn't a correct optimisation:
- the left-side variant applies
g
to all even numbers, and then appliesf
to all even numbers that are left afterf
(becausef
might've turned some even numbers into odd ones) - the right-side variant applies
f
andg
to all even numbers
Of course, when you are careful and know what you're doing, you won't try to make such an optimisation. However, if you export an illegal traversal created with filtered
and someone tries to use it, they might mistakenly assume that it's legal, do the optimisation, and silently get an incorrect result.
If you are using filtered
with some another traversal that doesn't overlap with -whatever the predicate checks-, the resulting traversal will be legal. For instance, here the predicate looks at the 1st element of a tuple, but the resulting traversal only gives you access to the 2nd:
pairedWithEvens ::Traversal
[(Int, a)] [(Int, b)] a b pairedWithEvens =each
.
filtered
(even
.
fst
).
_2
Since you can't do anything with the 1st components through this traversal, the following holds for any f
and g
:
over
pairedWithEvens f.
over
pairedWithEvens g =over
pairedWithEvens (f.
g)
both :: forall a b f. Applicative f => (a -> f b) -> (a, a) -> f (b, b) #
_head :: Cons s s a a => Traversal' s a #
_head
traverses the 1st element of something (usually a list, but can also be a Seq
, etc):
>>>
[1..5] ^? _head
Just 1
It can be used to modify too, as in this example where the 1st letter of a sentence is capitalised:
>>>
"mary had a little lamb." & _head %~ toTitle
"Mary had a little lamb."
The reason it's a traversal and not a lens is that there's nothing to traverse when the list is empty:
>>>
[] ^? _head
Nothing
This package only lets you use _head
on lists, but if you use microlens-ghc you get instances for ByteString
and Seq
, and if you use microlens-platform you additionally get instances for Text
and Vector
.
_tail :: Cons s s a a => Traversal' s s #
_tail
gives you access to the tail of a list (or Seq
, etc):
>>>
[1..5] ^? _tail
Just [2,3,4,5]
You can modify the tail as well:
>>>
[4,1,2,3] & _tail %~ reverse
[4,3,2,1]
Since lists are monoids, you can use _tail
with plain (^.
) (and then it'll return an empty list if you give it an empty list):
>>>
[1..5] ^. _tail
[2,3,4,5]
>>>
[] ^. _tail
[]
If you want to traverse each element of the tail, use _tail
with each
:
>>>
"I HATE CAPS." & _tail.each %~ toLower
"I hate caps."
This package only lets you use _tail
on lists, but if you use microlens-ghc you get instances for ByteString
and Seq
, and if you use microlens-platform you additionally get instances for Text
and Vector
.
_init :: Snoc s s a a => Traversal' s s #
_last :: Snoc s s a a => Traversal' s a #
mapAccumLOf :: LensLike (State acc) s t a b -> (acc -> a -> (acc, b)) -> acc -> s -> (acc, t) #
This generalizes mapAccumL
to an arbitrary Traversal
. (Note that it doesn't work on folds, only traversals.)
mapAccumL
≡mapAccumLOf
traverse
worded :: Traversal' String String #
Focus on the words
of a string.
>>>
"avoid success at all costs" & worded . _head %~ toUpper
"Avoid Success At All Costs"
This violates the traversal laws when whitespace is set or when the source has space at the ends or more than one contiguous space anywhere.
lined :: Traversal' String String #
Focus on the lines
of a string.
countAndMarkEmptyLines :: String -> State Int String countAndMarkEmptyLines s = runState (f s) 0 where f =traverseOf
(lined .filtered
null) $ \_ -> do modify' (+ 1) return "# Empty line"
This violates the traversal laws when newlines are set or when the source has more than one contiguous newline anywhere.
packedBytes :: IsByteString t => Lens' [Word8] t #
Treat a list of bytes as a strict or lazy ByteString
.
unpackedBytes :: IsByteString t => Lens' t [Word8] #
Treat a strict or lazy ByteString
as a list of bytes.
packedChars :: IsByteString t => Lens' String t #
unpackedChars :: IsByteString t => Lens' t String #
Treat a strict or lazy ByteString
as a String
. (Just as packedChars
, it will garble characters above 0xFF.)
chars :: IsByteString t => Traversal' t Char #
Traverse characters in a strict or lazy ByteString
(to traverse bytes instead of characters, use each
).
(+=) :: (MonadState s m, Num a) => ASetter s s a a -> a -> m () infix 4 #
use :: MonadState s m => Getting a s a -> m a #
use
is (^.
) (or view
) which implicitly operates on the state; for instance, if your state is a record containing a field foo
, you can write
x <- use
foo
to extract foo
from the state. In other words, use
is the same as gets
, but for getters instead of functions.
The implementation of use
is straightforward:
use
l =gets
(view
l)
If you need to extract something with a fold or traversal, you need preuse
.
(.=) :: MonadState s m => ASetter s s a b -> b -> m () infix 4 #
magnify :: Magnify m n b a => LensLike' (Magnified m c) a b -> m c -> n c infixr 2 #
This is an equivalent of local
which lets you apply a getter to your environment instead of merely applying a function (and it also lets you change the type of the environment).
local
:: (r -> r) ->Reader
r a ->Reader
r amagnify
:: Getter r x ->Reader
x a ->Reader
r a
magnify
works with Reader
/ ReaderT
, RWS
/ RWST
, and (->)
.
Here's an example of magnify
being used to work with a part of a bigger config. First, the types:
data URL = URL { _protocol :: Maybe String, _path :: String } data Config = Config { _base :: URL, ... } makeLenses ''URL makeLenses ''Config
Now, let's define a function which returns the base url:
getBase ::Reader
Config String getBase = do protocol <-fromMaybe
"https"<$>
view
(base.protocol) path <-view
(base.path) return (protocol ++ path)
With magnify
, we can factor out base
:
getBase =magnify
base $ do protocol <-fromMaybe
"https"<$>
view
protocol path <-view
path return (protocol ++ path)
This concludes the example.
Finally, you should know writing instances of Magnify
for your own types can be done as follows:
import Lens.Micro.Mtl.Internal type instanceMagnified
(MyReaderT r m) =Magnified
(ReaderT r m) instance Monad m =>Magnify
(MyReaderT r m) (MyReaderT t m) r t wheremagnify
l (MyReaderT m) = MyReaderT (magnify
l m)
zoom :: Zoom m n s t => LensLike' (Zoomed m c) t s -> m c -> n c infixr 2 #
When you're in a state monad, this function lets you operate on a part of your state. For instance, if your state was a record containing a position
field, after zooming position
would become your whole state (and when you modify it, the bigger structure would be modified as well).
(Your State
/ StateT
or RWS
/ RWST
can be anywhere in the stack, but you can't use zoom
with arbitrary MonadState
because it doesn't provide any methods to change the type of the state. See this issue for details.)
For the sake of the example, let's define some types first:
data Position = Position { _x, _y :: Int } data Player = Player { _position :: Position, ... } data Game = Game { _player :: Player, _obstacles :: [Position], ... } concat <$> mapM makeLenses [''Position, ''Player, ''Game]
Now, here's an action that moves the player north-east:
moveNE ::State
Game () moveNE = do player.position.x+=
1 player.position.y+=
1
With zoom
, you can use player.position
to focus just on a part of the state:
moveNE ::State
Game () moveNE = dozoom
(player.position) $ do x+=
1 y+=
1
You can just as well use it for retrieving things out of the state:
getCoords ::State
Game (Int, Int) getCoords =zoom
(player.position) ((,)<$>
use
x<*>
use
y)
Or more explicitly:
getCoords =zoom
(player.position) $ do x' <-use
x y' <-use
y return (x', y')
When you pass a traversal to zoom
, it'll work as a loop. For instance, here we move all obstacles:
moveObstaclesNE ::State
Game () moveObstaclesNE = dozoom
(obstacles.each
) $ do x+=
1 y+=
1
If the action returns a result, all results would be combined with <>
– the same way they're combined when ^.
is passed a traversal. In this example, moveObstaclesNE
returns a list of old coordinates of obstacles in addition to moving them:
moveObstaclesNE = do xys <-zoom
(obstacles.each
) $ do -- Get old coordinates. x' <-use
x y' <-use
y -- Update them. x.=
x' + 1 y.=
y' + 1 -- Return a single-element list with old coordinates. return [(x', y')] ...
Finally, you might need to write your own instances of Zoom
if you use newtype
d transformers in your monad stack. This can be done as follows:
import Lens.Micro.Mtl.Internal type instanceZoomed
(MyStateT s m) =Zoomed
(StateT s m) instance Monad m =>Zoom
(MyStateT s m) (MyStateT t m) s t wherezoom
l (MyStateT m) = MyStateT (zoom
l m)
(&~) :: s -> State s a -> s infixl 1 #
This can be used to chain lens operations using op=
syntax
rather than op~
syntax for simple non-type-changing cases.
>>> (10,20) & _1 .~ 30 & _2 .~ 40
(30,40)
>>>
(10,20) &~ do _1 .= 30; _2 .= 40
(30,40)
This does not support type-changing assignment, e.g.
>>>
(10,20) & _1 .~ "hello"
("hello",20)
assign :: MonadState s m => ASetter s s a b -> b -> m () #
A synonym for (.=
).
(?=) :: MonadState s m => ASetter s s a (Maybe b) -> b -> m () infix 4 #
(<~) :: MonadState s m => ASetter s s a b -> m b -> m () infixr 2 #
(%=) :: MonadState s m => ASetter s s a b -> (a -> b) -> m () infix 4 #
Modify state by applying a function to a part of the state. An example:
>>>
execState (do _1 %= (+1); _2 %= reverse) (1,"hello")
(2,"olleh")
Implementation:
l%=
f =modify
(l%~
f)
If you also want to get the value before/after the modification, use (<<%=
)/(<%=
).
There are a few specialised versions of (%=
) which mimic C operators:
modifying :: MonadState s m => ASetter s s a b -> (a -> b) -> m () #
A synonym for (%=
).
(-=) :: (MonadState s m, Num a) => ASetter s s a a -> a -> m () infix 4 #
(*=) :: (MonadState s m, Num a) => ASetter s s a a -> a -> m () infix 4 #
(//=) :: (MonadState s m, Fractional a) => ASetter s s a a -> a -> m () infix 4 #
(<%=) :: MonadState s m => LensLike ((,) b) s s a b -> (a -> b) -> m b infix 4 #
(<<%=) :: MonadState s m => LensLike ((,) a) s s a b -> (a -> b) -> m a infix 4 #
(<<.=) :: MonadState s m => LensLike ((,) a) s s a b -> b -> m a infix 4 #
(<.=) :: MonadState s m => LensLike ((,) b) s s a b -> b -> m b infix 4 #
makeLenses :: Name -> DecsQ #
Generate lenses for a data type or a newtype.
To use it, you have to enable Template Haskell first:
{-# LANGUAGE TemplateHaskell #-}
Then, after declaring the datatype (let's say Foo
), add makeLenses ''Foo
on a separate line (if you do it before the type is declared, you'll get a “not in scope” error – see the section at the top of this page):
data Foo = Foo {
_x :: Int,
_y :: Bool }
makeLenses
''Foo
This would generate the following lenses, which can be used to access the fields of Foo
:
x ::Lens'
Foo Int x f foo = (\x' -> foo {_x = x'})<$>
f (_x foo) y ::Lens'
Foo Bool y f foo = (\y' -> foo {_y = y'})<$>
f (_y foo)
(If you don't want a lens to be generated for some field, don't prefix it with “_”.)
If you want to create lenses for many types, you can do it all in one place like this (of course, instead you just can use makeLenses
several times if you feel it would be more readable):
data Foo = ... data Bar = ... data Quux = ...concat
<$>
mapM
makeLenses
[''Foo, ''Bar, ''Quux]
When the data type has type parameters, it's possible for a lens to do a polymorphic update – i.e. change the type of the thing along with changing the type of the field. For instance, with this type
data Foo a = Foo { _x :: a, _y :: Bool }
the following lenses would be generated:
x ::Lens
(Foo a) (Foo b) a b y ::Lens'
(Foo a) Bool
However, when there are several fields using the same type parameter, type-changing updates are no longer possible:
data Foo a = Foo { _x :: a, _y :: a }
generates
x ::Lens'
(Foo a) a y ::Lens'
(Foo a) a
Finally, when the type has several constructors, some of fields may not be always present – for those, a Traversal
is generated instead. For instance, in this example y
can be present or absent:
data FooBar = Foo { _x :: Int, _y :: Bool } | Bar { _x :: Int }
and the following accessors would be generated:
x ::Lens'
FooBar Int y ::Traversal'
FooBar Bool
So, to get _y
, you'd have to either use (^?
) if you're not sure it's there, or (^?!
) if you're absolutely sure (and if you're wrong, you'll get an exception). Setting and updating _y
can be done as usual.
makeLensesFor :: [(String, String)] -> Name -> DecsQ #
Like makeLenses
, but lets you choose your own names for lenses:
data Foo = Foo {foo :: Int, bar :: Bool}
makeLensesFor
[("foo", "fooLens"), ("bar", "_bar")] ''Foo
would create lenses called fooLens
and _bar
. This is useful, for instance, when you don't want to prefix your fields with underscores and want to prefix lenses with underscores instead.
If you give the same name to different fields, it will generate a Traversal
instead:
data Foo = Foo {slot1, slot2, slot3 :: Int}
makeLensesFor
[("slot1", "slots"),
("slot2", "slots"),
("slot3", "slots")] ''Foo
would generate
slots ::Traversal'
Foo Int slots f foo = Foo<$>
f (slot1 foo)<*>
f (slot2 foo)<*>
f (slot3 foo)
makeLensesWith :: LensRules -> Name -> DecsQ #
Generate lenses with custom parameters.
To see what exactly you can customise, look at the “Configuring lens rules” section. Usually you would build upon the lensRules
configuration, which is used by makeLenses
:
makeLenses
=makeLensesWith
lensRules
Here's an example of generating lenses that would use lazy patterns:
data Foo = Foo {_x, _y :: Int}makeLensesWith
(lensRules
&
generateLazyPatterns
.~
True) ''Foo
When there are several modifications to the rules, the code looks nicer when you use flip
:
flip
makeLensesWith
''Foo $lensRules
&
generateLazyPatterns
.~
True&
generateSignatures
.~
False
makeFields :: Name -> DecsQ #
Generate overloaded lenses.
This lets you deal with several data types having same fields. For instance, let's say you have Foo
and Bar
, and both have a field named x
. To avoid those fields clashing, you would have to use prefixes:
data Foo a = Foo { fooX :: Int, fooY :: a } data Bar = Bar { barX :: Char }
However, if you use makeFields
on both Foo
and Bar
now, it would generate lenses called x
and y
– and x
would be able to access both fooX
and barX
! This is done by generating a separate class for each field, and making relevant types instances of that class:
class HasX s a | s -> a where x ::Lens'
s a instance HasX (Foo a) Int where x ::Lens'
(Foo a) Int x = ... instance HasX Bar Char where x ::Lens'
Bar Char x = ... class HasY s a | s -> a where y ::Lens'
s a instance HasY (Foo a) a where y ::Lens'
(Foo a) a y = ...
(There's a minor drawback, though: you can't perform type-changing updates with these lenses.)
If you only want to make lenses for some fields, you can prefix them with underscores – the rest would be untouched. If no fields are prefixed with underscores, lenses would be created for all fields.
The prefix must be the same as the name of the name of the data type (not the constructor). If you don't like this behavior, use
– it allows any prefix (and even different prefixes).makeLensesWith
abbreviatedFields
If you want to use makeFields
on types declared in different modules, you can do it, but then you would have to export the Has*
classes from one of the modules – makeFields
creates a class if it's not in scope yet, so the class must be in scope or else there would be duplicate classes and you would get an “Ambiguous occurrence” error.
Finally, makeFields
is implemented as
, so you can build on makeLensesWith
camelCaseFields
camelCaseFields
if you want to customise behavior of makeFields
.
makeClassy :: Name -> DecsQ #
Generate overloaded lenses without ad-hoc classes; useful when there's a collection of fields that you want to make common for several types.
Like makeFields
, each lens is a member of a class. However, the classes are per-type and not per-field. Let's take the following type:
data Person = Person { _name :: String, _age :: Double }
makeClassy
would generate a single class with 3 methods:
class HasPerson c where person :: Lens' c Person age :: Lens' c Double age = person.age name :: Lens' c String name = person.name
And an instance:
instance HasPerson Person where person = id name = ... age = ...
So, you can use name
and age
to refer to the _name
and _age
fields, as usual. However, the extra lens – person
– allows you to do a kind of subtyping. Let's say that there's a type called Worker
and every worker has the same fields that a person has, but also a job
. If you were using makeFields
, you'd do the following:
data Worker = Worker { _workerName :: String, _workerAge :: Double, _workerJob :: String }
However, with makeClassy
you can say “every worker is a person” in a more principled way:
data Worker = Worker { _workerPerson :: Person, _job :: String } makeClassy ''Worker instance HasPerson Worker where person = workerPerson
Now you can use age
and name
to access name/age of a Worker
, but you also can use person
to “downgrade” a Worker
to a Person
(and e.g. apply some Person
-specific function to it).
Unlike makeFields
, makeClassy
doesn't make use of prefixes. _workerPerson
could've just as well been named _foobar
.
makeClassy
is implemented as
, so you can build on makeLensesWith
classyRules
classyRules
if you want to customise behavior of makeClassy
.
simpleLenses :: Lens' LensRules Bool #
Generate simple (monomorphic) lenses even when type-changing lenses are possible – i.e. Lens'
instead of Lens
and Traversal'
instead of Traversal
. Just in case, here's an example of a situation when type-changing lenses would be normally generated:
data Foo a = Foo { _foo :: a }
Generated lens:
foo :: Lens
(Foo a) (Foo b) a b
Generated lens with simpleLenses
turned on:
foo :: Lens'
(Foo a) a
This option is disabled by default.
generateSignatures :: Lens' LensRules Bool #
Supply type signatures for the generated lenses.
This option is enabled by default. Disable it if you want to write the signature by yourself – for instance, if the signature should be more restricted, or if you want to write haddocks for the lens (as haddocks are attached to the signature and not to the definition).
generateUpdateableOptics :: Lens' LensRules Bool #
Generate “updateable” optics. When turned off, SimpleFold
s will be generated instead of Traversal
s and SimpleGetter
s will be generated instead of Lens
es.
This option is enabled by default. Disabling it can be useful for types with invariants (also known as “types with smart constructors”) – if you generate updateable optics, anyone would be able to use them to break your invariants.
generateLazyPatterns :: Lens' LensRules Bool #
Generate lenses using lazy pattern matches. This can allow fields of an undefined value to be initialized with lenses:
data Foo = Foo {_x :: Int, _y :: Bool} deriving ShowmakeLensesWith
(lensRules
&
generateLazyPatterns
.~
True) ''Foo
>>>undefined
&
x.~
8&
y.~
True Foo {_x = 8, _y = True}
(Without generateLazyPatterns
, the result would be just undefined
.)
This option is disabled by default. The downside of enabling it is that it can lead to space-leaks and code-size/compile-time increases when lenses are generated for large records.
When you have a lazy lens, you can get a strict lens from it by composing with ($!
):
strictLens = ($!
) . lazyLens
lensField :: Lens' LensRules (Name -> [Name] -> Name -> [DefName]) #
This lets you choose which fields would have lenses generated for them and how would those lenses be called. To do that, you provide a function that would take a field name and output a list (possibly empty) of lenses that should be generated for that field.
Here's the full type of the function you have to provide:
Name
-> -- The datatype lenses are being generated for [Name
] -> -- A list of all fields of the datatypeName
-> -- The current field [DefName
] -- A list of lens names
Most of the time you won't need the first 2 parameters, but sometimes they are useful – for instance, the list of all fields would be useful if you wanted to implement a slightly more complicated rule like “if some fields are prefixed with underscores, generate lenses for them, but if no fields are prefixed with underscores, generate lenses for all fields”.
As an example, here's a function used by default. It strips “_” off the field name, lowercases the next character after “_”, and skips the field entirely if it doesn't start with “_”:
\_ _ n -> casenameBase
n of '_':x:xs -> [TopName
(mkName
(toLower
x : xs))] _ -> []
You can also generate classes (i.e. what makeFields
does) by using
instead of MethodName
className lensName
.TopName
lensName
lensClass :: Lens' LensRules (Name -> Maybe (Name, Name)) #
This lets you choose whether a class would be generated for the type itself (like makeClassy
does). If so, you can choose the name of the class and the name of the type-specific lens.
For makeLenses
and makeFields
this is just const Nothing
. For makeClassy
this function is defined like this:
\n -> casenameBase
n of x:xs -> Just (mkName
(Has ++ x:xs),mkName
(toLower
x : xs)) [] -> Nothing
createClass :: Lens' LensRules Bool #
Decide whether generation of classes is allowed at all.
If this is disabled, neither makeFields
nor makeClassy
would work, regardless of values of lensField
or lensClass
. On the other hand, if lensField
and lensClass
don't generate any classes, enabling this won't have any effect.
Lens rules used by default (i.e. in makeLenses
):
generateSignatures
is turned ongenerateUpdateableOptics
is turned ongenerateLazyPatterns
is turned offsimpleLenses
is turned offlensField
strips “_” off the field name, lowercases the next character after “_”, and skips the field entirely if it doesn't start with “_” (you can see how it's implemented in the docs forlensField
)lensClass
isn't used (i.e. defined asconst Nothing
)
A modification of lensRules
used by makeLensesFor
(the only difference is that a simple lookup function is used for lensField
).
camelCaseFields :: LensRules #
Lens rules used by makeFields
:
generateSignatures
is turned ongenerateUpdateableOptics
is turned ongenerateLazyPatterns
is turned offsimpleLenses
is turned on (unlike inlensRules
)lensField
is more complicated – it takes fields which are prefixed with the name of the type they belong to (e.g. “fooFieldName” for “Foo”), strips that prefix, and generates a class called “HasFieldName” with a single method called “fieldName”. If some fields are prefixed with underscores, underscores would be stripped too, but then fields without underscores won't have any lenses generated for them. Also note that e.g. “foolish” won't have a lens called “lish” generated for it – the prefix must be followed by a capital letter (or else it wouldn't be camel case).lensClass
isn't used (i.e. defined asconst Nothing
)
abbreviatedFields :: LensRules #
Like standard rules used by makeFields
, but doesn't put any restrictions on the prefix. I.e. if you have fields called
_fooBarBaz
_someX
someY
then the generated lenses would be called barBaz
and x
.
Lens rules used by makeClassy
:
generateSignatures
is turned ongenerateUpdateableOptics
is turned ongenerateLazyPatterns
is turned offsimpleLenses
is turned on (unlike inlensRules
)lensField
is the same as inlensRules
lensClass
just adds “Has” to the name of the type (so for “Person” the generated class would be called “HasPerson” and the type-specific lens in that class would be called “person”)