README.md

# LiveStreamAsync

Extends LiveView with `stream_async/4` macro.

## Installation

The package can be installed by adding `live_stream_async` to your list of dependencies in `mix.exs`:

```elixir
def deps do
  [
    {:live_stream_async, "~> 0.1.0", runtime: false}
  ]
end
```

The docs can be found at <https://hexdocs.pm/live_stream_async>.

## New asynchronous operations in LiveView v.20

  New release of [LiveView library - v 0.20](https://hexdocs.pm/phoenix_live_view/Phoenix.LiveView.html) - introduced built-in functions for [asynchronous work](https://hexdocs.pm/phoenix_live_view/0.20.14/Phoenix.LiveView.html#module-async-operations).  It's a perfect solution to deliver a snappy user experience by delegating some time-consuming tasks (ex. fetching from external services) to background jobs without blocking the UI or event handlers. New operations include:

  - `assign_sync/3`: a straight forward way to load the results asynchronously into socket assigns.
  - `start_async/4`: allows lower level control of asynchronous operations with `handle_async` callbacks.
  - `<.async_result ...>` - function component to handle the asynchronous operation state on the UI side (for success, loading and errors).

  ## Async streams

  Streams in LiveView allow working with large collections without keeping them on the server. In case you want to work with `streams` assigns asynchronously you may need to resort to low level control functions.

  This library provides a convenient macro `stream_async/4` that auto-generates all the necessary boilerplate behind the scenes and injects it into your LiveView module.

  ## Usage

  ### Extending LiveView

  Extend your live view module with `use LiveAsyncStream` and you can leverage the `stream_async/4` macro:

  ```elixir
  use MyAppWeb, :live_view
  use LiveAsyncStream

  def mount(%{"location" => location}, _, socket) do
  {:ok,
  socket
  |> stream_async(:hotels, fn -> Hotels.fetch!(location) end, reset: true)
  }
  end
  ```

  ### Accessing the result

  The `<.async_result ...>` component is designed to work with the `%Phoenix.LiveView.AsyncResult{}` structs. The struct is passed via "`assign={}`" attribute of the component. The component's inner block receives the `@streams` assign key through `:let={}` attribute. Example:
  ```elixir
    def render(assigns) do
      ~H"""
      <.async_result :let={stream_key} assign={@hotels}>
      <:loading>Loading hotels...</:loading>
      <:failed :let={_failure}>There was an error loading the hotels. Please try again later.</:failed>
        <ul id="hotels_stream" phx-update="stream">
          <li :for={{id, hotel} <- @streams[stream_key]} id={id}>
          <%= hotel.name %>
          </li>
        </ul>
      </.async_result>
    """
    end
  ```