2023-02-28 22:08:49 +08:00
|
|
|
defmodule LivebookWeb.AppsLive do
|
|
|
|
use LivebookWeb, :live_view
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def mount(_params, _session, socket) do
|
|
|
|
if connected?(socket) do
|
2023-05-20 01:40:56 +08:00
|
|
|
Livebook.Apps.subscribe()
|
2023-02-28 22:08:49 +08:00
|
|
|
end
|
|
|
|
|
2023-05-20 01:40:56 +08:00
|
|
|
apps = Livebook.Apps.list_apps()
|
2023-11-06 16:08:28 +08:00
|
|
|
empty_apps_path? = Livebook.Apps.empty_apps_path?()
|
2023-02-28 22:08:49 +08:00
|
|
|
|
2023-11-06 16:08:28 +08:00
|
|
|
{:ok, assign(socket, apps: apps, empty_apps_path?: empty_apps_path?)}
|
2023-02-28 22:08:49 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def render(assigns) do
|
|
|
|
~H"""
|
2023-11-06 16:08:28 +08:00
|
|
|
<div class="h-full flex flex-col overflow-y-auto">
|
|
|
|
<div class="px-4 py-3 flex items-center justify-between">
|
|
|
|
<div class="w-10 h-10">
|
|
|
|
<.link navigate={~p"/"}>
|
|
|
|
<img src={~p"/images/logo.png"} height="40" widthz="40" alt="logo livebook" />
|
|
|
|
</.link>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<.link navigate={~p"/apps-dashboard"} class="flex items-center text-blue-600">
|
|
|
|
<span class="font-semibold">Dashboard</span>
|
|
|
|
<.remix_icon icon="arrow-right-line" class="align-middle ml-1" />
|
|
|
|
</.link>
|
2023-03-07 01:14:33 +08:00
|
|
|
</div>
|
2023-02-28 23:32:48 +08:00
|
|
|
</div>
|
2023-11-06 16:08:28 +08:00
|
|
|
<div class="w-full max-w-screen-lg px-4 md:px-20 py-4 mx-auto">
|
|
|
|
<div class="flex flex-col items-center">
|
|
|
|
<h1 class="text-2xl text-gray-800 font-medium">
|
|
|
|
Apps
|
|
|
|
</h1>
|
|
|
|
<div :if={@apps != []} class="w-full mt-5 max-w-[400px]">
|
|
|
|
<div class="w-full flex flex-col space-y-4">
|
|
|
|
<.link
|
|
|
|
:for={app <- apps_listing(@apps)}
|
|
|
|
navigate={~p"/apps/#{app.slug}"}
|
|
|
|
class="px-4 py-3 border border-gray-200 rounded-xl text-gray-800 pointer hover:bg-gray-50 flex items-center justify-between"
|
|
|
|
>
|
|
|
|
<span class="font-semibold"><%= app.notebook_name %></span>
|
|
|
|
<.remix_icon :if={not app.public?} icon="lock-password-line" />
|
|
|
|
</.link>
|
2023-07-05 17:15:35 +08:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div
|
2023-11-06 16:08:28 +08:00
|
|
|
:if={@apps == [] and not @empty_apps_path?}
|
|
|
|
class="mt-5 flex flex-col w-full max-w-[400px]"
|
2023-05-20 01:40:56 +08:00
|
|
|
>
|
2023-11-06 16:08:28 +08:00
|
|
|
<.no_entries :if={@apps == []}>
|
|
|
|
No apps running.
|
|
|
|
</.no_entries>
|
2023-07-05 17:15:35 +08:00
|
|
|
</div>
|
2023-11-06 16:08:28 +08:00
|
|
|
<div :if={@apps == [] and @empty_apps_path?} class="mt-5 text-gray-600">
|
|
|
|
<div>
|
|
|
|
No app notebooks found. Follow these steps to list your apps here:
|
2023-07-05 17:15:35 +08:00
|
|
|
</div>
|
2023-11-06 16:08:28 +08:00
|
|
|
<ol class="mt-4 pl-4 flex flex-col space-y-1 list-decimal list-inside">
|
|
|
|
<li>
|
|
|
|
Open a notebook
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
Click <.remix_icon icon="rocket-line" class="align-baseline text-lg" />
|
|
|
|
in the sidebar and configure the app as public
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
Save the notebook to the
|
|
|
|
<span class="font-medium"><%= Livebook.Config.apps_path() %></span>
|
|
|
|
folder
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
Relaunch your Livebook app
|
|
|
|
</li>
|
|
|
|
</ol>
|
2023-07-05 17:15:35 +08:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
2023-02-28 22:08:49 +08:00
|
|
|
</div>
|
|
|
|
"""
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
2023-05-20 01:40:56 +08:00
|
|
|
def handle_info({type, _app} = event, socket)
|
|
|
|
when type in [:app_created, :app_updated, :app_closed] do
|
2023-11-06 16:08:28 +08:00
|
|
|
{:noreply, update(socket, :apps, &LivebookWeb.AppHelpers.update_app_list(&1, event))}
|
2023-02-28 22:08:49 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info(_message, socket), do: {:noreply, socket}
|
|
|
|
|
2023-11-06 16:08:28 +08:00
|
|
|
defp apps_listing(apps) do
|
|
|
|
Enum.sort_by(apps, & &1.notebook_name)
|
2023-02-28 22:08:49 +08:00
|
|
|
end
|
|
|
|
end
|