defmodule LivebookWeb.Output do use Phoenix.Component import LivebookWeb.Helpers alias Phoenix.LiveView.JS alias LivebookWeb.Output @doc """ Renders a list of cell outputs. """ def outputs(assigns) do ~H""" <%= for {idx, output} <- Enum.reverse(@outputs) do %>
<%= render_output(output, %{ id: "output-#{idx}", socket: @socket, session_id: @session_id, input_values: @input_values, client_id: @client_id }) %>
<% end %> """ end defp border?({:stdout, _text}), do: true defp border?({:text, _text}), do: true defp border?({:error, _message}), do: true defp border?(_output), do: false defp wrapper?({:frame, _outputs, _info}), do: true defp wrapper?({:tabs, _tabs, _info}), do: true defp wrapper?({:grid, _tabs, _info}), do: true defp wrapper?(_output), do: false defp render_output({:stdout, text}, %{id: id}) do text = if(text == :__pruned__, do: nil, else: text) live_component(Output.StdoutComponent, id: id, text: text) end defp render_output({:text, text}, %{id: id}) do assigns = %{id: id, text: text} ~H""" """ end defp render_output({:markdown, markdown}, %{id: id, session_id: session_id}) do live_component(Output.MarkdownComponent, id: id, session_id: session_id, content: markdown ) end defp render_output({:image, content, mime_type}, %{id: id}) do assigns = %{id: id, content: content, mime_type: mime_type} ~H""" """ end defp render_output({:js, js_info}, %{id: id, session_id: session_id, client_id: client_id}) do live_component(LivebookWeb.JSViewComponent, id: id, js_view: js_info.js_view, session_id: session_id, client_id: client_id, timeout_message: "Output data no longer available, please reevaluate this cell" ) end defp render_output({:frame, outputs, _info}, %{ id: id, session_id: session_id, input_values: input_values, client_id: client_id }) do live_component(Output.FrameComponent, id: id, outputs: outputs, session_id: session_id, input_values: input_values, client_id: client_id ) end defp render_output({:tabs, outputs, info}, %{ id: id, socket: socket, session_id: session_id, input_values: input_values, client_id: client_id }) do {labels, active_idx} = if info == :__pruned__ do {[], nil} else labels = Enum.zip_with(info.labels, outputs, fn label, {output_idx, _} -> {output_idx, label} end) active_idx = get_in(outputs, [Access.at(0), Access.elem(0)]) {labels, active_idx} end assigns = %{ id: id, active_idx: active_idx, labels: labels, outputs: outputs, socket: socket, session_id: session_id, input_values: input_values, client_id: client_id } # After pruning we don't render labels and we render only those # outputs that are kept during pruning ~H"""
<%= for {output_idx, label} <- @labels do %> <% end %>
<%= for {output_idx, output} <- @outputs do %> <% # We use data-keep-attribute, because we know active_idx only on the first render %>
<.outputs outputs={[{output_idx, output}]} dom_id_map={%{}} socket={@socket} session_id={@session_id} input_values={@input_values} client_id={@client_id} />
<% end %>
""" end defp render_output({:grid, outputs, info}, %{ id: id, session_id: session_id, socket: socket, input_values: input_values, client_id: client_id }) do style = if info == :__pruned__ do nil else columns = info[:columns] || 1 "grid-template-columns: repeat(#{columns}, minmax(0, 1fr));" end assigns = %{ id: id, style: style, outputs: outputs, socket: socket, session_id: session_id, input_values: input_values, client_id: client_id } ~H"""
<%= for {output_idx, output} <- @outputs do %>
<.outputs outputs={[{output_idx, output}]} dom_id_map={%{}} socket={@socket} session_id={@session_id} input_values={@input_values} client_id={@client_id} />
<% end %>
""" end defp render_output({:input, attrs}, %{id: id, input_values: input_values, client_id: client_id}) do live_component(Output.InputComponent, id: id, attrs: attrs, input_values: input_values, client_id: client_id ) end defp render_output({:control, attrs}, %{ id: id, input_values: input_values, client_id: client_id }) do live_component(Output.ControlComponent, id: id, attrs: attrs, input_values: input_values, client_id: client_id ) end defp render_output({:error, formatted}, %{}) do assigns = %{message: formatted} ~H""" """ end # TODO: remove on Livebook v0.7 defp render_output(output, %{}) when elem(output, 0) in [ :vega_lite_static, :vega_lite_dynamic, :table_dynamic, :frame_dynamic ] do render_error_message(""" Legacy output format: #{inspect(output)}. Please update Kino to the latest version. """) end defp render_output(output, %{}) do render_error_message(""" Unknown output format: #{inspect(output)}. If you're using Kino, you may want to update Kino and Livebook to the latest version. """) end defp render_error_message(message) do assigns = %{message: message} ~H""" """ end end