2021-03-04 05:56:28 +08:00
|
|
|
defmodule LivebookWeb.HomeLive do
|
|
|
|
use LivebookWeb, :live_view
|
2021-01-08 05:13:17 +08:00
|
|
|
|
2021-06-03 03:51:43 +08:00
|
|
|
import LivebookWeb.SessionHelpers
|
2021-11-03 05:34:44 +08:00
|
|
|
import LivebookWeb.UserHelpers
|
2021-05-04 02:03:19 +08:00
|
|
|
|
2021-07-07 20:32:49 +08:00
|
|
|
alias LivebookWeb.{SidebarHelpers, ExploreHelpers}
|
2021-09-05 01:16:01 +08:00
|
|
|
alias Livebook.{Sessions, Session, LiveMarkdown, Notebook, FileSystem}
|
2021-02-21 23:54:44 +08:00
|
|
|
|
|
|
|
@impl true
|
2021-11-02 02:33:43 +08:00
|
|
|
def mount(_params, _session, socket) do
|
2021-02-21 23:54:44 +08:00
|
|
|
if connected?(socket) do
|
2021-09-05 01:16:01 +08:00
|
|
|
Phoenix.PubSub.subscribe(Livebook.PubSub, "tracker_sessions")
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-11-11 01:50:39 +08:00
|
|
|
sessions = Sessions.list_sessions()
|
2021-11-03 05:32:58 +08:00
|
|
|
notebook_infos = Notebook.Explore.visible_notebook_infos() |> Enum.take(3)
|
2021-02-21 23:54:44 +08:00
|
|
|
|
2021-05-04 02:03:19 +08:00
|
|
|
{:ok,
|
2022-01-13 19:22:34 +08:00
|
|
|
socket
|
|
|
|
|> SidebarHelpers.shared_home_handlers()
|
|
|
|
|> assign(
|
2021-08-14 03:17:43 +08:00
|
|
|
file: Livebook.Config.default_dir(),
|
|
|
|
file_info: %{exists: true, access: :read_write},
|
2021-09-05 01:16:01 +08:00
|
|
|
sessions: sessions,
|
2022-01-07 01:37:55 +08:00
|
|
|
notebook_infos: notebook_infos,
|
|
|
|
page_title: "Livebook"
|
2021-05-04 02:03:19 +08:00
|
|
|
)}
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-01-08 05:13:17 +08:00
|
|
|
@impl true
|
|
|
|
def render(assigns) do
|
2021-07-07 20:32:49 +08:00
|
|
|
~H"""
|
2021-12-30 05:06:19 +08:00
|
|
|
<div class="flex grow h-full">
|
2021-07-07 20:32:49 +08:00
|
|
|
<SidebarHelpers.sidebar>
|
2022-01-13 19:22:34 +08:00
|
|
|
<SidebarHelpers.shared_home_footer
|
|
|
|
socket={@socket}
|
|
|
|
current_user={@current_user}
|
|
|
|
user_path={Routes.home_path(@socket, :user)} />
|
2021-07-07 20:32:49 +08:00
|
|
|
</SidebarHelpers.sidebar>
|
2021-12-30 05:06:19 +08:00
|
|
|
<div class="grow px-6 py-8 overflow-y-auto">
|
2021-06-03 03:51:43 +08:00
|
|
|
<div class="max-w-screen-lg w-full mx-auto px-4 pb-8 space-y-4">
|
2021-07-07 20:32:49 +08:00
|
|
|
<div class="flex flex-col space-y-2 items-center pb-4 border-b border-gray-200
|
|
|
|
sm:flex-row sm:space-y-0 sm:justify-between">
|
2021-03-20 21:10:15 +08:00
|
|
|
<div class="text-2xl text-gray-800 font-semibold">
|
2021-06-03 03:51:43 +08:00
|
|
|
<img src="/images/logo-with-text.png" class="h-[50px]" alt="Livebook" />
|
2021-03-20 21:10:15 +08:00
|
|
|
</div>
|
2021-04-04 18:42:46 +08:00
|
|
|
<div class="flex space-x-2 pt-2">
|
2021-07-07 20:32:49 +08:00
|
|
|
<%= live_patch "Import",
|
|
|
|
to: Routes.home_path(@socket, :import, "url"),
|
2021-12-04 04:57:21 +08:00
|
|
|
class: "button-base button-outlined-gray whitespace-nowrap" %>
|
|
|
|
<button class="button-base button-blue" phx-click="new">
|
2021-03-31 03:42:02 +08:00
|
|
|
New notebook
|
|
|
|
</button>
|
|
|
|
</div>
|
2021-03-20 21:10:15 +08:00
|
|
|
</div>
|
2021-07-07 20:32:49 +08:00
|
|
|
|
2021-06-03 03:51:43 +08:00
|
|
|
<div class="h-80">
|
2021-11-02 02:33:43 +08:00
|
|
|
<.live_component module={LivebookWeb.FileSelectComponent}
|
|
|
|
id="home-file-select"
|
|
|
|
file={@file}
|
|
|
|
extnames={[LiveMarkdown.extension()]}
|
|
|
|
running_files={files(@sessions)}>
|
2021-03-20 21:10:15 +08:00
|
|
|
<div class="flex justify-end space-x-2">
|
2021-12-04 04:57:21 +08:00
|
|
|
<button class="button-base button-outlined-gray whitespace-nowrap"
|
2021-07-07 20:32:49 +08:00
|
|
|
phx-click="fork"
|
2021-08-14 03:17:43 +08:00
|
|
|
disabled={not path_forkable?(@file, @file_info)}>
|
2021-07-07 20:32:49 +08:00
|
|
|
<.remix_icon icon="git-branch-line" class="align-middle mr-1" />
|
2021-03-20 21:10:15 +08:00
|
|
|
<span>Fork</span>
|
2021-07-07 20:32:49 +08:00
|
|
|
</button>
|
2021-09-05 01:16:01 +08:00
|
|
|
<%= if file_running?(@file, @sessions) do %>
|
2021-07-07 20:32:49 +08:00
|
|
|
<%= live_redirect "Join session",
|
2021-09-05 01:16:01 +08:00
|
|
|
to: Routes.session_path(@socket, :page, session_id_by_file(@file, @sessions)),
|
2021-12-04 04:57:21 +08:00
|
|
|
class: "button-base button-blue" %>
|
2021-03-20 21:10:15 +08:00
|
|
|
<% else %>
|
2021-08-14 03:17:43 +08:00
|
|
|
<span {open_button_tooltip_attrs(@file, @file_info)}>
|
2021-12-04 04:57:21 +08:00
|
|
|
<button class="button-base button-blue"
|
2021-07-07 20:32:49 +08:00
|
|
|
phx-click="open"
|
2021-09-05 01:16:01 +08:00
|
|
|
disabled={not path_openable?(@file, @file_info, @sessions)}>
|
2021-07-07 20:32:49 +08:00
|
|
|
Open
|
|
|
|
</button>
|
2021-04-14 22:27:35 +08:00
|
|
|
</span>
|
2021-03-20 21:10:15 +08:00
|
|
|
<% end %>
|
|
|
|
</div>
|
2021-11-02 02:33:43 +08:00
|
|
|
</.live_component>
|
2021-03-20 21:10:15 +08:00
|
|
|
</div>
|
2021-07-07 20:32:49 +08:00
|
|
|
|
2021-06-03 03:51:43 +08:00
|
|
|
<div class="py-12">
|
|
|
|
<div class="mb-4 flex justify-between items-center">
|
2021-11-11 01:50:39 +08:00
|
|
|
<h2 class="uppercase font-semibold text-gray-500">
|
2021-06-03 03:51:43 +08:00
|
|
|
Explore
|
|
|
|
</h2>
|
|
|
|
<%= live_redirect to: Routes.explore_path(@socket, :page),
|
|
|
|
class: "flex items-center text-blue-600" do %>
|
|
|
|
<span class="font-semibold">See all</span>
|
2021-07-07 20:32:49 +08:00
|
|
|
<.remix_icon icon="arrow-right-line" class="align-middle ml-1" />
|
2021-06-03 03:51:43 +08:00
|
|
|
<% end %>
|
|
|
|
</div>
|
|
|
|
<div class="grid grid-cols-1 md:grid-cols-3 gap-4">
|
2021-07-07 20:32:49 +08:00
|
|
|
<%# Note: it's fine to use stateless components in this comprehension,
|
|
|
|
because @notebook_infos never change %>
|
|
|
|
<%= for info <- @notebook_infos do %>
|
|
|
|
<ExploreHelpers.notebook_card notebook_info={info} socket={@socket} />
|
2021-06-03 03:51:43 +08:00
|
|
|
<% end %>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div class="py-12">
|
2021-11-11 01:50:39 +08:00
|
|
|
<.live_component module={LivebookWeb.HomeLive.SessionListComponent}
|
|
|
|
id="session-list"
|
|
|
|
sessions={@sessions} />
|
2021-03-20 21:10:15 +08:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
2021-01-08 05:13:17 +08:00
|
|
|
</div>
|
2021-04-02 20:00:49 +08:00
|
|
|
|
2021-05-04 02:03:19 +08:00
|
|
|
<%= if @live_action == :user do %>
|
2021-11-03 05:34:44 +08:00
|
|
|
<.current_user_modal
|
|
|
|
return_to={Routes.home_path(@socket, :page)}
|
|
|
|
current_user={@current_user} />
|
2021-05-04 02:03:19 +08:00
|
|
|
<% end %>
|
|
|
|
|
2021-04-13 05:24:26 +08:00
|
|
|
<%= if @live_action == :close_session do %>
|
2021-11-03 05:34:44 +08:00
|
|
|
<.modal class="w-full max-w-xl" return_to={Routes.home_path(@socket, :page)}>
|
|
|
|
<.live_component module={LivebookWeb.HomeLive.CloseSessionComponent}
|
|
|
|
id="close-session"
|
|
|
|
return_to={Routes.home_path(@socket, :page)}
|
|
|
|
session={@session} />
|
|
|
|
</.modal>
|
2021-04-02 20:00:49 +08:00
|
|
|
<% end %>
|
2021-04-23 23:40:13 +08:00
|
|
|
|
|
|
|
<%= if @live_action == :import do %>
|
2021-11-03 05:34:44 +08:00
|
|
|
<.modal class="w-full max-w-xl" return_to={Routes.home_path(@socket, :page)}>
|
|
|
|
<.live_component module={LivebookWeb.HomeLive.ImportComponent}
|
|
|
|
id="import"
|
|
|
|
tab={@tab}
|
|
|
|
import_opts={@import_opts} />
|
|
|
|
</.modal>
|
2021-04-23 23:40:13 +08:00
|
|
|
<% end %>
|
2021-01-08 05:13:17 +08:00
|
|
|
"""
|
|
|
|
end
|
2021-02-21 23:54:44 +08:00
|
|
|
|
2021-08-14 03:17:43 +08:00
|
|
|
defp open_button_tooltip_attrs(file, file_info) do
|
|
|
|
if regular?(file, file_info) and not writable?(file_info) do
|
2021-11-02 01:20:56 +08:00
|
|
|
[class: "tooltip top", data_tooltip: "This file is write-protected, please fork instead"]
|
2021-07-07 20:32:49 +08:00
|
|
|
else
|
|
|
|
[]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-03-20 21:10:15 +08:00
|
|
|
@impl true
|
2021-03-26 00:39:18 +08:00
|
|
|
def handle_params(%{"session_id" => session_id}, _url, socket) do
|
2021-09-05 01:16:01 +08:00
|
|
|
session = Enum.find(socket.assigns.sessions, &(&1.id == session_id))
|
|
|
|
{:noreply, assign(socket, session: session)}
|
2021-03-26 00:39:18 +08:00
|
|
|
end
|
|
|
|
|
2021-10-16 18:23:08 +08:00
|
|
|
def handle_params(%{"tab" => tab} = params, _url, %{assigns: %{live_action: :import}} = socket) do
|
|
|
|
import_opts = [url: params["url"]]
|
|
|
|
{:noreply, assign(socket, tab: tab, import_opts: import_opts)}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_params(%{"url" => url}, _url, %{assigns: %{live_action: :public_import}} = socket) do
|
2021-11-12 22:49:22 +08:00
|
|
|
origin = Livebook.ContentLoader.url_to_location(url)
|
|
|
|
|
|
|
|
origin
|
|
|
|
|> Livebook.ContentLoader.fetch_content_from_location()
|
2021-10-16 18:23:08 +08:00
|
|
|
|> case do
|
|
|
|
{:ok, content} ->
|
2021-11-12 22:49:22 +08:00
|
|
|
socket = import_content(socket, content, origin: origin)
|
2021-10-16 18:23:08 +08:00
|
|
|
{:noreply, socket}
|
|
|
|
|
|
|
|
{:error, _message} ->
|
|
|
|
{:noreply, push_patch(socket, to: Routes.home_path(socket, :import, "url", url: url))}
|
|
|
|
end
|
2021-04-23 23:40:13 +08:00
|
|
|
end
|
|
|
|
|
2021-03-20 21:10:15 +08:00
|
|
|
def handle_params(_params, _url, socket), do: {:noreply, socket}
|
|
|
|
|
2021-02-21 23:54:44 +08:00
|
|
|
@impl true
|
|
|
|
def handle_event("new", %{}, socket) do
|
2021-06-03 03:51:43 +08:00
|
|
|
{:noreply, create_session(socket)}
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def handle_event("fork", %{}, socket) do
|
2021-08-14 03:17:43 +08:00
|
|
|
file = socket.assigns.file
|
|
|
|
|
|
|
|
socket =
|
|
|
|
case import_notebook(file) do
|
|
|
|
{:ok, {notebook, messages}} ->
|
|
|
|
notebook = Notebook.forked(notebook)
|
|
|
|
images_dir = Session.images_dir_for_notebook(file)
|
|
|
|
|
|
|
|
socket
|
2021-10-22 05:21:54 +08:00
|
|
|
|> put_import_warnings(messages)
|
2021-08-14 03:17:43 +08:00
|
|
|
|> create_session(
|
|
|
|
notebook: notebook,
|
|
|
|
copy_images_from: images_dir,
|
|
|
|
origin: {:file, file}
|
|
|
|
)
|
|
|
|
|
|
|
|
{:error, error} ->
|
|
|
|
put_flash(socket, :error, Livebook.Utils.upcase_first(error))
|
|
|
|
end
|
2021-07-11 03:49:50 +08:00
|
|
|
|
2021-08-14 03:17:43 +08:00
|
|
|
{:noreply, socket}
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def handle_event("open", %{}, socket) do
|
2021-08-14 03:17:43 +08:00
|
|
|
file = socket.assigns.file
|
2021-07-11 03:49:50 +08:00
|
|
|
|
2021-08-14 03:17:43 +08:00
|
|
|
socket =
|
|
|
|
case import_notebook(file) do
|
|
|
|
{:ok, {notebook, messages}} ->
|
|
|
|
socket
|
2021-10-22 05:21:54 +08:00
|
|
|
|> put_import_warnings(messages)
|
2021-08-14 03:17:43 +08:00
|
|
|
|> create_session(notebook: notebook, file: file, origin: {:file, file})
|
|
|
|
|
|
|
|
{:error, error} ->
|
|
|
|
put_flash(socket, :error, Livebook.Utils.upcase_first(error))
|
|
|
|
end
|
|
|
|
|
|
|
|
{:noreply, socket}
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-03-26 02:04:49 +08:00
|
|
|
def handle_event("fork_session", %{"id" => session_id}, socket) do
|
2021-09-05 01:16:01 +08:00
|
|
|
session = Enum.find(socket.assigns.sessions, &(&1.id == session_id))
|
|
|
|
%{images_dir: images_dir} = session
|
|
|
|
data = Session.get_data(session.pid)
|
2021-04-22 05:02:09 +08:00
|
|
|
notebook = Notebook.forked(data.notebook)
|
2021-07-11 03:49:50 +08:00
|
|
|
|
2021-08-14 03:17:43 +08:00
|
|
|
origin =
|
|
|
|
if data.file do
|
|
|
|
{:file, data.file}
|
2021-07-11 03:49:50 +08:00
|
|
|
else
|
2021-08-14 03:17:43 +08:00
|
|
|
data.origin
|
2021-07-11 03:49:50 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
{:noreply,
|
|
|
|
create_session(socket,
|
|
|
|
notebook: notebook,
|
|
|
|
copy_images_from: images_dir,
|
2021-08-14 03:17:43 +08:00
|
|
|
origin: origin
|
2021-07-11 03:49:50 +08:00
|
|
|
)}
|
2021-03-26 02:04:49 +08:00
|
|
|
end
|
|
|
|
|
2021-12-04 23:29:14 +08:00
|
|
|
def handle_event("open_autosave_directory", %{}, socket) do
|
|
|
|
file =
|
|
|
|
Livebook.Config.autosave_path()
|
|
|
|
|> FileSystem.Utils.ensure_dir_path()
|
|
|
|
|> FileSystem.File.local()
|
|
|
|
|
|
|
|
file_info = %{exists: true, access: file_access(file)}
|
|
|
|
{:noreply, assign(socket, file: file, file_info: file_info)}
|
|
|
|
end
|
|
|
|
|
2021-02-21 23:54:44 +08:00
|
|
|
@impl true
|
2021-08-14 03:17:43 +08:00
|
|
|
def handle_info({:set_file, file, info}, socket) do
|
2021-12-04 23:29:14 +08:00
|
|
|
file_info = %{exists: info.exists, access: file_access(file)}
|
2021-08-14 03:17:43 +08:00
|
|
|
{:noreply, assign(socket, file: file, file_info: file_info)}
|
|
|
|
end
|
|
|
|
|
2021-09-05 01:16:01 +08:00
|
|
|
def handle_info({:session_created, session}, socket) do
|
|
|
|
if session in socket.assigns.sessions do
|
|
|
|
{:noreply, socket}
|
|
|
|
else
|
2021-11-11 01:50:39 +08:00
|
|
|
{:noreply, assign(socket, sessions: [session | socket.assigns.sessions])}
|
2021-09-05 01:16:01 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info({:session_updated, session}, socket) do
|
|
|
|
sessions =
|
|
|
|
Enum.map(socket.assigns.sessions, fn other ->
|
|
|
|
if other.id == session.id, do: session, else: other
|
|
|
|
end)
|
|
|
|
|
|
|
|
{:noreply, assign(socket, sessions: sessions)}
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-09-05 01:16:01 +08:00
|
|
|
def handle_info({:session_closed, session}, socket) do
|
|
|
|
sessions = Enum.reject(socket.assigns.sessions, &(&1.id == session.id))
|
|
|
|
{:noreply, assign(socket, sessions: sessions)}
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-07-11 03:49:50 +08:00
|
|
|
def handle_info({:import_content, content, session_opts}, socket) do
|
2021-10-16 18:23:08 +08:00
|
|
|
socket = import_content(socket, content, session_opts)
|
|
|
|
{:noreply, socket}
|
2021-04-23 23:40:13 +08:00
|
|
|
end
|
|
|
|
|
2021-02-21 23:54:44 +08:00
|
|
|
def handle_info(_message, socket), do: {:noreply, socket}
|
|
|
|
|
2021-09-05 01:16:01 +08:00
|
|
|
defp files(sessions) do
|
|
|
|
Enum.map(sessions, & &1.file)
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-08-14 03:17:43 +08:00
|
|
|
defp path_forkable?(file, file_info) do
|
|
|
|
regular?(file, file_info)
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-09-05 01:16:01 +08:00
|
|
|
defp path_openable?(file, file_info, sessions) do
|
|
|
|
regular?(file, file_info) and not file_running?(file, sessions) and
|
2021-08-14 03:17:43 +08:00
|
|
|
writable?(file_info)
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-08-14 03:17:43 +08:00
|
|
|
defp regular?(file, file_info) do
|
|
|
|
file_info.exists and not FileSystem.File.dir?(file)
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-08-14 03:17:43 +08:00
|
|
|
defp writable?(file_info) do
|
|
|
|
file_info.access in [:read_write, :write]
|
2021-04-14 22:27:35 +08:00
|
|
|
end
|
|
|
|
|
2021-09-05 01:16:01 +08:00
|
|
|
defp file_running?(file, sessions) do
|
|
|
|
running_files = files(sessions)
|
2021-08-14 03:17:43 +08:00
|
|
|
file in running_files
|
|
|
|
end
|
|
|
|
|
|
|
|
defp import_notebook(file) do
|
|
|
|
with {:ok, content} <- FileSystem.File.read(file) do
|
|
|
|
{:ok, LiveMarkdown.Import.notebook_from_markdown(content)}
|
|
|
|
end
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
|
|
|
|
2021-09-05 01:16:01 +08:00
|
|
|
defp session_id_by_file(file, sessions) do
|
|
|
|
session = Enum.find(sessions, &(&1.file == file))
|
|
|
|
session.id
|
2021-02-21 23:54:44 +08:00
|
|
|
end
|
2021-10-10 18:29:45 +08:00
|
|
|
|
2021-10-16 18:23:08 +08:00
|
|
|
defp import_content(socket, content, session_opts) do
|
|
|
|
{notebook, messages} = Livebook.LiveMarkdown.Import.notebook_from_markdown(content)
|
2021-10-22 05:21:54 +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."
|
|
|
|
)
|
|
|
|
|
2021-10-16 18:23:08 +08:00
|
|
|
session_opts = Keyword.merge(session_opts, notebook: notebook)
|
|
|
|
create_session(socket, session_opts)
|
|
|
|
end
|
2021-12-04 23:29:14 +08:00
|
|
|
|
|
|
|
defp file_access(file) do
|
|
|
|
case FileSystem.File.access(file) do
|
|
|
|
{:ok, access} -> access
|
|
|
|
{:error, _} -> :none
|
|
|
|
end
|
|
|
|
end
|
2021-01-08 05:13:17 +08:00
|
|
|
end
|