livebook/lib/livebook_web/live/session_live/insert_buttons_component.ex

198 lines
6.5 KiB
Elixir
Raw Normal View History

defmodule LivebookWeb.SessionLive.InsertButtonsComponent do
use LivebookWeb, :live_component
def render(assigns) do
~H"""
2022-08-02 21:51:02 +08:00
<div
class="relative top-0.5 m-0 flex justify-center"
role="toolbar"
aria-label="insert new"
2022-08-02 21:51:02 +08:00
data-el-insert-buttons
>
<div class={
"w-full absolute z-10 hover:z-[11] #{if(@persistent, do: "opacity-100", else: "opacity-0")} hover:opacity-100 focus-within:opacity-100 flex space-x-2 justify-center items-center"
}>
<button
class="button-base button-small"
phx-click="insert_cell_below"
phx-value-type="code"
phx-value-section_id={@section_id}
phx-value-cell_id={@cell_id}
2022-08-02 21:51:02 +08:00
>
+ Code
</button>
2023-02-23 02:34:54 +08:00
<.menu id={"#{@id}-block-menu"} position={:bottom_left}>
<:toggle>
<button class="button-base button-small">+ Block</button>
</:toggle>
2023-02-24 02:07:00 +08:00
<.menu_item>
2022-08-02 21:51:02 +08:00
<button
role="menuitem"
phx-click="insert_cell_below"
phx-value-type="markdown"
phx-value-section_id={@section_id}
2022-08-02 21:51:02 +08:00
phx-value-cell_id={@cell_id}
>
<.remix_icon icon="markdown-fill" />
2023-02-24 02:07:00 +08:00
<span>Markdown</span>
</button>
2023-02-24 02:07:00 +08:00
</.menu_item>
<.menu_item>
2022-08-02 21:51:02 +08:00
<button
role="menuitem"
phx-click="insert_section_below"
phx-value-section_id={@section_id}
2022-08-02 21:51:02 +08:00
phx-value-cell_id={@cell_id}
>
<.remix_icon icon="h-2" />
2023-02-24 02:07:00 +08:00
<span>Section</span>
</button>
2023-02-24 02:07:00 +08:00
</.menu_item>
<div class="my-2 border-b border-gray-200"></div>
<.menu_item>
2022-08-02 21:51:02 +08:00
<button
2022-04-19 22:51:46 +08:00
role="menuitem"
phx-click="insert_cell_below"
phx-value-type="diagram"
phx-value-section_id={@section_id}
2022-08-02 21:51:02 +08:00
phx-value-cell_id={@cell_id}
>
<.remix_icon icon="organization-chart" />
2023-02-24 02:07:00 +08:00
<span>Diagram</span>
2022-08-02 21:51:02 +08:00
</button>
2023-02-24 02:07:00 +08:00
</.menu_item>
<.menu_item>
2023-02-23 02:34:54 +08:00
<.link
patch={
~p"/sessions/#{@session_id}/cell-upload?section_id=#{@section_id}&cell_id=#{@cell_id || ""}"
}
aria-label="insert image"
role="menuitem"
>
<.remix_icon icon="image-add-line" />
2023-02-24 02:07:00 +08:00
<span>Image</span>
2023-02-23 02:34:54 +08:00
</.link>
2023-02-24 02:07:00 +08:00
</.menu_item>
</.menu>
<%= cond do %>
<% not Livebook.Runtime.connected?(@runtime) -> %>
2022-08-02 21:51:02 +08:00
<button
class="button-base button-small"
phx-click={
with_confirm(
JS.push("setup_default_runtime"),
title: "Setup runtime",
description: ~s'''
To see the available smart cells, you need a connected runtime.
Do you want to connect and setup the default one?
''',
confirm_text: "Setup runtime",
confirm_icon: "play-line",
danger: false
)
2022-08-02 21:51:02 +08:00
}
>
+ Smart
</button>
<% @smart_cell_definitions == [] -> %>
<span class="tooltip right" data-tooltip="No smart cells available">
<button class="button-base button-small" disabled>+ Smart</button>
</span>
<% true -> %>
2023-02-23 02:34:54 +08:00
<.menu id={"#{@id}-smart-menu"} position={:bottom_left}>
<:toggle>
<button class="button-base button-small">+ Smart</button>
</:toggle>
2023-02-24 02:07:00 +08:00
<.menu_item :for={definition <- Enum.sort_by(@smart_cell_definitions, & &1.name)}>
2023-02-23 02:34:54 +08:00
<.smart_cell_insert_button
definition={definition}
section_id={@section_id}
cell_id={@cell_id}
/>
2023-02-24 02:07:00 +08:00
</.menu_item>
</.menu>
<% end %>
</div>
</div>
"""
end
defp smart_cell_insert_button(%{definition: %{requirement: %{variants: [_, _ | _]}}} = assigns) do
~H"""
<.submenu>
2023-02-24 02:07:00 +08:00
<:primary>
<button role="menuitem">
<span><%= @definition.name %></span>
</button>
</:primary>
<.menu_item :for={{variant, idx} <- Enum.with_index(@definition.requirement.variants)}>
2023-02-23 02:34:54 +08:00
<button
role="menuitem"
phx-click={on_smart_cell_click(@definition, idx, @section_id, @cell_id)}
>
2023-02-24 02:07:00 +08:00
<span><%= variant.name %></span>
2023-02-23 02:34:54 +08:00
</button>
2023-02-24 02:07:00 +08:00
</.menu_item>
</.submenu>
"""
end
defp smart_cell_insert_button(assigns) do
~H"""
2023-02-24 02:07:00 +08:00
<button role="menuitem" phx-click={on_smart_cell_click(@definition, 0, @section_id, @cell_id)}>
<span><%= @definition.name %></span>
</button>
"""
end
defp on_smart_cell_click(%{requirement: nil} = definition, _variant_idx, section_id, cell_id) do
insert_smart_cell(definition, section_id, cell_id)
end
defp on_smart_cell_click(%{requirement: %{}} = definition, variant_idx, section_id, cell_id) do
variant = Enum.fetch!(definition.requirement.variants, variant_idx)
with_confirm(
JS.push("add_smart_cell_dependencies",
value: %{kind: definition.kind, variant_idx: variant_idx}
)
|> insert_smart_cell(definition, section_id, cell_id)
|> JS.push("queue_cells_reevaluation"),
title: "Add packages",
description:
case variant.packages do
[%{name: name}] ->
~s'''
The <span class="font-semibold">#{definition.name}“</span>
smart cell requires the #{code_tag(name)} package. Do you want to add
it as a dependency and restart?
'''
packages ->
~s'''
The <span class="font-semibold">#{definition.name}“</span>
smart cell requires the #{packages |> Enum.map(&code_tag(&1.name)) |> format_items()}
packages. Do you want to add them as dependencies and restart?
'''
end,
confirm_text: "Add and restart",
confirm_icon: "add-line",
danger: false,
html: true
)
end
defp code_tag(text), do: "<code>#{text}</code>"
defp insert_smart_cell(js \\ %JS{}, definition, section_id, cell_id) do
JS.push(js, "insert_cell_below",
value: %{
type: "smart",
kind: definition.kind,
section_id: section_id,
cell_id: cell_id
}
)
end
end