livebook/lib/livebook_web/live/output/control_component.ex
Jonatan Kłosko e2490c0f7f
Implement signature intellisense (#640)
* Remove Code.Fragment backport

* Fix tests compatibility with Elixir 1.13

* Implement signature intellisense

* Don't show signatures on module attributes

* Add tests for calls with do-end block

* Unify spec formatting

* Insert parentheses when completing a function call

* Send all text until cursor in signature request

* Add configuration for completion/signature popups (#693)

* Add editor settings form

* Add configuration for intellisense defaults

* Read fresh settings when editor mounts

* Scope attribute names

* Fix disabled button styling

* Simplify signature box and enable by default

* Split settings into system and user sections

* Update lib/livebook_web/live/settings_live.ex

Co-authored-by: José Valim <jose.valim@dashbit.co>

* Update lib/livebook_web/live/settings_live.ex

Co-authored-by: José Valim <jose.valim@dashbit.co>

Co-authored-by: José Valim <jose.valim@dashbit.co>

* Fix spacing in documentation tests

Co-authored-by: José Valim <jose.valim@dashbit.co>
2021-12-03 21:57:21 +01:00

81 lines
2.2 KiB
Elixir

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"""
<div class="flex"
id={"#{@id}-root"}
phx-hook="KeyboardControl"
data-keydown-enabled={to_string(@keyboard_enabled and :keydown in @attrs.events)}
data-keyup-enabled={to_string(@keyboard_enabled and :keyup in @attrs.events)}
data-target={@myself}>
<span class="tooltip right" data-tooltip="Toggle keyboard control">
<button class={"button-base #{if @keyboard_enabled, do: "button-blue", else: "button-gray"} button-square-icon"}
type="button"
aria-label="toggle keyboard control"
phx-click={JS.push("toggle_keyboard", target: @myself)}>
<.remix_icon icon="keyboard-line" />
</button>
</span>
</div>
"""
end
def render(%{attrs: %{type: :button}} = assigns) do
~H"""
<div class="flex">
<button class="button-base button-gray"
type="button"
phx-click={JS.push("button_click", target: @myself)}>
<%= @attrs.label %>
</button>
</div>
"""
end
def render(assigns) do
~H"""
<div class="text-red-600">
Unknown control type <%= @attrs.type %>
</div>
"""
end
@impl true
def handle_event("toggle_keyboard", %{}, socket) do
socket = update(socket, :keyboard_enabled, &not/1)
if :status in socket.assigns.attrs.events do
report_event(socket, %{type: :status, enabled: socket.assigns.keyboard_enabled})
end
{:noreply, socket}
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 report_event(socket, attrs) do
topic = socket.assigns.attrs.ref
event = Map.merge(%{origin: self()}, attrs)
send(socket.assigns.attrs.destination, {:event, topic, event})
end
end