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

108 lines
3.2 KiB
Elixir
Raw Normal View History

2022-12-29 22:09:09 +08:00
defmodule LivebookWeb.Output.AudioInputComponent do
use LivebookWeb, :live_component
@impl true
def mount(socket) do
{:ok, assign(socket, endianness: System.endianness(), initialized: false)}
end
@impl true
def update(assigns, socket) do
{value, assigns} = Map.pop!(assigns, :value)
socket = assign(socket, assigns)
socket =
if socket.assigns.initialized do
socket
else
socket =
if value do
push_event(socket, "audio_input_init:#{socket.assigns.id}", %{
data: Base.encode64(value.data),
num_channels: value.num_channels,
sampling_rate: value.sampling_rate
})
else
socket
end
assign(socket, initialized: true)
end
{:ok, socket}
end
@impl true
def render(assigns) do
~H"""
<div
id={"#{@id}-root"}
class="inline-flex flex-col gap-4 p-4 border-2 border-dashed border-gray-200 rounded-lg"
phx-hook="AudioInput"
phx-update="ignore"
data-id={@id}
2023-01-05 04:44:04 +08:00
data-phx-target={@myself}
2022-12-29 22:09:09 +08:00
data-format={@format}
data-sampling-rate={@sampling_rate}
data-endianness={@endianness}
>
<input type="file" data-input class="hidden" name="value" accept="audio/*" capture="user" />
<audio controls data-preview></audio>
<div class="flex items-center justify-center gap-4">
<button
class="button-base button-gray border-transparent py-2 px-4 inline-flex text-gray-500"
data-btn-record
>
<.remix_icon icon="mic-line" class="text-lg leading-none mr-2" />
<span>Record</span>
</button>
<button
class="hidden button-base button-gray border-transparent py-2 px-4 inline-flex text-gray-500 items-center"
data-btn-stop
>
<span class="mr-2 flex h-3 w-3 relative">
<span class="animate-ping absolute inline-flex h-full w-full rounded-full bg-gray-400 opacity-75">
</span>
<span class="relative inline-flex rounded-full h-3 w-3 bg-gray-500"></span>
</span>
<span>Stop recording</span>
</button>
<button
class="hidden button-base button-gray border-transparent py-2 px-4 inline-flex text-gray-500"
data-btn-cancel
>
<.remix_icon icon="close-circle-line" class="text-lg leading-none mr-2" />
<span>Cancel</span>
</button>
<button
class="button-base button-gray border-transparent py-2 px-4 inline-flex text-gray-500"
data-btn-upload
>
<.remix_icon icon="upload-2-line" class="text-lg leading-none mr-2" />
<span>Upload</span>
</button>
</div>
</div>
"""
end
2023-01-05 04:44:04 +08:00
@impl true
def handle_event("change", params, socket) do
value = %{
data: Base.decode64!(params["data"]),
num_channels: params["num_channels"],
sampling_rate: params["sampling_rate"],
format: socket.assigns.format
}
send_update(LivebookWeb.Output.InputComponent,
id: socket.assigns.input_component_id,
event: :change,
value: value
)
{:noreply, socket}
end
2022-12-29 22:09:09 +08:00
end