livebook/lib/livebook_web/live/output/control_component.ex

116 lines
3.3 KiB
Elixir
Raw Normal View History

defmodule LivebookWeb.Output.ControlComponent do
use LivebookWeb, :live_component
@impl true
def mount(socket) do
{:ok, assign(socket, keyboard_enabled: false)}
end
@impl true
2023-08-24 05:25:04 +08:00
def render(assigns) when assigns.control.attrs.type == :keyboard do
~H"""
2022-08-02 21:51:02 +08:00
<div
class="flex"
id={"#{@id}-root"}
phx-hook="KeyboardControl"
2023-11-23 23:18:06 +08:00
data-p-cell-id={hook_prop(@cell_id)}
data-p-default-handlers={hook_prop(@control.attrs.default_handlers)}
data-p-keydown-enabled={hook_prop(@keyboard_enabled and :keydown in @control.attrs.events)}
data-p-keyup-enabled={hook_prop(@keyboard_enabled and :keyup in @control.attrs.events)}
data-p-target={hook_prop(@myself)}
2022-08-02 21:51:02 +08:00
>
<span class="tooltip right" data-tooltip="Toggle keyboard control">
<.button
color={if(@keyboard_enabled, do: "blue", else: "gray")}
small
type="button"
aria-label="toggle keyboard control"
2022-08-02 21:51:02 +08:00
phx-click={JS.push("toggle_keyboard", target: @myself)}
>
<.remix_icon icon="keyboard-line" class="text-xl leading-none py-1" />
</.button>
</span>
</div>
"""
end
2023-08-24 05:25:04 +08:00
def render(assigns) when assigns.control.attrs.type == :button do
~H"""
<div class="flex">
<.button color="gray" type="button" phx-click={JS.push("button_click", target: @myself)}>
2023-08-24 05:25:04 +08:00
<%= @control.attrs.label %>
</.button>
</div>
"""
end
2023-08-24 05:25:04 +08:00
def render(assigns) when assigns.control.attrs.type == :form do
~H"""
<div>
2022-08-02 21:51:02 +08:00
<.live_component
module={LivebookWeb.Output.ControlFormComponent}
id={@id}
2023-08-24 05:25:04 +08:00
control={@control}
input_views={@input_views}
session_pid={@session_pid}
client_id={@client_id}
2022-08-02 21:51:02 +08:00
/>
</div>
"""
end
def render(assigns) do
~H"""
<div class="text-red-600">
2023-08-24 05:25:04 +08:00
Unknown control type <%= @control.attrs.type %>
</div>
"""
end
@impl true
def handle_event("toggle_keyboard", %{}, socket) do
enabled = !socket.assigns.keyboard_enabled
maybe_report_status(socket, enabled)
{:noreply, assign(socket, keyboard_enabled: enabled)}
end
def handle_event("enable_keyboard", %{}, socket) do
maybe_report_status(socket, true)
{:noreply, assign(socket, keyboard_enabled: true)}
end
def handle_event("disable_keyboard", %{}, socket) do
maybe_report_status(socket, false)
{:noreply, assign(socket, keyboard_enabled: false)}
end
def handle_event("button_click", %{}, socket) do
report_event(socket, %{type: :click})
{:noreply, socket}
end
def handle_event("keydown", %{"key" => key}, socket) do
report_event(socket, %{type: :keydown, key: key})
{:noreply, socket}
end
def handle_event("keyup", %{"key" => key}, socket) do
report_event(socket, %{type: :keyup, key: key})
{:noreply, socket}
end
defp maybe_report_status(socket, enabled) do
2023-10-28 00:37:21 +08:00
%{assigns: %{control: %{attrs: attrs}, keyboard_enabled: current}} = socket
if :status in attrs.events and enabled != current do
report_event(socket, %{type: :status, enabled: enabled})
end
end
defp report_event(socket, attrs) do
2023-08-24 05:25:04 +08:00
topic = socket.assigns.control.ref
event = Map.merge(%{origin: socket.assigns.client_id}, attrs)
2023-08-24 05:25:04 +08:00
send(socket.assigns.control.destination, {:event, topic, event})
end
end