defmodule BDS.Desktop.ShellLive do @moduledoc false use Phoenix.LiveView import Phoenix.HTML alias BDS.Desktop.ShellData alias BDS.UI.Commands alias BDS.UI.Registry alias BDS.UI.Workbench @refresh_interval 1_500 embed_templates "shell_live/*" @impl true def mount(_params, _session, socket) do if connected?(socket) do :timer.send_interval(@refresh_interval, :refresh_task_status) end workbench = Workbench.new() {:ok, socket |> assign(:page_title, ShellData.title()) |> assign(:page_language, ShellData.ui_language()) |> assign(:offline_mode, true) |> assign(:tab_meta, %{}) |> reload_shell(workbench)} end @impl true def handle_event("toggle_sidebar", _params, socket) do {:noreply, reload_shell(socket, Workbench.toggle_sidebar(socket.assigns.workbench))} end def handle_event("toggle_panel", _params, socket) do {:noreply, reload_shell(socket, Workbench.toggle_panel(socket.assigns.workbench))} end def handle_event("toggle_assistant_sidebar", _params, socket) do {:noreply, reload_shell(socket, Workbench.toggle_assistant_sidebar(socket.assigns.workbench))} end def handle_event("select_view", %{"view" => view_id}, socket) do workbench = Workbench.click_activity(socket.assigns.workbench, String.to_existing_atom(view_id)) {:noreply, reload_shell(socket, workbench)} end def handle_event("select_panel_tab", %{"tab" => tab}, socket) do workbench = socket.assigns.workbench |> Workbench.set_panel_visible(true) |> Workbench.set_panel_tab(String.to_existing_atom(tab)) {:noreply, reload_shell(socket, workbench)} end def handle_event("open_sidebar_item", %{"route" => _route, "id" => _id} = params, socket) do {:noreply, open_sidebar_item(socket, params, :preview)} end def handle_event("pin_sidebar_item", %{"route" => _route, "id" => _id} = params, socket) do {:noreply, open_sidebar_item(socket, params, :pin)} end def handle_event("sync_layout", params, socket) do {:noreply, reload_shell(socket, sync_layout(socket.assigns.workbench, params))} end def handle_event("resize_panel", %{"target" => target, "width" => width}, socket) do {:noreply, reload_shell(socket, resize_panel(socket.assigns.workbench, target, width))} end def handle_event("shortcut", params, socket) do if ignore_shortcut?(params) do {:noreply, socket} else {:noreply, reload_shell(socket, Commands.handle_shortcut(socket.assigns.workbench, params))} end end def handle_event("select_tab", %{"type" => type, "id" => id}, socket) do workbench = Workbench.open_tab(socket.assigns.workbench, String.to_existing_atom(type), id, :preview) {:noreply, reload_shell(socket, workbench)} end def handle_event("close_tab", %{"type" => type, "id" => id}, socket) do type_atom = String.to_existing_atom(type) workbench = Workbench.close_tab(socket.assigns.workbench, type_atom, id) tab_meta = Map.delete(socket.assigns.tab_meta, {type_atom, id}) {:noreply, socket |> assign(:tab_meta, tab_meta) |> reload_shell(workbench)} end def handle_event("toggle_offline_mode", _params, socket) do socket = assign(socket, :offline_mode, not socket.assigns.offline_mode) {:noreply, reload_shell(socket, socket.assigns.workbench)} end @impl true def handle_info(:refresh_task_status, socket) do task_status = BDS.Tasks.status_snapshot() {:noreply, socket |> assign(:task_status, task_status) |> assign(:editor_meta, ShellData.editor_meta(task_status)) |> assign( :status, ShellData.status_bar(socket.assigns.workbench, task_status, socket.assigns.dashboard, ui_language: socket.assigns.page_language, offline_mode: socket.assigns.offline_mode ) )} end @impl true def render(assigns), do: index(assigns) defp reload_shell(socket, workbench) do projects = ShellData.project_snapshot() dashboard = ShellData.dashboard(projects.active_project_id) sidebar_data = ShellData.sidebar_view(projects.active_project_id, Atom.to_string(workbench.active_view)) task_status = BDS.Tasks.status_snapshot() activity_buttons = Workbench.activity_buttons(workbench, 0) page_language = socket.assigns[:page_language] || ShellData.ui_language() offline_mode = Map.get(socket.assigns, :offline_mode, true) socket |> assign(:workbench, workbench) |> assign(:projects, projects) |> assign(:current_project, ShellData.current_project(projects)) |> assign(:dashboard, dashboard) |> assign(:dashboard_timeline_entries, Map.get(dashboard, :timeline_entries, [])) |> assign(:dashboard_category_counts, Map.get(dashboard, :category_counts, [])) |> assign(:dashboard_recent_posts, Map.get(dashboard, :recent_posts, [])) |> assign(:dashboard_tag_cloud_items, ShellData.dashboard_tag_cloud_items(Map.get(dashboard, :tag_cloud_items, []))) |> assign(:sidebar_data, sidebar_data) |> assign(:sidebar_header, active_sidebar_label(activity_buttons, workbench.active_view, sidebar_data)) |> assign(:assistant_cards, ShellData.assistant_cards()) |> assign(:editor_meta, ShellData.editor_meta(task_status)) |> assign(:task_status, task_status) |> assign( :status, ShellData.status_bar(workbench, task_status, dashboard, ui_language: page_language, offline_mode: offline_mode ) ) |> assign(:activity_buttons, activity_buttons) |> assign(:panel_tabs, ShellData.panel_tabs(workbench)) |> assign(:supported_ui_languages, ShellData.supported_ui_languages()) |> assign(:current_tab, current_tab(workbench)) end defp render_sidebar_body(assigns) do case assigns.sidebar_data.layout do "post_list" -> render_post_sidebar(assigns) "media_grid" -> render_media_sidebar(assigns) "entity_list" -> render_entity_sidebar(assigns) "nav_list" -> render_nav_sidebar(assigns) _other -> render_default_sidebar(assigns) end end defp render_post_sidebar(assigns) do ~H""" <%= for section <- Map.get(@sidebar_data, :sections, []) do %> <% end %> <%= if Enum.empty?(Map.get(@sidebar_data, :sections, [])) do %> <% end %> """ end defp render_media_sidebar(assigns) do ~H""" <%= if Enum.any?(Map.get(@sidebar_data, :items, [])) do %> <% else %> <% end %> """ end defp render_entity_sidebar(assigns) do ~H""" <%= if Enum.any?(Map.get(@sidebar_data, :items, [])) do %>
<%= for item <- Map.get(@sidebar_data, :items, []) do %> <% end %>
<% else %> <% end %> """ end defp render_nav_sidebar(assigns) do ~H"""
<%= for item <- Map.get(@sidebar_data, :items, []) do %> <% end %>
""" end defp render_default_sidebar(assigns) do ~H""" <%= for section <- Map.get(@sidebar_data, :sections, []) do %> <% end %> """ end defp render_panel_body(assigns) do case assigns.workbench.panel.active_tab do :tasks -> render_task_entries(assigns) :output -> render_output_entries(assigns) :git_log -> render_git_log(assigns) other -> render_generic_panel(assigns, other) end end defp render_task_entries(assigns) do ~H""" <%= if Enum.empty?(Map.get(@task_status, :tasks, [])) do %>
<%= translated("Tasks") %> <%= translated("No background tasks running") %>
<% else %>
<%= for task <- Map.get(@task_status, :tasks, []) do %>
<%= task.name %> <%= task.status |> to_string() |> String.capitalize() %>
<%= task.message || task.group_name || "" %>
<% end %>
<% end %> """ end defp render_output_entries(assigns) do ~H"""
<%= translated("Output") %> <%= translated("No shell output yet") %>
""" end defp render_git_log(assigns) do ~H"""
<%= translated("Git Log") %> <%= translated("Working tree integration is not wired yet in the shell, but the tab is selectable and ready for command output.") %>
""" end defp render_generic_panel(assigns, tab) do assigns = assign(assigns, :panel_label, ShellData.route_label(tab)) ~H"""
<%= @panel_label %> <%= translated("The shared lower panel is available for tasks, output, git details, and editor-specific diagnostics.") %>
""" end defp translated(text, bindings \\ %{}), do: ShellData.translate(text, bindings) defp panel_tab_label(:tasks), do: translated("Tasks") defp panel_tab_label(:output), do: translated("Output") defp panel_tab_label(:git_log), do: translated("Git Log") defp panel_tab_label(tab), do: ShellData.route_label(tab) defp activity_label("AI Assistant"), do: "Chat" defp activity_label("Source Control"), do: "Git" defp activity_label(label), do: translated(label) defp active_sidebar_label(activity_buttons, active_view, sidebar_data) do Enum.find_value(activity_buttons, translated(Map.get(sidebar_data, :title, "")), fn button -> if button.id == active_view, do: activity_label(button.label), else: nil end) end defp sidebar_header_label(label), do: translated(label) defp timeline_height(entry, entries) do max_count = entries |> Enum.map(&(&1.count || 0)) |> Enum.max(fn -> 1 end) max(4, ((entry.count || 0) / max_count) * 100) end defp format_sidebar_timestamp(nil), do: "" defp format_sidebar_timestamp(timestamp) do timestamp |> DateTime.from_unix!(:millisecond) |> Calendar.strftime("%x") end defp image_media?(item), do: String.starts_with?(to_string(item.mime_type || ""), "image/") defp media_thumbnail_class(item) do if image_media?(item), do: "media-thumbnail has-image", else: "media-thumbnail" end defp current_tab(%{active_tab: nil}), do: nil defp current_tab(%{tabs: tabs, active_tab: {type, id}}) do Enum.find(tabs, &(&1.type == type and &1.id == id)) end defp sync_layout(workbench, params) do workbench |> maybe_set_sidebar_width(Map.get(params, "sidebar_width")) |> maybe_set_assistant_width(Map.get(params, "assistant_sidebar_width")) end defp resize_panel(workbench, "sidebar", width) do workbench |> Workbench.set_sidebar_width(parse_width(width)) |> Map.put(:sidebar_visible, true) end defp resize_panel(workbench, "assistant", width) do workbench |> Workbench.set_assistant_sidebar_width(parse_width(width)) |> Map.put(:assistant_sidebar_visible, true) end defp resize_panel(workbench, _target, _width), do: workbench defp maybe_set_sidebar_width(workbench, nil), do: workbench defp maybe_set_sidebar_width(workbench, width), do: Workbench.set_sidebar_width(workbench, parse_width(width)) defp maybe_set_assistant_width(workbench, nil), do: workbench defp maybe_set_assistant_width(workbench, width) do Workbench.set_assistant_sidebar_width(workbench, parse_width(width)) end defp parse_width(width) when is_integer(width), do: width defp parse_width(width) when is_binary(width) do case Integer.parse(width) do {parsed, _rest} -> parsed :error -> 0 end end defp ignore_shortcut?(params) do Map.get(params, "alt", false) or Map.get(params, "contentEditable", false) or Map.get(params, "content_editable", false) or Map.get(params, "tag") in ["INPUT", "TEXTAREA", "SELECT"] or Map.get(params, :tag) in ["INPUT", "TEXTAREA", "SELECT"] end defp open_sidebar_item(socket, params, intent) do route_atom = sidebar_route_atom(Map.fetch!(params, "route")) tab_id = tab_id_for_route(route_atom, Map.fetch!(params, "id")) workbench = Workbench.open_tab(socket.assigns.workbench, route_atom, tab_id, tab_intent(route_atom, intent)) tab_meta = Map.put(socket.assigns.tab_meta, {route_atom, tab_id}, %{ title: Map.get(params, "title", ""), subtitle: Map.get(params, "subtitle", "") }) socket |> assign(:tab_meta, tab_meta) |> reload_shell(workbench) end defp sidebar_route_atom(route) when is_atom(route), do: route defp sidebar_route_atom(route) when is_binary(route), do: String.to_existing_atom(route) defp tab_id_for_route(route, id) do case Registry.editor_route(route) do %{singleton: true} -> Atom.to_string(route) _other -> id end end defp tab_intent(route, requested_intent) do case Registry.editor_route(route) do %{singleton: true} -> :pin _other -> requested_intent end end defp sidebar_item_selected?(workbench, route, id) do route_atom = sidebar_route_atom(route) workbench.active_tab == {route_atom, tab_id_for_route(route_atom, id)} end defp tab_title(nil, _tab_meta), do: translated("Dashboard") defp tab_title(tab, tab_meta) do case Map.get(tab_meta, {tab.type, tab.id}) do %{title: title} when is_binary(title) and title != "" -> title _other -> default_tab_title(tab) end end defp tab_subtitle(nil, _tab_meta), do: translated("dashboard.subtitle") defp tab_subtitle(tab, tab_meta) do case Map.get(tab_meta, {tab.type, tab.id}) do %{subtitle: subtitle} when is_binary(subtitle) and subtitle != "" -> subtitle _other -> "Desktop workbench content routed through the Elixir shell." end end defp default_tab_title(%{type: type, id: id}) do case Registry.editor_route(type) do %{singleton: true} -> ShellData.route_label(type) _other -> id end end defp tab_route_label(nil), do: translated("Dashboard") defp tab_route_label(%{type: type}), do: ShellData.route_label(type) defp tab_icon_id(nil), do: "posts" defp tab_icon_id(%{type: :post}), do: "posts" defp tab_icon_id(%{type: :git_diff}), do: "git" defp tab_icon_id(%{type: :style}), do: "settings" defp tab_icon_id(%{type: type}), do: Atom.to_string(type) defp media_thumbnail_glyph(mime_type) do case String.split(to_string(mime_type || ""), "/", parts: 2) do ["image", _rest] -> "IMG" ["video", _rest] -> "VID" ["audio", _rest] -> "AUD" ["application", _rest] -> "DOC" _other -> "FILE" end end end