2021-11-09 03:45:30 +08:00
|
|
|
defmodule LivebookWeb.Output do
|
|
|
|
use Phoenix.Component
|
|
|
|
|
|
|
|
@doc """
|
2021-11-26 01:43:42 +08:00
|
|
|
Renders a list of cell outputs.
|
2021-11-09 03:45:30 +08:00
|
|
|
"""
|
2021-11-26 01:43:42 +08:00
|
|
|
def outputs(assigns) do
|
|
|
|
~H"""
|
2022-01-17 03:37:00 +08:00
|
|
|
<%= for {idx, output} <- Enum.reverse(@outputs) do %>
|
|
|
|
<div class="max-w-full" id={"output-wrapper-#{idx}"}
|
|
|
|
data-element="output"
|
|
|
|
data-border={border?(output)}
|
|
|
|
data-wrapper={wrapper?(output)}>
|
|
|
|
<%= render_output(output, %{
|
|
|
|
id: "output-#{idx}",
|
|
|
|
socket: @socket,
|
|
|
|
session_id: @session_id,
|
|
|
|
runtime: @runtime,
|
|
|
|
cell_validity_status: @cell_validity_status,
|
|
|
|
input_values: @input_values
|
|
|
|
}) %>
|
|
|
|
</div>
|
|
|
|
<% end %>
|
2021-11-26 01:43:42 +08:00
|
|
|
"""
|
|
|
|
end
|
|
|
|
|
2022-01-17 03:37:00 +08:00
|
|
|
defp border?({:stdout, _text}), do: true
|
|
|
|
defp border?({:text, _text}), do: true
|
|
|
|
defp border?({:error, _message, _type}), do: true
|
|
|
|
defp border?(_output), do: false
|
2021-11-26 01:43:42 +08:00
|
|
|
|
2022-01-17 03:37:00 +08:00
|
|
|
defp wrapper?({:frame, _outputs, _info}), do: true
|
|
|
|
defp wrapper?(_output), do: false
|
2021-11-26 01:43:42 +08:00
|
|
|
|
2022-01-17 03:37:00 +08:00
|
|
|
defp render_output({:stdout, text}, %{id: id}) do
|
|
|
|
text = if(text == :__pruned__, do: nil, else: text)
|
|
|
|
live_component(LivebookWeb.Output.StdoutComponent, id: id, text: text, follow: true)
|
2021-11-09 03:45:30 +08:00
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output({:text, text}, %{id: id}) do
|
2022-01-17 03:37:00 +08:00
|
|
|
assigns = %{id: id, text: text}
|
|
|
|
|
|
|
|
~H"""
|
|
|
|
<LivebookWeb.Output.TextComponent.render id={@id} content={@text} follow={false} />
|
|
|
|
"""
|
2021-11-09 03:45:30 +08:00
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output({:markdown, markdown}, %{id: id}) do
|
2021-11-09 03:45:30 +08:00
|
|
|
live_component(LivebookWeb.Output.MarkdownComponent, id: id, content: markdown)
|
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output({:image, content, mime_type}, %{id: id}) do
|
2022-01-17 03:37:00 +08:00
|
|
|
assigns = %{id: id, content: content, mime_type: mime_type}
|
|
|
|
|
|
|
|
~H"""
|
|
|
|
<LivebookWeb.Output.ImageComponent.render content={@content} mime_type={@mime_type} />
|
|
|
|
"""
|
2021-11-09 03:45:30 +08:00
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output({:vega_lite_static, spec}, %{id: id}) do
|
2021-11-09 03:45:30 +08:00
|
|
|
live_component(LivebookWeb.Output.VegaLiteStaticComponent, id: id, spec: spec)
|
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output({:vega_lite_dynamic, pid}, %{id: id, socket: socket}) do
|
2021-11-09 03:45:30 +08:00
|
|
|
live_render(socket, LivebookWeb.Output.VegaLiteDynamicLive,
|
|
|
|
id: id,
|
|
|
|
session: %{"id" => id, "pid" => pid}
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2022-01-06 23:31:26 +08:00
|
|
|
defp render_output({:js, info}, %{id: id, session_id: session_id}) do
|
|
|
|
live_component(LivebookWeb.Output.JSComponent, id: id, info: info, session_id: session_id)
|
2021-12-24 21:18:34 +08:00
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output({:table_dynamic, pid}, %{id: id, socket: socket}) do
|
2021-11-09 03:45:30 +08:00
|
|
|
live_render(socket, LivebookWeb.Output.TableDynamicLive,
|
|
|
|
id: id,
|
|
|
|
session: %{"id" => id, "pid" => pid}
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2021-12-04 01:59:08 +08:00
|
|
|
defp render_output({:frame_dynamic, pid}, %{
|
|
|
|
id: id,
|
|
|
|
socket: socket,
|
2021-12-24 21:18:34 +08:00
|
|
|
session_id: session_id,
|
2021-12-04 01:59:08 +08:00
|
|
|
input_values: input_values,
|
|
|
|
cell_validity_status: cell_validity_status
|
|
|
|
}) do
|
2021-11-09 03:45:30 +08:00
|
|
|
live_render(socket, LivebookWeb.Output.FrameDynamicLive,
|
|
|
|
id: id,
|
2021-12-04 01:59:08 +08:00
|
|
|
session: %{
|
|
|
|
"id" => id,
|
|
|
|
"pid" => pid,
|
2021-12-24 21:18:34 +08:00
|
|
|
"session_id" => session_id,
|
2021-12-04 01:59:08 +08:00
|
|
|
"input_values" => input_values,
|
|
|
|
"cell_validity_status" => cell_validity_status
|
|
|
|
}
|
2021-11-26 01:43:42 +08:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2022-01-17 03:37:00 +08:00
|
|
|
defp render_output({:frame, outputs, _info}, %{
|
|
|
|
id: id,
|
|
|
|
input_values: input_values,
|
|
|
|
session_id: session_id
|
|
|
|
}) do
|
|
|
|
live_component(LivebookWeb.Output.FrameComponent,
|
|
|
|
id: id,
|
|
|
|
outputs: outputs,
|
|
|
|
session_id: session_id,
|
|
|
|
input_values: input_values
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output({:input, attrs}, %{id: id, input_values: input_values}) do
|
|
|
|
live_component(LivebookWeb.Output.InputComponent,
|
|
|
|
id: id,
|
|
|
|
attrs: attrs,
|
|
|
|
input_values: input_values
|
2021-11-09 03:45:30 +08:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
2021-12-12 07:09:35 +08:00
|
|
|
defp render_output({:control, attrs}, %{id: id, input_values: input_values}) do
|
|
|
|
live_component(LivebookWeb.Output.ControlComponent,
|
|
|
|
id: id,
|
|
|
|
attrs: attrs,
|
|
|
|
input_values: input_values
|
|
|
|
)
|
2021-12-02 23:45:00 +08:00
|
|
|
end
|
|
|
|
|
2021-12-03 21:03:06 +08:00
|
|
|
defp render_output({:error, formatted, :runtime_restart_required}, %{
|
|
|
|
runtime: runtime,
|
|
|
|
cell_validity_status: cell_validity_status
|
|
|
|
})
|
|
|
|
when runtime != nil and cell_validity_status == :evaluated do
|
2021-11-09 03:45:30 +08:00
|
|
|
assigns = %{formatted: formatted, is_standalone: Livebook.Runtime.standalone?(runtime)}
|
|
|
|
|
|
|
|
~H"""
|
|
|
|
<div class="flex flex-col space-y-4">
|
2022-01-17 03:37:00 +08:00
|
|
|
<%= render_error_message_output(@formatted) %>
|
|
|
|
<%= if @is_standalone do %>
|
|
|
|
<div>
|
|
|
|
<button class="button-base button-gray" phx-click="restart_runtime">
|
|
|
|
Reconnect runtime
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
<% else %>
|
|
|
|
<div class="text-red-600">
|
|
|
|
<span class="font-semibold">Note:</span>
|
|
|
|
This operation requires restarting the runtime, but we cannot
|
|
|
|
do it automatically for the current runtime
|
|
|
|
</div>
|
|
|
|
<% end %>
|
2021-11-09 03:45:30 +08:00
|
|
|
</div>
|
|
|
|
"""
|
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output({:error, formatted, _type}, %{}) do
|
2021-11-09 03:45:30 +08:00
|
|
|
render_error_message_output(formatted)
|
|
|
|
end
|
|
|
|
|
2021-11-26 01:43:42 +08:00
|
|
|
defp render_output(output, %{}) do
|
2021-11-09 03:45:30 +08:00
|
|
|
render_error_message_output("""
|
|
|
|
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_output(message) do
|
|
|
|
assigns = %{message: message}
|
|
|
|
|
|
|
|
~H"""
|
|
|
|
<div class="overflow-auto whitespace-pre text-red-600 tiny-scrollbar"><%= @message %></div>
|
|
|
|
"""
|
|
|
|
end
|
|
|
|
end
|