2023-01-05 04:44:04 +08:00
|
|
|
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,
|
2023-05-24 20:17:47 +08:00
|
|
|
# The file input specifies the accepted formats, but in order
|
|
|
|
# to handle unknown MIME types, we need to pass :any to
|
2023-06-07 18:30:52 +08:00
|
|
|
# allow_upload and override the accept attribute ourselves
|
2023-05-24 20:17:47 +08:00
|
|
|
accept: :any,
|
2023-01-05 04:44:04 +08:00
|
|
|
max_entries: 1,
|
2023-08-14 21:20:55 +08:00
|
|
|
max_file_size: 100_000_000_000,
|
2023-01-05 04:44:04 +08:00
|
|
|
progress: &handle_progress/3,
|
|
|
|
auto_upload: true
|
|
|
|
)
|
2023-05-24 20:17:47 +08:00
|
|
|
|> assign(
|
|
|
|
initialized: true,
|
|
|
|
accept:
|
|
|
|
case socket.assigns.accept do
|
|
|
|
:any -> nil
|
|
|
|
types when is_list(types) -> Enum.join(types, ",")
|
|
|
|
end
|
|
|
|
)
|
2023-01-05 04:44:04 +08:00
|
|
|
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"
|
2023-02-09 19:34:46 +08:00
|
|
|
id={"#{@id}-upload-dropzone"}
|
2023-01-05 04:44:04 +08:00
|
|
|
>
|
|
|
|
<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>
|
2023-05-24 20:17:47 +08:00
|
|
|
<.live_file_input upload={@uploads.file} class="hidden" accept={@accept} />
|
2023-01-05 04:44:04 +08:00
|
|
|
</label>
|
2023-08-14 21:20:55 +08:00
|
|
|
<p :for={msg <- upload_error_messages(@uploads.file)} class="mt-0.5 text-red-600 text-sm">
|
|
|
|
<%= msg %>
|
|
|
|
</p>
|
2023-01-05 04:44:04 +08:00
|
|
|
</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 ->
|
2023-02-02 02:59:26 +08:00
|
|
|
{:ok, file_ref} =
|
2023-02-02 01:49:12 +08:00
|
|
|
if socket.assigns.local do
|
|
|
|
key = "#{socket.assigns.input_id}-#{socket.assigns.client_id}"
|
2023-01-05 04:44:04 +08:00
|
|
|
|
2023-02-02 01:49:12 +08:00
|
|
|
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
|
2023-01-05 04:44:04 +08:00
|
|
|
|
2023-02-02 02:59:26 +08:00
|
|
|
{:ok, {file_ref, entry.client_name}}
|
2023-01-05 04:44:04 +08:00
|
|
|
end)
|
|
|
|
|> case do
|
2023-02-02 02:59:26 +08:00
|
|
|
[{file_ref, client_name}] ->
|
|
|
|
value = %{file_ref: file_ref, client_name: client_name}
|
2023-01-05 04:44:04 +08:00
|
|
|
|
|
|
|
send_update(LivebookWeb.Output.InputComponent,
|
|
|
|
id: socket.assigns.input_component_id,
|
|
|
|
event: :change,
|
|
|
|
value: value
|
|
|
|
)
|
|
|
|
|
|
|
|
[] ->
|
|
|
|
:ok
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
{:noreply, socket}
|
|
|
|
end
|
|
|
|
end
|