defprotocol Focusable do
@doc "View the data that an optic focuses on."
def view(optic, structure)
@doc "Modify the data that an optic focuses on."
def over(optic, structure, f)
@doc "Set the data that an optic focuses on."
def set(optic, structure, value)
end
defmodule Focus do
alias Focus.Types
@moduledoc "Common functions usable by lenses, prisms, and traversals."
@doc """
Wrapper around Focusable.view/2
Arguments can be passed in with either the lens first and data structure second or vice versa.
Passing the data structure first allows Focus.view/2 to fit neatly in pipeline operations.
## Examples
iex> marge = %{
...> name: "Marge",
...> address: %{
...> street: "123 Fake St.",
...> city: "Springfield"
...> }
...> }
iex> address_lens = Lens.make_lens(:address)
iex> address_lens
...> |> Focus.view(marge)
%{street: "123 Fake St.", city: "Springfield"}
iex> marge
...> |> Focus.view(address_lens)
%{street: "123 Fake St.", city: "Springfield"}
"""
@spec view(Types.optic() | Types.traversable(), Types.traversable() | Types.optic()) ::
any | nil
def view(%{get: _, put: _} = optic, structure), do: Focusable.view(optic, structure)
def view(structure, %{get: _, put: _} = optic), do: Focusable.view(optic, structure)
@doc """
Wrapper around Focusable.over/3
Arguments can be passed in with either the lens first and data structure second or vice versa.
Passing the data structure first allows Focus.over/3 to fit neatly in pipeline operations.
## Examples
iex> marge = %{
...> name: "Marge",
...> address: %{
...> street: "123 Fake St.",
...> city: "Springfield"
...> }
...> }
iex> name_lens = Lens.make_lens(:name)
iex> name_lens
...> |> Focus.over(marge, &String.upcase/1)
%{
name: "MARGE",
address: %{
street: "123 Fake St.",
city: "Springfield"
}
}
iex> marge
...> |> Focus.over(name_lens, &String.upcase/1)
%{
name: "MARGE",
address: %{
street: "123 Fake St.",
city: "Springfield"
}
}
"""
@spec over(
Types.optic() | Types.traversable(),
Types.traversable() | Types.optic(),
(any -> any)
) :: Types.traversable()
def over(%{get: _, put: _} = optic, structure, f), do: Focusable.over(optic, structure, f)
def over(structure, %{get: _, put: _} = optic, f), do: Focusable.over(optic, structure, f)
@doc """
Wrapper around Focusable.set/3
Arguments can be passed in with either the lens first and data structure second or vice versa.
Passing the data structure first allows Focus.set/3 to fit neatly in pipeline operations.
## Examples
iex> marge = %{
...> name: "Marge",
...> address: %{
...> street: "123 Fake St.",
...> city: "Springfield"
...> }
...> }
iex> name_lens = Lens.make_lens(:name)
iex> name_lens
...> |> Focus.set(marge, "Marjorie")
%{
name: "Marjorie",
address: %{
street: "123 Fake St.",
city: "Springfield"
}
}
iex> marge
...> |> Focus.set(name_lens, "Marjorie")
%{
name: "Marjorie",
address: %{
street: "123 Fake St.",
city: "Springfield"
}
}
"""
@spec set(Types.traversable() | Types.optic(), Types.optic() | Types.traversable(), any) ::
Types.traversable()
def set(%{get: _, put: _} = optic, structure, v), do: Focusable.set(optic, structure, v)
def set(structure, %{get: _, put: _} = optic, v), do: Focusable.set(optic, structure, v)
@doc """
Compose with most general lens on the left
## Examples
iex> marge = %{
...> name: "Marge",
...> address: %{
...> street: "123 Fake St.",
...> city: "Springfield"
...> }
...> }
iex> address_lens = Lens.make_lens(:address)
iex> street_lens = Lens.make_lens(:street)
iex> composed = Focus.compose(address_lens, street_lens)
iex> Focus.view(composed, marge)
"123 Fake St."
"""
@spec compose(Types.optic(), Types.optic()) :: Types.optic()
def compose(%{get: get_x, put: set_x}, %{get: get_y, put: set_y}) do
%Lens{
get: fn s ->
case get_x.(s) do
{:error, {:lens, :bad_path}} ->
{:error, {:lens, :bad_path}}
x ->
get_y.(x)
end
end,
put: fn s ->
fn f ->
case get_x.(s) do
{:error, {:lens, :bad_path}} ->
{:error, {:lens, :bad_path}}
x ->
set_x.(s).(set_y.(x).(f))
end
end
end
}
end
@doc """
Infix lens composition
## Examples
iex> import Focus
iex> marge = %{name: "Marge", address: %{
...> local: %{number: 123, street: "Fake St."},
...> city: "Springfield"}
...> }
iex> address_lens = Lens.make_lens(:address)
iex> local_lens = Lens.make_lens(:local)
iex> street_lens = Lens.make_lens(:street)
iex> address_lens ~> local_lens ~> street_lens |> Focus.view(marge)
"Fake St."
"""
@spec Types.optic() ~> Types.optic() :: Types.optic()
def x ~> y do
compose(x, y)
end
@doc """
Compose a pair of lenses to operate at the same level as one another.
Calling Focus.view/2, Focus.over/3, or Focus.set/3 on an alongside composed
pair returns a two-element tuple of the result.
## Examples
iex> nums = [1,2,3,4,5,6]
iex> Focus.alongside(Lens.idx(0), Lens.idx(3))
...> |> Focus.view(nums)
{1, 4}
iex> bart = %{name: "Bart", parents: {"Homer", "Marge"}, age: 10}
iex> Focus.alongside(Lens.make_lens(:name), Lens.make_lens(:age))
...> |> Focus.view(bart)
{"Bart", 10}
"""
@spec alongside(Types.optic(), Types.optic()) :: Types.optic()
def alongside(%{get: get_x, put: set_x}, %{get: get_y, put: set_y}) do
%Lens{
get: fn s ->
{get_x.(s), get_y.(s)}
end,
put: fn s ->
fn f ->
{set_x.(s).(f), set_y.(s).(f)}
end
end
}
end
@doc """
Given a list of lenses and a structure, apply Focus.view/2 for each lens
to the structure.
## Examples
iex> homer = %{
...> name: "Homer",
...> job: "Nuclear Safety Inspector",
...> children: ["Bart", "Lisa", "Maggie"]
...> }
iex> lenses = Lens.make_lenses(homer)
iex> [lenses.name, lenses.children]
...> |> Focus.view_list(homer)
["Homer", ["Bart", "Lisa", "Maggie"]]
"""
@spec view_list(list(Types.optic()), Types.traversable()) :: [any]
def view_list(lenses, structure) when is_list(lenses) do
for lens <- lenses do
Focus.view(lens, structure)
end
end
@doc """
Check whether an optic's target is present in a data structure.
## Examples
iex> first_elem = Lens.idx(1)
iex> first_elem |> Focus.has([0])
false
iex> name = Lens.make_lens(:name)
iex> name |> Focus.has(%{name: "Homer"})
true
"""
@spec has(Types.optic(), Types.traversable()) :: boolean
def has(optic, structure) do
case Focus.view(optic, structure) do
nil -> false
{:error, _} -> false
_ -> true
end
end
@doc """
Check whether an optic's target is not present in a data structure.
## Examples
iex> first_elem = Lens.idx(1)
iex> first_elem |> Focus.hasnt([0])
true
iex> name = Lens.make_lens(:name)
iex> name |> Focus.hasnt(%{name: "Homer"})
false
"""
@spec hasnt(Types.optic(), Types.traversable()) :: boolean
def hasnt(optic, structure), do: !has(optic, structure)
@doc """
Partially apply a lens to Focus.over/3, fixing the lens argument and
returning a function that takes a Types.traversable and an update function.
## Examples
iex> upcase_name = Lens.make_lens(:name)
...> |> Focus.fix_over(&String.upcase/1)
iex> %{name: "Bart", parents: {"Homer", "Marge"}}
...> |> upcase_name.()
%{name: "BART", parents: {"Homer", "Marge"}}
iex> fst = Lens.idx(0)
iex> states = [:maryland, :texas, :illinois]
iex> Focus.over(fst, states, &String.upcase(Atom.to_string(&1)))
["MARYLAND", :texas, :illinois]
"""
@spec fix_over(Types.optic(), (any -> any)) :: (Types.traversable() -> Types.traversable())
def fix_over(%{get: _, put: _} = lens, f \\ fn x -> x end) when is_function(f) do
fn structure ->
Focus.over(lens, structure, f)
end
end
@doc """
Partially apply a lens to Focus.set/3, fixing the optic argument and
returning a function that takes a Types.traversable and a new value.
## Examples
iex> name_setter = Lens.make_lens(:name)
...> |> Focus.fix_set
iex> %{name: "Bart", parents: {"Homer", "Marge"}}
...> |> name_setter.("Lisa")
%{name: "Lisa", parents: {"Homer", "Marge"}}
iex> fst = Lens.idx(0)
iex> states = [:maryland, :texas, :illinois]
iex> Focus.over(fst, states, &String.upcase(Atom.to_string(&1)))
["MARYLAND", :texas, :illinois]
"""
@spec fix_set(Types.optic()) :: (Types.traversable(), any -> Types.traversable())
def fix_set(%{get: _, put: _} = lens) do
fn structure, val ->
Focus.set(lens, structure, val)
end
end
@doc """
Fix Focus.view/2 on a given optic. This partially applies Focus.view/2 with the given
optic and returns a function that takes a Types.traversable structure.
## Examples
iex> view_name = Lens.make_lens(:name)
...> |> Focus.fix_view
iex> homer = %{name: "Homer"}
iex> view_name.(homer)
"Homer"
iex> [homer, %{name: "Marge"}, %{name: "Bart"}]
...> |> Enum.map(&view_name.(&1))
["Homer", "Marge", "Bart"]
"""
@spec fix_view(Types.optic()) :: (Types.traversable() -> any)
def fix_view(%{get: _, put: _} = optic) do
fn structure ->
Focus.view(optic, structure)
end
end
end