2023-03-07 01:14:33 +08:00
|
|
|
defmodule LivebookWeb.OpenLive do
|
|
|
|
use LivebookWeb, :live_view
|
|
|
|
|
|
|
|
import LivebookWeb.SessionHelpers
|
|
|
|
|
2024-01-26 12:47:56 +08:00
|
|
|
alias LivebookWeb.LayoutComponents
|
2023-03-07 01:14:33 +08:00
|
|
|
alias Livebook.{Sessions, Notebook, FileSystem}
|
|
|
|
|
|
|
|
on_mount LivebookWeb.SidebarHook
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def mount(params, _session, socket) do
|
|
|
|
if connected?(socket) do
|
|
|
|
Livebook.Sessions.subscribe()
|
|
|
|
Livebook.NotebookManager.subscribe_recent_notebooks()
|
|
|
|
end
|
|
|
|
|
|
|
|
sessions = Sessions.list_sessions() |> Enum.filter(&(&1.mode == :default))
|
|
|
|
recent_notebooks = Livebook.NotebookManager.recent_notebooks()
|
|
|
|
|
|
|
|
{:ok,
|
|
|
|
assign(socket,
|
|
|
|
tab: "file",
|
|
|
|
initial_file: file_from_params(params),
|
|
|
|
url: params["url"],
|
|
|
|
sessions: sessions,
|
|
|
|
recent_notebooks: recent_notebooks,
|
2023-03-27 18:56:53 +08:00
|
|
|
page_title: "Open - Livebook"
|
2023-03-07 01:14:33 +08:00
|
|
|
)}
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def render(assigns) do
|
|
|
|
~H"""
|
2024-01-26 12:47:56 +08:00
|
|
|
<LayoutComponents.layout
|
|
|
|
current_page={~p"/"}
|
|
|
|
current_user={@current_user}
|
|
|
|
saved_hubs={@saved_hubs}
|
|
|
|
>
|
2023-03-07 01:14:33 +08:00
|
|
|
<:topbar_action>
|
2024-02-09 02:27:16 +08:00
|
|
|
<.button color="blue" navigate={~p"/new"}>
|
|
|
|
<.remix_icon icon="add-line" />
|
2023-03-07 01:14:33 +08:00
|
|
|
<span>New notebook</span>
|
2024-02-09 02:27:16 +08:00
|
|
|
</.button>
|
2023-03-07 01:14:33 +08:00
|
|
|
</:topbar_action>
|
|
|
|
<div class="p-4 md:px-12 md:py-6 max-w-screen-lg mx-auto space-y-4">
|
|
|
|
<div class="flex flex-row space-y-0 items-center pb-4 justify-between">
|
2024-01-26 12:47:56 +08:00
|
|
|
<LayoutComponents.title text="Open notebook" back_navigate={~p"/"} />
|
2023-03-07 01:14:33 +08:00
|
|
|
<div class="hidden md:flex" role="navigation" aria-label="new notebook">
|
2024-02-09 02:27:16 +08:00
|
|
|
<.button color="blue" navigate={~p"/new"}>
|
|
|
|
<.remix_icon icon="add-line" />
|
2023-03-07 01:14:33 +08:00
|
|
|
<span>New notebook</span>
|
2024-02-09 02:27:16 +08:00
|
|
|
</.button>
|
2023-03-07 01:14:33 +08:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="tabs">
|
2023-08-11 17:52:19 +08:00
|
|
|
<.link patch={~p"/open/storage"} class={["tab", @tab == "storage" && "active"]}>
|
2023-07-06 02:01:12 +08:00
|
|
|
<.remix_icon icon="file-3-line" class="align-middle" />
|
2023-08-11 17:52:19 +08:00
|
|
|
<span class="font-medium">From storage</span>
|
2023-03-07 01:14:33 +08:00
|
|
|
</.link>
|
|
|
|
<.link patch={~p"/open/url"} class={["tab", @tab == "url" && "active"]}>
|
|
|
|
<.remix_icon icon="download-cloud-2-line" class="align-middle" />
|
|
|
|
<span class="font-medium">From URL</span>
|
|
|
|
</.link>
|
|
|
|
<.link patch={~p"/open/source"} class={["tab", @tab == "source" && "active"]}>
|
|
|
|
<.remix_icon icon="clipboard-line" class="align-middle" />
|
|
|
|
<span class="font-medium">From source</span>
|
|
|
|
</.link>
|
|
|
|
<.link patch={~p"/open/upload"} class={["tab", @tab == "upload" && "active"]}>
|
|
|
|
<.remix_icon icon="file-upload-line" class="align-middle" />
|
|
|
|
<span class="font-medium">File upload</span>
|
|
|
|
</.link>
|
|
|
|
<div class="grow tab"></div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="h-96">
|
|
|
|
<.live_component
|
2023-08-11 17:52:19 +08:00
|
|
|
:if={@tab == "storage"}
|
2023-03-07 01:14:33 +08:00
|
|
|
module={LivebookWeb.OpenLive.FileComponent}
|
|
|
|
id="import-file"
|
|
|
|
sessions={@sessions}
|
|
|
|
initial_file={@initial_file}
|
|
|
|
/>
|
|
|
|
<.live_component
|
|
|
|
:if={@tab == "url"}
|
|
|
|
module={LivebookWeb.OpenLive.UrlComponent}
|
|
|
|
id="import-url"
|
|
|
|
url={@url}
|
|
|
|
/>
|
|
|
|
<.live_component
|
|
|
|
:if={@tab == "source"}
|
|
|
|
module={LivebookWeb.OpenLive.SourceComponent}
|
|
|
|
id="import-source"
|
|
|
|
/>
|
|
|
|
<.live_component
|
|
|
|
:if={@tab == "upload"}
|
|
|
|
module={LivebookWeb.OpenLive.UploadComponent}
|
|
|
|
id="import-upload"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
2023-03-07 01:39:54 +08:00
|
|
|
<div id="recent-notebooks" class="pb-10" role="region" aria-label="recent notebooks">
|
2023-03-07 01:14:33 +08:00
|
|
|
<div class="mb-4 flex items-center md:items-end justify-between">
|
|
|
|
<h2 class="uppercase font-semibold text-gray-500 text-sm md:text-base">
|
|
|
|
Recent notebooks
|
|
|
|
</h2>
|
|
|
|
</div>
|
|
|
|
<%= if @recent_notebooks == [] do %>
|
|
|
|
<.no_entries>
|
|
|
|
Your most recently opened notebooks will appear here.
|
|
|
|
</.no_entries>
|
|
|
|
<% else %>
|
|
|
|
<.live_component
|
|
|
|
module={LivebookWeb.NotebookCardsComponent}
|
|
|
|
id="recent-notebook-list"
|
|
|
|
notebook_infos={@recent_notebooks}
|
|
|
|
sessions={@sessions}
|
|
|
|
added_at_label="Opened"
|
2023-03-28 02:59:12 +08:00
|
|
|
>
|
|
|
|
<:card_icon :let={{_info, idx}}>
|
|
|
|
<span class="tooltip top" data-tooltip="Hide notebook">
|
|
|
|
<button
|
|
|
|
aria-label="hide notebook"
|
2023-05-11 00:23:08 +08:00
|
|
|
phx-click={JS.push("hide_recent_notebook", value: %{idx: idx})}
|
2023-03-28 02:59:12 +08:00
|
|
|
>
|
|
|
|
<.remix_icon icon="close-fill" class="text-gray-600 text-lg" />
|
|
|
|
</button>
|
|
|
|
</span>
|
|
|
|
</:card_icon>
|
|
|
|
</.live_component>
|
2023-03-07 01:14:33 +08:00
|
|
|
<% end %>
|
|
|
|
<div class="mt-3 text-gray-600 text-sm">
|
|
|
|
Looking for unsaved notebooks? <.link
|
|
|
|
class="font-semibold"
|
2023-08-11 17:52:19 +08:00
|
|
|
navigate={~p"/open/storage?autosave=true"}
|
2023-03-07 01:14:33 +08:00
|
|
|
phx-no-format
|
|
|
|
>Browse them here</.link>.
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
2024-01-26 12:47:56 +08:00
|
|
|
</LayoutComponents.layout>
|
2023-03-07 01:14:33 +08:00
|
|
|
"""
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def handle_params(%{"tab" => tab}, _url, socket) when socket.assigns.live_action == :page do
|
|
|
|
{:noreply, assign(socket, tab: tab)}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_params(%{"url" => url}, _url, socket)
|
|
|
|
when socket.assigns.live_action == :public_import do
|
|
|
|
origin = Notebook.ContentLoader.url_to_location(url)
|
|
|
|
|
|
|
|
origin
|
|
|
|
|> Notebook.ContentLoader.fetch_content_from_location()
|
|
|
|
|> case do
|
|
|
|
{:ok, content} ->
|
|
|
|
socket = import_source(socket, content, origin: origin)
|
|
|
|
{:noreply, socket}
|
|
|
|
|
|
|
|
{:error, _message} ->
|
|
|
|
{:noreply, push_patch(socket, to: ~p"/open/url?url=#{url}")}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_params(%{"path" => path} = _params, _uri, socket)
|
|
|
|
when socket.assigns.live_action == :public_open do
|
|
|
|
expanded_path = Path.expand(path)
|
|
|
|
|
|
|
|
if File.dir?(expanded_path) do
|
2023-08-11 17:52:19 +08:00
|
|
|
{:noreply, push_patch(socket, to: ~p"/open/storage?path=#{path}")}
|
2023-03-07 01:14:33 +08:00
|
|
|
else
|
|
|
|
file = FileSystem.File.local(expanded_path)
|
|
|
|
|
|
|
|
if file_running?(file, socket.assigns.sessions) do
|
|
|
|
session_id = session_id_by_file(file, socket.assigns.sessions)
|
|
|
|
{:noreply, push_navigate(socket, to: ~p"/sessions/#{session_id}")}
|
|
|
|
else
|
|
|
|
{:noreply, open_notebook(socket, file)}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_params(_params, _url, socket), do: {:noreply, socket}
|
|
|
|
|
|
|
|
@impl true
|
2023-03-28 02:59:12 +08:00
|
|
|
def handle_event("hide_recent_notebook", %{"idx" => idx}, socket) do
|
2023-05-11 00:23:08 +08:00
|
|
|
on_confirm = fn socket ->
|
|
|
|
%{file: file} = Enum.fetch!(socket.assigns.recent_notebooks, idx)
|
|
|
|
Livebook.NotebookManager.remove_recent_notebook(file)
|
|
|
|
socket
|
|
|
|
end
|
|
|
|
|
|
|
|
{:noreply,
|
|
|
|
confirm(socket, on_confirm,
|
|
|
|
title: "Hide notebook",
|
|
|
|
description: "The notebook will reappear here when you open it again.",
|
|
|
|
confirm_text: "Hide",
|
|
|
|
opt_out_id: "hide-notebook"
|
|
|
|
)}
|
2023-03-28 02:59:12 +08:00
|
|
|
end
|
|
|
|
|
2023-03-07 01:14:33 +08:00
|
|
|
@impl true
|
|
|
|
def handle_info({type, session} = event, socket)
|
|
|
|
when type in [:session_created, :session_updated, :session_closed] and
|
|
|
|
session.mode == :default do
|
|
|
|
{:noreply, update(socket, :sessions, &update_session_list(&1, event))}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info({:fork, file}, socket) do
|
|
|
|
{:noreply, fork_notebook(socket, file)}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info({:open, file}, socket) do
|
|
|
|
{:noreply, open_notebook(socket, file)}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info({:import_source, source, session_opts}, socket) do
|
|
|
|
socket = import_source(socket, source, session_opts)
|
|
|
|
{:noreply, socket}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info({:recent_notebooks_updated, recent_notebooks}, socket) do
|
|
|
|
{:noreply, assign(socket, recent_notebooks: recent_notebooks)}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info(_message, socket), do: {:noreply, socket}
|
|
|
|
|
|
|
|
defp file_from_params(%{"autosave" => _} = _params) do
|
|
|
|
Livebook.Settings.autosave_path()
|
|
|
|
|> FileSystem.Utils.ensure_dir_path()
|
|
|
|
|> FileSystem.File.local()
|
|
|
|
end
|
|
|
|
|
|
|
|
defp file_from_params(%{"path" => path} = _params) do
|
|
|
|
path = Path.expand(path)
|
|
|
|
|
|
|
|
cond do
|
|
|
|
File.dir?(path) ->
|
|
|
|
path
|
|
|
|
|> FileSystem.Utils.ensure_dir_path()
|
|
|
|
|> FileSystem.File.local()
|
|
|
|
|
|
|
|
File.regular?(path) ->
|
|
|
|
FileSystem.File.local(path)
|
|
|
|
|
|
|
|
true ->
|
|
|
|
Livebook.Config.local_file_system_home()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-07-12 19:28:51 +08:00
|
|
|
defp file_from_params(_params), do: Livebook.Settings.default_dir()
|
2023-03-07 01:14:33 +08:00
|
|
|
|
|
|
|
defp import_source(socket, source, session_opts) do
|
2023-07-27 04:39:33 +08:00
|
|
|
{notebook, messages} = Livebook.LiveMarkdown.notebook_from_livemd(source)
|
2023-03-07 01:14:33 +08:00
|
|
|
|
|
|
|
socket =
|
|
|
|
socket
|
|
|
|
|> put_import_warnings(messages)
|
|
|
|
|> put_flash(
|
|
|
|
:info,
|
|
|
|
"You have imported a notebook, no code has been executed so far. You should read and evaluate code as needed."
|
|
|
|
)
|
|
|
|
|
|
|
|
session_opts = Keyword.merge(session_opts, notebook: notebook)
|
|
|
|
create_session(socket, session_opts)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp file_running?(file, sessions) do
|
|
|
|
Enum.any?(sessions, &(&1.file == file))
|
|
|
|
end
|
|
|
|
|
|
|
|
defp session_id_by_file(file, sessions) do
|
|
|
|
session = Enum.find(sessions, &(&1.file == file))
|
|
|
|
session.id
|
|
|
|
end
|
|
|
|
end
|