mirror of
https://github.com/livebook-dev/livebook.git
synced 2024-11-16 04:46:46 +08:00
93 lines
2.4 KiB
Elixir
93 lines
2.4 KiB
Elixir
defmodule LivebookWeb.Output.FileInputComponent do
|
|
use LivebookWeb, :live_component
|
|
|
|
@impl true
|
|
def mount(socket) do
|
|
{:ok, assign(socket, initialized: false)}
|
|
end
|
|
|
|
@impl true
|
|
def update(assigns, socket) do
|
|
socket = assign(socket, assigns)
|
|
|
|
socket =
|
|
if socket.assigns.initialized do
|
|
socket
|
|
else
|
|
socket
|
|
|> allow_upload(:file,
|
|
accept: socket.assigns.accept,
|
|
max_entries: 1,
|
|
progress: &handle_progress/3,
|
|
auto_upload: true
|
|
)
|
|
|> assign(initialized: true)
|
|
end
|
|
|
|
{:ok, socket}
|
|
end
|
|
|
|
@impl true
|
|
def render(assigns) do
|
|
~H"""
|
|
<form id={"#{@id}-root"} phx-change="validate" phx-target={@myself}>
|
|
<label
|
|
class="inline-flex flex-col gap-4 p-4 border-2 border-dashed border-gray-200 rounded-lg cursor-pointer"
|
|
phx-drop-target={@uploads.file.ref}
|
|
phx-hook="Dropzone"
|
|
id={"#{@id}-upload-dropzone"}
|
|
>
|
|
<div class="flex justify-center text-gray-500">
|
|
<%= if @value do %>
|
|
<%= @value.client_name %>
|
|
<% else %>
|
|
Click to select a file or drag a local file here
|
|
<% end %>
|
|
</div>
|
|
<.live_file_input upload={@uploads.file} class="hidden" />
|
|
</label>
|
|
</form>
|
|
"""
|
|
end
|
|
|
|
@impl true
|
|
def handle_event("validate", %{}, socket) do
|
|
{:noreply, socket}
|
|
end
|
|
|
|
defp handle_progress(:file, entry, socket) do
|
|
if entry.done? do
|
|
socket
|
|
|> consume_uploaded_entries(:file, fn %{path: path}, entry ->
|
|
{:ok, file_ref} =
|
|
if socket.assigns.local do
|
|
key = "#{socket.assigns.input_id}-#{socket.assigns.client_id}"
|
|
|
|
Livebook.Session.register_file(socket.assigns.session_pid, path, key,
|
|
linked_client_id: socket.assigns.client_id
|
|
)
|
|
else
|
|
key = "#{socket.assigns.input_id}-global"
|
|
Livebook.Session.register_file(socket.assigns.session_pid, path, key)
|
|
end
|
|
|
|
{:ok, {file_ref, entry.client_name}}
|
|
end)
|
|
|> case do
|
|
[{file_ref, client_name}] ->
|
|
value = %{file_ref: file_ref, client_name: client_name}
|
|
|
|
send_update(LivebookWeb.Output.InputComponent,
|
|
id: socket.assigns.input_component_id,
|
|
event: :change,
|
|
value: value
|
|
)
|
|
|
|
[] ->
|
|
:ok
|
|
end
|
|
end
|
|
|
|
{:noreply, socket}
|
|
end
|
|
end
|