2023-03-07 01:14:33 +08:00
|
|
|
defmodule LivebookWeb.OpenLive.UploadComponent do
|
2021-11-01 20:59:39 +08:00
|
|
|
use LivebookWeb, :live_component
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def mount(socket) do
|
|
|
|
{:ok,
|
|
|
|
socket
|
|
|
|
|> assign(:error, false)
|
|
|
|
|> allow_upload(:notebook, accept: ~w(.livemd), max_entries: 1)}
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def render(assigns) do
|
|
|
|
~H"""
|
|
|
|
<div class="flex-col space-y-5">
|
2022-02-05 00:05:38 +08:00
|
|
|
<p class="text-gray-700" id="import-from-file">
|
2023-03-07 01:14:33 +08:00
|
|
|
Drag and drop a .livemd file below to import it.
|
2021-11-01 20:59:39 +08:00
|
|
|
</p>
|
2022-08-02 21:51:02 +08:00
|
|
|
<form
|
|
|
|
id="upload-file-form"
|
2021-11-01 20:59:39 +08:00
|
|
|
phx-submit="save"
|
|
|
|
phx-change="validate"
|
|
|
|
phx-drop-target={@uploads.notebook.ref}
|
|
|
|
phx-target={@myself}
|
|
|
|
class="flex flex-col items-start"
|
|
|
|
>
|
2023-02-23 02:34:54 +08:00
|
|
|
<.live_file_input
|
|
|
|
upload={@uploads.notebook}
|
|
|
|
class="hidden"
|
|
|
|
aria-labelledby="import-from-file"
|
|
|
|
/>
|
2022-08-02 21:51:02 +08:00
|
|
|
<div
|
|
|
|
class="flex flex-col justify-center items-center w-full rounded-xl border-2 border-dashed border-gray-400 h-48"
|
2022-03-16 18:33:53 +08:00
|
|
|
phx-hook="Dropzone"
|
2023-02-09 19:34:46 +08:00
|
|
|
id="import-file-upload-dropzone"
|
2022-08-02 21:51:02 +08:00
|
|
|
>
|
2021-11-01 20:59:39 +08:00
|
|
|
<%= if @uploads.notebook.entries == [] do %>
|
|
|
|
<span name="placeholder" class="font-medium text-gray-400">Drop your notebook here</span>
|
|
|
|
<% else %>
|
2023-02-23 02:34:54 +08:00
|
|
|
<div :for={file <- @uploads.notebook.entries} class="flex items-center">
|
|
|
|
<span class="font-medium text-gray-400"><%= file.client_name %></span>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
class="icon-button"
|
|
|
|
phx-click="clear-file"
|
|
|
|
phx-target={@myself}
|
|
|
|
tabindex="-1"
|
|
|
|
>
|
|
|
|
<.remix_icon icon="close-line" class="text-xl text-gray-300 hover:text-gray-500" />
|
|
|
|
</button>
|
|
|
|
</div>
|
2021-11-01 20:59:39 +08:00
|
|
|
<% end %>
|
|
|
|
</div>
|
|
|
|
<%= if @error do %>
|
|
|
|
<div class="text-red-500 text-sm py-2">
|
|
|
|
You can only upload files with .livemd extension.
|
|
|
|
</div>
|
|
|
|
<% end %>
|
2022-08-02 21:51:02 +08:00
|
|
|
<button
|
|
|
|
type="submit"
|
|
|
|
class="mt-5 button-base button-blue"
|
|
|
|
disabled={@uploads.notebook.entries == [] || @error}
|
|
|
|
>
|
2021-11-01 20:59:39 +08:00
|
|
|
Import
|
|
|
|
</button>
|
|
|
|
</form>
|
|
|
|
</div>
|
|
|
|
"""
|
|
|
|
end
|
|
|
|
|
2023-07-06 02:01:12 +08:00
|
|
|
@impl true
|
2021-11-01 20:59:39 +08:00
|
|
|
def handle_event("clear-file", _params, socket) do
|
|
|
|
{socket, _entries} = Phoenix.LiveView.Upload.maybe_cancel_uploads(socket)
|
|
|
|
{:noreply, assign(socket, error: false)}
|
|
|
|
end
|
|
|
|
|
2023-07-06 02:01:12 +08:00
|
|
|
@impl true
|
2021-11-01 20:59:39 +08:00
|
|
|
def handle_event("validate", _params, socket) do
|
|
|
|
has_error? = Enum.any?(socket.assigns.uploads.notebook.entries, &(not &1.valid?))
|
|
|
|
|
|
|
|
{:noreply, assign(socket, error: has_error?)}
|
|
|
|
end
|
|
|
|
|
2023-07-06 02:01:12 +08:00
|
|
|
@impl true
|
2021-11-01 20:59:39 +08:00
|
|
|
def handle_event("save", _params, socket) do
|
|
|
|
consume_uploaded_entries(socket, :notebook, fn %{path: path}, _entry ->
|
|
|
|
content = File.read!(path)
|
|
|
|
|
2023-03-07 01:14:33 +08:00
|
|
|
send(self(), {:import_source, content, []})
|
2022-03-16 18:33:53 +08:00
|
|
|
|
|
|
|
{:ok, :ok}
|
2021-11-01 20:59:39 +08:00
|
|
|
end)
|
|
|
|
|
|
|
|
{:noreply, socket}
|
|
|
|
end
|
|
|
|
end
|