mini-1.5.3.0: Minimal essentials
Safe HaskellSafe-Inferred
LanguageHaskell2010

Mini.Optics.Lens

Description

Compose polymorphic record updates with van Laarhoven lenses

Synopsis
  • type Lens s t a b = forall f. Functor f => (a -> f b) -> s -> f t
  • lens :: (s -> a) -> (s -> b -> t) -> Lens s t a b
  • view :: Lens s t a b -> s -> a
  • over :: Lens s t a b -> (a -> b) -> s -> t
  • set :: Lens s t a b -> b -> s -> t

Type

type Lens s t a b = forall f. Functor f => (a -> f b) -> s -> f t Source #

A reference updating structures from s to t and fields from a to b

Construction

lens :: (s -> a) -> (s -> b -> t) -> Lens s t a b Source #

Make a lens from a getter and a setter

Operations

view :: Lens s t a b -> s -> a Source #

Fetch the field referenced by a lens from a structure

over :: Lens s t a b -> (a -> b) -> s -> t Source #

Update the field referenced by a lens with an operation on a structure

set :: Lens s t a b -> b -> s -> t Source #

Overwrite the field referenced by a lens with a value on a structure

Tutorial

Record syntax is excellent for modeling data. But modifying records can be cumbersome, especially when nested. Below we define some data and compare common ways of modifying it, illustrating the need for something better. Then we satisfy that need with lenses, showing how to use them.

Lenses essentially take the place of accessor functions. The accessor names of the records below start with an underscore; this is to avoid name clashing when creating the corresponding lenses.

data Parcel = Parcel
  { _weight :: Int
  , _size :: Size
  }

data Size = Size
  { _length :: Int
  , _width :: Int
  , _height :: Int
  }

Let's create a function that doubles the width of a parcel and sets its weight to 500. First using record syntax, then pattern matching.

foo :: Parcel -> Parcel
foo p =
  let s = _size p
      w = _width s
   in p
        { _weight = 500
        , _size = s{_width = w * 2}
        }

foo' :: Parcel -> Parcel
foo' (Parcel _ (Size l w h)) =
  Parcel
    500
    (Size l (w * 2) h)

Record syntax lets us specify only the fields we wish to modify but we have to use the accessors repeatedly. Pattern matching looks cleaner but we have to unpack and repack all the data. It's not hard to imagine how unwieldy it would be with heavily nested records no matter which way we choose.

Lenses provide a concise, declarative, composable way to inspect and modify records. The downside is that some initial boilerplate code is required to create the lenses.

weight :: Lens Parcel Parcel Int Int
weight = lens _weight $ \s b -> s{_weight = b}

size :: Lens Parcel Parcel Size Size
size = lens _size $ \s b -> s{_size = b}

width :: Lens Size Size Int Int
width = lens _width $ \s b -> s{_width = b}

Each lens can be used with view, over, and set to inspect or modify a field, letting us complete our task with ease.

foo'' :: Parcel -> Parcel
foo'' =
  over (size . width) (* 2)
    . set weight 500

Note the reversed ordering of the composed lenses. Inspecting is similar to using regular accessors, but again, composed in reverse.

bar :: Parcel -> Int
bar = view (size . width)

This covers the most typical use case, where the lenses preserve the structure of the records (e.g. weight is a Lens from Parcel to Parcel, from Int to Int). Creating lenses that change the structure is left as an exercise to the reader.