defmodule LivebookWeb.Output.ControlComponent do
use LivebookWeb, :live_component
@impl true
def mount(socket) do
{:ok, assign(socket, keyboard_enabled: false)}
end
@impl true
def render(%{attrs: %{type: :keyboard}} = assigns) do
~H"""
"""
end
def render(%{attrs: %{type: :button}} = assigns) do
~H"""
"""
end
def render(%{attrs: %{type: :form}} = assigns) do
~H"""
<.live_component
module={LivebookWeb.Output.ControlFormComponent}
id={@id}
attrs={@attrs}
input_views={@input_views}
session_pid={@session_pid}
client_id={@client_id}
/>
"""
end
def render(assigns) do
~H"""
Unknown control type <%= @attrs.type %>
"""
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
%{assigns: %{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
topic = socket.assigns.attrs.ref
event = Map.merge(%{origin: socket.assigns.client_id}, attrs)
send(socket.assigns.attrs.destination, {:event, topic, event})
end
end