From d464775c4ac08ced896a14394b420fccd5c06500 Mon Sep 17 00:00:00 2001 From: Abdenasser Date: Wed, 13 Nov 2024 03:16:45 +0100 Subject: [PATCH 01/14] ui refactoring --- src/lib/components/ProcessTable.svelte | 430 ---------------- src/lib/components/StatsBar.svelte | 461 ------------------ .../components/process/ActionButtons.svelte | 147 ++++++ src/lib/components/process/ProcessIcon.svelte | 70 +++ src/lib/components/process/ProcessRow.svelte | 60 +++ .../components/process/ProcessTable.svelte | 86 ++++ src/lib/components/process/TableHeader.svelte | 136 ++++++ .../process/cells/ProcessCell.svelte | 289 +++++++++++ src/lib/components/process/index.ts | 6 + src/lib/components/stats/CpuPanel.svelte | 60 +++ src/lib/components/stats/MemoryPanel.svelte | 58 +++ src/lib/components/stats/NetworkPanel.svelte | 47 ++ src/lib/components/stats/PanelHeader.svelte | 50 ++ src/lib/components/stats/ProgressBar.svelte | 80 +++ src/lib/components/stats/StatItem.svelte | 30 ++ src/lib/components/stats/StatPanel.svelte | 34 ++ src/lib/components/stats/StatsBar.svelte | 50 ++ src/lib/components/stats/StoragePanel.svelte | 56 +++ src/lib/components/stats/SystemPanel.svelte | 38 ++ src/lib/components/stats/index.ts | 9 + src/routes/+page.svelte | 6 +- 21 files changed, 1309 insertions(+), 894 deletions(-) delete mode 100644 src/lib/components/ProcessTable.svelte delete mode 100644 src/lib/components/StatsBar.svelte create mode 100644 src/lib/components/process/ActionButtons.svelte create mode 100644 src/lib/components/process/ProcessIcon.svelte create mode 100644 src/lib/components/process/ProcessRow.svelte create mode 100644 src/lib/components/process/ProcessTable.svelte create mode 100644 src/lib/components/process/TableHeader.svelte create mode 100644 src/lib/components/process/cells/ProcessCell.svelte create mode 100644 src/lib/components/process/index.ts create mode 100644 src/lib/components/stats/CpuPanel.svelte create mode 100644 src/lib/components/stats/MemoryPanel.svelte create mode 100644 src/lib/components/stats/NetworkPanel.svelte create mode 100644 src/lib/components/stats/PanelHeader.svelte create mode 100644 src/lib/components/stats/ProgressBar.svelte create mode 100644 src/lib/components/stats/StatItem.svelte create mode 100644 src/lib/components/stats/StatPanel.svelte create mode 100644 src/lib/components/stats/StatsBar.svelte create mode 100644 src/lib/components/stats/StoragePanel.svelte create mode 100644 src/lib/components/stats/SystemPanel.svelte create mode 100644 src/lib/components/stats/index.ts diff --git a/src/lib/components/ProcessTable.svelte b/src/lib/components/ProcessTable.svelte deleted file mode 100644 index e0d5d2f..0000000 --- a/src/lib/components/ProcessTable.svelte +++ /dev/null @@ -1,430 +0,0 @@ - - -
- - - - {#each columns.filter((col) => col.visible) as column} - - {/each} - - - - - {#each processes as process (process.pid)} - 50 || - process.memory_usage / (systemStats?.memory_total || 0) > 0.1} - class:pinned={pinnedProcesses.has(process.command)} - > - {#each columns.filter((col) => col.visible) as column} - - {/each} - - - {/each} - -
onToggleSort(column.id)}> -
- {column.label} - - {getSortIndicator(column.id)} - -
-
Actions
- {#if column.id === "name"} -
- - {process.name} -
- {:else if column.format} - {@html column.format(process[column.id])} - {:else} - {process[column.id]} - {/if} -
-
- - - -
-
-
- - diff --git a/src/lib/components/StatsBar.svelte b/src/lib/components/StatsBar.svelte deleted file mode 100644 index 034c00b..0000000 --- a/src/lib/components/StatsBar.svelte +++ /dev/null @@ -1,461 +0,0 @@ - - -
- {#if systemStats} -
- -
-
- -

CPU Usage

-
- {formatPercentage( - systemStats.cpu_usage.reduce((a, b) => a + b, 0) / - systemStats.cpu_usage.length, - )} -
-
-
- {#each systemStats.cpu_usage as usage, i} -
-
- Core {i} -
-
-
- {Math.round(usage)}% -
-
- {/each} -
-
- - -
-
- -

Memory

-
{formatPercentage(memoryPercentage)}
-
-
-
-
- Memory usage -
-
-
- {formatPercentage(memoryPercentage)} -
-
-
- Total - {formatMemorySize(systemStats.memory_total)} -
-
- Used - {formatMemorySize(systemStats.memory_used)} -
-
- Free - {formatMemorySize(systemStats.memory_free)} -
-
-
- - -
-
- -

Storage

-
- {formatPercentage(diskUsagePercentage)} -
-
-
-
- Total - {formatBytes(systemStats.disk_total_bytes)} -
-
- Used - {formatBytes(systemStats.disk_used_bytes)} -
-
- Free - {formatBytes(systemStats.disk_free_bytes)} -
-
-
- - -
-
- -

System

-
-
-
- Uptime - {formatUptime(systemStats.uptime)} -
-
- 1m Load - {systemStats.load_avg[0].toFixed(2)} -
-
- 5m Load - {systemStats.load_avg[1].toFixed(2)} -
-
- 15m Load - {systemStats.load_avg[2].toFixed(2)} -
-
-
- - -
-
- -

Network I/O

-
-
-
- ↓ Receiving - {formatBytes(systemStats.network_rx_bytes)}/s -
-
- ↑ Sending - {formatBytes(systemStats.network_tx_bytes)}/s -
-
-
-
- {/if} -
- - diff --git a/src/lib/components/process/ActionButtons.svelte b/src/lib/components/process/ActionButtons.svelte new file mode 100644 index 0000000..e219821 --- /dev/null +++ b/src/lib/components/process/ActionButtons.svelte @@ -0,0 +1,147 @@ + + +
+ + + +
+ + diff --git a/src/lib/components/process/ProcessIcon.svelte b/src/lib/components/process/ProcessIcon.svelte new file mode 100644 index 0000000..660134a --- /dev/null +++ b/src/lib/components/process/ProcessIcon.svelte @@ -0,0 +1,70 @@ + + + + + diff --git a/src/lib/components/process/ProcessRow.svelte b/src/lib/components/process/ProcessRow.svelte new file mode 100644 index 0000000..d79481c --- /dev/null +++ b/src/lib/components/process/ProcessRow.svelte @@ -0,0 +1,60 @@ + + + + {#each columns.filter((col) => col.visible) as column} + + {/each} + + + + + + diff --git a/src/lib/components/process/ProcessTable.svelte b/src/lib/components/process/ProcessTable.svelte new file mode 100644 index 0000000..dfbb576 --- /dev/null +++ b/src/lib/components/process/ProcessTable.svelte @@ -0,0 +1,86 @@ + + +
+ + + + {#each processes as process (process.pid)} + 50 || + process.memory_usage / (systemStats?.memory_total || 0) > 0.1} + {onTogglePin} + {onShowDetails} + {onKillProcess} + /> + {/each} + +
+
+ + diff --git a/src/lib/components/process/TableHeader.svelte b/src/lib/components/process/TableHeader.svelte new file mode 100644 index 0000000..9d90706 --- /dev/null +++ b/src/lib/components/process/TableHeader.svelte @@ -0,0 +1,136 @@ + + + + + {#each columns.filter((col) => col.visible) as column} + onToggleSort(column.id)} + > +
+ {column.label} + + {getSortIndicator(column.id)} + +
+ + {/each} + Actions + + + + diff --git a/src/lib/components/process/cells/ProcessCell.svelte b/src/lib/components/process/cells/ProcessCell.svelte new file mode 100644 index 0000000..107fb3d --- /dev/null +++ b/src/lib/components/process/cells/ProcessCell.svelte @@ -0,0 +1,289 @@ + + + + {#if field === "name"} +
+ + {process.name} +
+ {:else} + {formatValue(process[field])} + {/if} + + + diff --git a/src/lib/components/process/index.ts b/src/lib/components/process/index.ts new file mode 100644 index 0000000..76c3579 --- /dev/null +++ b/src/lib/components/process/index.ts @@ -0,0 +1,6 @@ +export { default as ProcessTable } from "./ProcessTable.svelte"; +export { default as ProcessRow } from "./ProcessRow.svelte"; +export { default as TableHeader } from "./TableHeader.svelte"; +export { default as ProcessCell } from "./cells/ProcessCell.svelte"; +export { default as ActionButtons } from "./ActionButtons.svelte"; +export { default as ProcessIcon } from "./ProcessIcon.svelte"; diff --git a/src/lib/components/stats/CpuPanel.svelte b/src/lib/components/stats/CpuPanel.svelte new file mode 100644 index 0000000..2111940 --- /dev/null +++ b/src/lib/components/stats/CpuPanel.svelte @@ -0,0 +1,60 @@ + + +
+ +
+ {#each cpuUsage as usage, i} +
+ +
+ {/each} +
+
+ + diff --git a/src/lib/components/stats/MemoryPanel.svelte b/src/lib/components/stats/MemoryPanel.svelte new file mode 100644 index 0000000..08f2930 --- /dev/null +++ b/src/lib/components/stats/MemoryPanel.svelte @@ -0,0 +1,58 @@ + + +
+ +
+
+ +
+ + + +
+
+ + diff --git a/src/lib/components/stats/NetworkPanel.svelte b/src/lib/components/stats/NetworkPanel.svelte new file mode 100644 index 0000000..e373e51 --- /dev/null +++ b/src/lib/components/stats/NetworkPanel.svelte @@ -0,0 +1,47 @@ + + +
+ +
+ + +
+
+ + diff --git a/src/lib/components/stats/PanelHeader.svelte b/src/lib/components/stats/PanelHeader.svelte new file mode 100644 index 0000000..11798df --- /dev/null +++ b/src/lib/components/stats/PanelHeader.svelte @@ -0,0 +1,50 @@ + + +
+ +

{title}

+ {#if usageValue} +
{usageValue}
+ {/if} +
+ + diff --git a/src/lib/components/stats/ProgressBar.svelte b/src/lib/components/stats/ProgressBar.svelte new file mode 100644 index 0000000..0834121 --- /dev/null +++ b/src/lib/components/stats/ProgressBar.svelte @@ -0,0 +1,80 @@ + + +
+ {label} +
+
+
+ {Math.round(value)}% +
+ + diff --git a/src/lib/components/stats/StatItem.svelte b/src/lib/components/stats/StatItem.svelte new file mode 100644 index 0000000..d575317 --- /dev/null +++ b/src/lib/components/stats/StatItem.svelte @@ -0,0 +1,30 @@ + + +
+ {label} + {value} +
+ + diff --git a/src/lib/components/stats/StatPanel.svelte b/src/lib/components/stats/StatPanel.svelte new file mode 100644 index 0000000..90cdc20 --- /dev/null +++ b/src/lib/components/stats/StatPanel.svelte @@ -0,0 +1,34 @@ + + +
+

{title}

+
+ +
+
+ + diff --git a/src/lib/components/stats/StatsBar.svelte b/src/lib/components/stats/StatsBar.svelte new file mode 100644 index 0000000..949d770 --- /dev/null +++ b/src/lib/components/stats/StatsBar.svelte @@ -0,0 +1,50 @@ + + +
+ {#if systemStats} +
+ + + + + + + + + +
+ {/if} +
+ + diff --git a/src/lib/components/stats/StoragePanel.svelte b/src/lib/components/stats/StoragePanel.svelte new file mode 100644 index 0000000..cf1fae1 --- /dev/null +++ b/src/lib/components/stats/StoragePanel.svelte @@ -0,0 +1,56 @@ + + +
+ +
+ + + +
+
+ + diff --git a/src/lib/components/stats/SystemPanel.svelte b/src/lib/components/stats/SystemPanel.svelte new file mode 100644 index 0000000..d8faf32 --- /dev/null +++ b/src/lib/components/stats/SystemPanel.svelte @@ -0,0 +1,38 @@ + + +
+ +
+ + + + +
+
+ + diff --git a/src/lib/components/stats/index.ts b/src/lib/components/stats/index.ts new file mode 100644 index 0000000..e79623e --- /dev/null +++ b/src/lib/components/stats/index.ts @@ -0,0 +1,9 @@ +export { default as StatsBar } from "./StatsBar.svelte"; +export { default as CpuPanel } from "./CpuPanel.svelte"; +export { default as MemoryPanel } from "./MemoryPanel.svelte"; +export { default as StoragePanel } from "./StoragePanel.svelte"; +export { default as SystemPanel } from "./SystemPanel.svelte"; +export { default as NetworkPanel } from "./NetworkPanel.svelte"; +export { default as PanelHeader } from "./PanelHeader.svelte"; +export { default as ProgressBar } from "./ProgressBar.svelte"; +export { default as StatItem } from "./StatItem.svelte"; diff --git a/src/routes/+page.svelte b/src/routes/+page.svelte index 1e35955..ae5ff10 100644 --- a/src/routes/+page.svelte +++ b/src/routes/+page.svelte @@ -1,9 +1,9 @@ - -
- - - -
- - diff --git a/src/lib/components/process/ProcessIcon.svelte b/src/lib/components/process/ProcessIcon.svelte index 660134a..a4c71a6 100644 --- a/src/lib/components/process/ProcessIcon.svelte +++ b/src/lib/components/process/ProcessIcon.svelte @@ -2,6 +2,7 @@ import * as SimpleIcons from "simple-icons"; export let processName: string; + export let size: number = 16; function handleImageError(event: Event) { const img = event.target as HTMLImageElement; @@ -20,11 +21,20 @@ SimpleIcons[companyIconKey as keyof typeof SimpleIcons]; if (companyIcon) { - return createSvgDataUrl(companyIcon); + // Use theme color instead of brand color + const color = getComputedStyle(document.documentElement) + .getPropertyValue("--text") + .trim(); + const svg = + typeof companyIcon === "object" && "svg" in companyIcon + ? companyIcon.svg + : ""; + const svgWithColor = svg.replace(" diff --git a/src/lib/components/process/ProcessRow.svelte b/src/lib/components/process/ProcessRow.svelte index d79481c..e69de29 100644 --- a/src/lib/components/process/ProcessRow.svelte +++ b/src/lib/components/process/ProcessRow.svelte @@ -1,60 +0,0 @@ - - - - {#each columns.filter((col) => col.visible) as column} - - {/each} - - - - - - diff --git a/src/lib/components/process/ProcessTable.svelte b/src/lib/components/process/ProcessTable.svelte index dfbb576..b658ed9 100644 --- a/src/lib/components/process/ProcessTable.svelte +++ b/src/lib/components/process/ProcessTable.svelte @@ -1,7 +1,12 @@
- + + + {#each columns.filter((col) => col.visible) as column} + + {/each} + + + {#each processes as process (process.pid)} - 50 || + 50 || process.memory_usage / (systemStats?.memory_total || 0) > 0.1} - {onTogglePin} - {onShowDetails} - {onKillProcess} - /> + class:pinned={pinnedProcesses.has(process.command)} + > + {#each columns.filter((col) => col.visible) as column} + + {/each} + + {/each}
onToggleSort(column.id)}> +
+ {column.label} + + {getSortIndicator(column.id)} + +
+
Actions
+ {#if column.id === "name"} +
+ + {process.name} +
+ {:else if column.format} + {@html column.format(process[column.id])} + {:else} + {process[column.id]} + {/if} +
+
+ + + +
+
@@ -83,4 +146,211 @@ border-collapse: collapse; font-size: 13px; } + + th { + position: sticky; + top: 0; + background: var(--mantle); + text-align: left; + padding: 8px 12px; + font-weight: 500; + color: var(--subtext0); + border-bottom: 1px solid var(--surface0); + transition: background-color 0.2s ease; + z-index: 3; + } + + td { + padding: 6px 12px; + border-bottom: 1px solid var(--surface0); + color: var(--text); + z-index: 1; + } + + tr:hover { + background-color: var(--surface0); + } + + .truncate { + white-space: nowrap; + overflow: hidden; + text-overflow: ellipsis; + max-width: 0; + } + + .sortable { + cursor: pointer; + user-select: none; + } + + .th-content { + display: flex; + align-items: center; + gap: 8px; + } + + .sort-indicator { + color: var(--overlay0); + font-size: 12px; + opacity: 0.5; + transition: all 0.2s ease; + } + + .sort-indicator.active { + color: var(--blue); + opacity: 1; + } + + .sortable:hover .sort-indicator { + opacity: 1; + } + + .high-usage { + background-color: color-mix(in srgb, var(--red) 10%, transparent); + } + + .high-usage:hover { + background-color: color-mix(in srgb, var(--red) 15%, transparent); + } + + tr.pinned { + background-color: color-mix(in srgb, var(--blue) 10%, transparent); + } + + tr.pinned:hover { + background-color: color-mix(in srgb, var(--blue) 15%, transparent); + } + + th:last-child { + width: 120px; + min-width: 120px; + max-width: 120px; + } + + td:last-child { + width: 120px; + min-width: 120px; + max-width: 120px; + padding: 6px 8px; + } + + .col-actions { + position: sticky; + right: 0; + z-index: 2; + background: var(--base); + border-left: 1px solid var(--surface0); + width: 120px; + } + + .action-buttons { + display: flex; + gap: 4px; + align-items: center; + justify-content: center; + } + + .btn-action { + display: inline-flex; + align-items: center; + justify-content: center; + border: none; + cursor: pointer; + background: transparent; + border-radius: 6px; + width: 28px; + height: 28px; + transition: all 0.2s ease; + position: relative; + overflow: hidden; + } + + .btn-action::before { + content: ""; + position: absolute; + inset: 0; + opacity: 0.1; + transition: opacity 0.2s ease; + } + + .btn-action:hover::before { + opacity: 0.15; + } + + .pin-btn { + color: var(--sapphire); + } + + .pin-btn::before { + background: var(--sapphire); + } + + .pin-btn.pinned { + color: var(--blue); + transform: rotate(45deg); + } + + .pin-btn.pinned::before { + background: var(--blue); + opacity: 0.15; + } + + .info-btn { + color: var(--lavender); + } + + .info-btn::before { + background: var(--lavender); + } + + .kill-btn { + color: var(--red); + border: 1px solid color-mix(in srgb, var(--red) 30%, transparent); + } + + .kill-btn:hover { + color: var(--base); + background: var(--red); + } + + .kill-btn:hover::before { + opacity: 1; + } + + .btn-action:hover { + box-shadow: 0 0 12px color-mix(in srgb, currentColor 20%, transparent); + } + + .btn-action:active { + transform: translateY(1px); + } + + .pin-btn.pinned:active { + transform: rotate(45deg) translateY(1px); + } + + .btn-action:focus { + outline: none; + box-shadow: 0 0 0 2px color-mix(in srgb, currentColor 30%, transparent); + } + + .btn-action:disabled { + opacity: 0.5; + cursor: not-allowed; + } + + .btn-action:disabled:hover { + transform: none; + box-shadow: none; + } + + .btn-action:disabled::before { + display: none; + } + + .name-cell { + display: flex; + align-items: center; + gap: 8px; + } diff --git a/src/lib/components/process/TableHeader.svelte b/src/lib/components/process/TableHeader.svelte index 9d90706..e69de29 100644 --- a/src/lib/components/process/TableHeader.svelte +++ b/src/lib/components/process/TableHeader.svelte @@ -1,136 +0,0 @@ - - - - - {#each columns.filter((col) => col.visible) as column} - onToggleSort(column.id)} - > -
- {column.label} - - {getSortIndicator(column.id)} - -
- - {/each} - Actions - - - - diff --git a/src/lib/components/process/cells/ProcessCell.svelte b/src/lib/components/process/cells/ProcessCell.svelte index 107fb3d..e69de29 100644 --- a/src/lib/components/process/cells/ProcessCell.svelte +++ b/src/lib/components/process/cells/ProcessCell.svelte @@ -1,289 +0,0 @@ - - - - {#if field === "name"} -
- - {process.name} -
- {:else} - {formatValue(process[field])} - {/if} - - - From bbe778e1204a686d414ce28774325a3cebedce96 Mon Sep 17 00:00:00 2001 From: Abdenasser Date: Wed, 13 Nov 2024 13:56:49 +0100 Subject: [PATCH 03/14] TableHeader component --- .../components/process/ProcessTable.svelte | 71 +---------------- src/lib/components/process/TableHeader.svelte | 79 +++++++++++++++++++ 2 files changed, 81 insertions(+), 69 deletions(-) diff --git a/src/lib/components/process/ProcessTable.svelte b/src/lib/components/process/ProcessTable.svelte index b658ed9..0c6f576 100644 --- a/src/lib/components/process/ProcessTable.svelte +++ b/src/lib/components/process/ProcessTable.svelte @@ -7,6 +7,7 @@ import Fa from "svelte-fa"; import type { Process, Column } from "$lib/types"; import ProcessIcon from "./ProcessIcon.svelte"; + import TableHeader from "./TableHeader.svelte"; export let processes: Process[]; export let columns: Column[]; @@ -18,33 +19,11 @@ export let onTogglePin: (command: string) => void; export let onShowDetails: (process: Process) => void; export let onKillProcess: (process: Process) => void; - - function getSortIndicator(field: keyof Process) { - if (sortConfig.field !== field) return "↕"; - return sortConfig.direction === "asc" ? "↑" : "↓"; - }
- - - {#each columns.filter((col) => col.visible) as column} - - {/each} - - - + {#each processes as process (process.pid)} + import type { Process, Column } from "$lib/types"; + + export let columns: Column[]; + export let sortConfig: { field: keyof Process; direction: "asc" | "desc" }; + export let onToggleSort: (field: keyof Process) => void; + + function getSortIndicator(field: keyof Process) { + if (sortConfig.field !== field) return "↕"; + return sortConfig.direction === "asc" ? "↑" : "↓"; + } + + + + + {#each columns.filter((col) => col.visible) as column} + + {/each} + + + + + From 44cb0c965e516b43b73ad6716869314377aaedbd Mon Sep 17 00:00:00 2001 From: Abdenasser Date: Wed, 13 Nov 2024 14:06:00 +0100 Subject: [PATCH 04/14] ActionButtons component --- .../components/process/ActionButtons.svelte | 164 ++++++++++++++++++ .../components/process/ProcessTable.svelte | 154 +--------------- 2 files changed, 172 insertions(+), 146 deletions(-) diff --git a/src/lib/components/process/ActionButtons.svelte b/src/lib/components/process/ActionButtons.svelte index e69de29..2c28fd5 100644 --- a/src/lib/components/process/ActionButtons.svelte +++ b/src/lib/components/process/ActionButtons.svelte @@ -0,0 +1,164 @@ + + + + + diff --git a/src/lib/components/process/ProcessTable.svelte b/src/lib/components/process/ProcessTable.svelte index 0c6f576..b26327c 100644 --- a/src/lib/components/process/ProcessTable.svelte +++ b/src/lib/components/process/ProcessTable.svelte @@ -1,13 +1,8 @@ + + + {#each columns.filter((col) => col.visible) as column} + + {/each} + + + + diff --git a/src/lib/components/process/ProcessTable.svelte b/src/lib/components/process/ProcessTable.svelte index b26327c..7291e9f 100644 --- a/src/lib/components/process/ProcessTable.svelte +++ b/src/lib/components/process/ProcessTable.svelte @@ -1,8 +1,7 @@
diff --git a/src/lib/components/KillProcessModal.svelte b/src/lib/components/modals/KillProcessModal.svelte similarity index 97% rename from src/lib/components/KillProcessModal.svelte rename to src/lib/components/modals/KillProcessModal.svelte index 6ca601c..c8d736a 100644 --- a/src/lib/components/KillProcessModal.svelte +++ b/src/lib/components/modals/KillProcessModal.svelte @@ -1,5 +1,5 @@ + +
+ +
+ + diff --git a/src/lib/components/ToolBar.svelte b/src/lib/components/toolbar/ToolBar.svelte similarity index 96% rename from src/lib/components/ToolBar.svelte rename to src/lib/components/toolbar/ToolBar.svelte index 80c16ff..419d7b6 100644 --- a/src/lib/components/ToolBar.svelte +++ b/src/lib/components/toolbar/ToolBar.svelte @@ -1,5 +1,5 @@ + + + + diff --git a/src/lib/components/toolbar/ToolBar.svelte b/src/lib/components/toolbar/ToolBar.svelte index 419d7b6..68d3243 100644 --- a/src/lib/components/toolbar/ToolBar.svelte +++ b/src/lib/components/toolbar/ToolBar.svelte @@ -11,6 +11,7 @@ import { configStore } from "$lib/stores/config"; import type { AppConfig } from "$lib/types/config"; import StatusFilter from "./StatusFilter.svelte"; + import SearchBox from "./SearchBox.svelte"; export let searchTerm: string; export let statusFilter: string = "all"; export let itemsPerPage: number; @@ -73,21 +74,7 @@
- +
From b9515c8ae35fa28c90d303ef06e7d1007f555f46 Mon Sep 17 00:00:00 2001 From: Abdenasser Date: Wed, 13 Nov 2024 14:50:25 +0100 Subject: [PATCH 09/14] refresh controls --- .../components/toolbar/RefreshControls.svelte | 117 ++++++++++++++++++ src/lib/components/toolbar/ToolBar.svelte | 36 +----- 2 files changed, 119 insertions(+), 34 deletions(-) diff --git a/src/lib/components/toolbar/RefreshControls.svelte b/src/lib/components/toolbar/RefreshControls.svelte index e69de29..b679dcf 100644 --- a/src/lib/components/toolbar/RefreshControls.svelte +++ b/src/lib/components/toolbar/RefreshControls.svelte @@ -0,0 +1,117 @@ + + +
+ + +
+ + diff --git a/src/lib/components/toolbar/ToolBar.svelte b/src/lib/components/toolbar/ToolBar.svelte index 68d3243..a615e2f 100644 --- a/src/lib/components/toolbar/ToolBar.svelte +++ b/src/lib/components/toolbar/ToolBar.svelte @@ -12,6 +12,7 @@ import type { AppConfig } from "$lib/types/config"; import StatusFilter from "./StatusFilter.svelte"; import SearchBox from "./SearchBox.svelte"; + import RefreshControls from "./RefreshControls.svelte"; export let searchTerm: string; export let statusFilter: string = "all"; export let itemsPerPage: number; @@ -37,14 +38,6 @@ })), ]; - const refreshRateOptions = [ - { value: 1000, label: "1s" }, - { value: 2000, label: "2s" }, - { value: 5000, label: "5s" }, - { value: 10000, label: "10s" }, - { value: 30000, label: "30s" }, - ]; - function changePage(page: number) { if (page >= 1 && page <= totalPages) { currentPage = page; @@ -166,32 +159,7 @@ {/if}
-
-
- - -
-
+
From d516169dee7cf2a07d3b53ee51d582f9ec4a5bf9 Mon Sep 17 00:00:00 2001 From: Abdenasser Date: Wed, 13 Nov 2024 14:53:18 +0100 Subject: [PATCH 10/14] pagination controls --- .../toolbar/PaginationControls.svelte | 149 ++++++++++++++++++ src/lib/components/toolbar/ToolBar.svelte | 134 +--------------- 2 files changed, 156 insertions(+), 127 deletions(-) diff --git a/src/lib/components/toolbar/PaginationControls.svelte b/src/lib/components/toolbar/PaginationControls.svelte index e69de29..fea258d 100644 --- a/src/lib/components/toolbar/PaginationControls.svelte +++ b/src/lib/components/toolbar/PaginationControls.svelte @@ -0,0 +1,149 @@ + + +
+ + + +
+ + diff --git a/src/lib/components/toolbar/ToolBar.svelte b/src/lib/components/toolbar/ToolBar.svelte index a615e2f..0073ab3 100644 --- a/src/lib/components/toolbar/ToolBar.svelte +++ b/src/lib/components/toolbar/ToolBar.svelte @@ -13,6 +13,7 @@ import StatusFilter from "./StatusFilter.svelte"; import SearchBox from "./SearchBox.svelte"; import RefreshControls from "./RefreshControls.svelte"; + import PaginationControls from "./PaginationControls.svelte"; export let searchTerm: string; export let statusFilter: string = "all"; export let itemsPerPage: number; @@ -28,7 +29,6 @@ export let refreshRate: number; export let isFrozen: boolean; - const itemsPerPageOptions = [15, 25, 50, 100, 250, 500]; let showColumnMenu = false; const statusOptions = [ { value: "all", label: "All Statuses" }, @@ -38,12 +38,6 @@ })), ]; - function changePage(page: number) { - if (page >= 1 && page <= totalPages) { - currentPage = page; - } - } - function handleColumnVisibilityChange(columnId: string, visible: boolean) { configStore.updateConfig({ appearance: { @@ -72,53 +66,12 @@
-
- - - -
+
@@ -255,79 +208,6 @@ box-shadow: 0 0 0 2px color-mix(in srgb, var(--blue) 25%, transparent); } - .pagination-controls { - display: flex; - align-items: center; - gap: 12px; - } - - .select-input { - padding: 6px 12px; - font-size: 12px; - color: var(--text); - background: var(--surface0); - border: 1px solid var(--surface1); - border-radius: 6px; - cursor: pointer; - appearance: none; - padding-right: 24px; - background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 24 24' fill='none' stroke='%23cdd6f4' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpolyline points='6 9 12 15 18 9'%3E%3C/polyline%3E%3C/svg%3E"); - background-repeat: no-repeat; - background-position: right 8px center; - } - - .select-input:hover { - background-color: var(--surface1); - } - - .select-input:focus { - outline: none; - border-color: var(--blue); - } - - .pagination { - display: flex; - align-items: center; - gap: 8px; - } - - .btn-page { - padding: 6px 10px; - font-size: 12px; - color: var(--text); - background: var(--surface0); - border: 1px solid var(--surface1); - border-radius: 6px; - cursor: pointer; - transition: all 0.2s ease; - } - - .btn-page:hover:not(:disabled) { - background: var(--surface1); - } - - .btn-page:disabled { - opacity: 0.5; - cursor: not-allowed; - } - - .page-info { - font-size: 12px; - color: var(--subtext0); - display: flex; - flex-direction: column; - align-items: center; - flex-shrink: 0; - } - - .page-info span { - display: block; - } - - .results-info { - color: var(--overlay0); - } - .column-toggle { position: relative; } From c2afdfeaef0d86e0399ff1ed0a9b78faf16275e9 Mon Sep 17 00:00:00 2001 From: Abdenasser Date: Wed, 13 Nov 2024 14:59:07 +0100 Subject: [PATCH 11/14] column toggle --- .../components/toolbar/ColumnToggle.svelte | 173 ++++++++++ src/lib/components/toolbar/ToolBar.svelte | 315 +----------------- 2 files changed, 181 insertions(+), 307 deletions(-) diff --git a/src/lib/components/toolbar/ColumnToggle.svelte b/src/lib/components/toolbar/ColumnToggle.svelte index e69de29..57704b6 100644 --- a/src/lib/components/toolbar/ColumnToggle.svelte +++ b/src/lib/components/toolbar/ColumnToggle.svelte @@ -0,0 +1,173 @@ + + +
+ + + {#if showColumnMenu} + +
(showColumnMenu = false)}> + {#each columns as column} + + {/each} +
+ {/if} +
+ + diff --git a/src/lib/components/toolbar/ToolBar.svelte b/src/lib/components/toolbar/ToolBar.svelte index 0073ab3..1962066 100644 --- a/src/lib/components/toolbar/ToolBar.svelte +++ b/src/lib/components/toolbar/ToolBar.svelte @@ -1,19 +1,13 @@
@@ -74,43 +39,7 @@ />
-
- - - {#if showColumnMenu} - -
(showColumnMenu = false)}> - {#each columns as column} - - {/each} -
- {/if} -
+ @@ -136,232 +65,4 @@ .toolbar-spacer { flex: 1; } - - .search-box { - width: 240px; - } - - .search-input-wrapper { - position: relative; - display: flex; - gap: 8px; - align-items: center; - } - - .search-input { - width: 240px; - height: 28px; - padding: 0 12px; - padding-right: 70px; - border: 1px solid var(--surface1); - border-radius: 6px; - font-size: 12px; - background-color: var(--surface0); - color: var(--text); - transition: all 0.2s ease; - } - - .btn-clear { - position: absolute; - right: 4px; - top: 50%; - transform: translateY(-50%); - padding: 4px 8px; - font-size: 11px; - color: var(--subtext0); - background: var(--surface1); - border: none; - border-radius: 4px; - cursor: pointer; - transition: all 0.2s ease; - } - - .btn-clear:hover { - background: var(--surface2); - color: var(--text); - } - - .search-input::-webkit-search-cancel-button { - -webkit-appearance: none; - height: 14px; - width: 14px; - margin-right: 4px; - background: var(--overlay0); - -webkit-mask: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='none' stroke='currentColor' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cline x1='18' y1='6' x2='6' y2='18'%3E%3C/line%3E%3Cline x1='6' y1='6' x2='18' y2='18'%3E%3C/line%3E%3C/svg%3E") - no-repeat 50% 50%; - mask: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='none' stroke='currentColor' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'%3E%3Cline x1='18' y1='6' x2='6' y2='18'%3E%3C/line%3E%3Cline x1='6' y1='6' x2='18' y2='18'%3E%3C/line%3E%3C/svg%3E") - no-repeat 50% 50%; - cursor: pointer; - } - - .search-input::-webkit-search-cancel-button:hover { - background: var(--text); - } - - .search-input:hover { - background-color: var(--surface1); - } - - .search-input:focus { - outline: none; - border-color: var(--blue); - box-shadow: 0 0 0 2px color-mix(in srgb, var(--blue) 25%, transparent); - } - - .column-toggle { - position: relative; - } - - .btn-toggle { - display: flex; - align-items: center; - gap: 6px; - padding: 6px 12px; - font-size: 12px; - color: var(--text); - background: var(--surface0); - border: 1px solid var(--surface1); - border-radius: 6px; - cursor: pointer; - transition: all 0.2s ease; - } - - .btn-toggle:hover { - background: var(--surface1); - } - - .icon { - font-size: 10px; - color: var(--subtext0); - } - - .column-menu { - position: absolute; - top: 100%; - right: 0; - margin-top: 4px; - padding: 8px; - background: var(--base); - border: 1px solid var(--surface0); - border-radius: 6px; - box-shadow: 0 4px 12px rgba(0, 0, 0, 0.2); - z-index: 100; - max-height: 300px; - overflow-y: auto; - min-width: 200px; - } - - .column-menu::-webkit-scrollbar { - width: 8px; - } - - .column-menu::-webkit-scrollbar-track { - background: var(--mantle); - border-radius: 4px; - } - - .column-menu::-webkit-scrollbar-thumb { - background: var(--surface2); - border-radius: 4px; - } - - .column-menu::-webkit-scrollbar-thumb:hover { - background: var(--surface1); - } - - .column-option { - display: flex; - align-items: center; - gap: 8px; - padding: 6px 8px; - font-size: 12px; - color: var(--text); - cursor: pointer; - border-radius: 4px; - } - - .column-option:hover { - background: var(--surface0); - } - - .column-option input[type="checkbox"] { - appearance: none; - width: 16px; - height: 16px; - border: 1px solid var(--surface2); - border-radius: 4px; - background: var(--mantle); - cursor: pointer; - position: relative; - } - - .column-option input[type="checkbox"]:checked { - background: var(--blue); - border-color: var(--blue); - } - - .column-option input[type="checkbox"]:checked::after { - content: "✓"; - position: absolute; - color: var(--base); - font-size: 12px; - top: 50%; - left: 50%; - transform: translate(-50%, -50%); - } - - .column-option input[type="checkbox"]:disabled { - opacity: 0.5; - cursor: not-allowed; - } - - .refresh-controls { - display: flex; - gap: 8px; - align-items: center; - } - - .refresh-controls :global(svg) { - font-size: 14px; - color: var(--subtext0); - } - - .btn-action { - display: inline-flex; - align-items: center; - justify-content: center; - width: 28px; - height: 28px; - border: none; - background: var(--surface0); - color: var(--text); - border-radius: 6px; - cursor: pointer; - transition: all 0.2s ease; - } - - .btn-action:hover { - background: var(--surface1); - } - - .btn-action.frozen { - background: var(--blue); - color: var(--base); - } - - .select-input { - height: 28px; - padding: 0 8px; - border: 1px solid var(--surface1); - border-radius: 6px; - background: var(--surface0); - color: var(--text); - font-size: 13px; - cursor: pointer; - } - - .select-input:disabled { - opacity: 0.7; - cursor: not-allowed; - } From c909ec9ee470c712fb1aff96d350d886471a32cf Mon Sep 17 00:00:00 2001 From: Abdenasser Date: Wed, 13 Nov 2024 16:58:53 +0100 Subject: [PATCH 12/14] big mess --- src/lib/components/AppInfo.svelte | 19 +- src/lib/components/ThemeSwitcher.svelte | 45 +-- src/lib/components/index.ts | 7 + .../components/modals/KillProcessModal.svelte | 2 +- .../modals/ProcessDetailsModal.svelte | 20 +- src/lib/components/process/ProcessRow.svelte | 4 +- .../components/process/ProcessTable.svelte | 4 +- src/lib/components/stats/CpuPanel.svelte | 3 +- src/lib/components/stats/MemoryPanel.svelte | 4 +- src/lib/components/stats/NetworkPanel.svelte | 17 +- src/lib/components/stats/ProgressBar.svelte | 2 +- src/lib/components/stats/StatsBar.svelte | 12 +- src/lib/components/stats/StoragePanel.svelte | 18 +- src/lib/components/stats/SystemPanel.svelte | 3 +- .../components/toolbar/ColumnToggle.svelte | 7 +- .../toolbar/PaginationControls.svelte | 13 +- .../components/toolbar/RefreshControls.svelte | 20 +- .../components/toolbar/StatusFilter.svelte | 20 +- src/lib/components/toolbar/ToolBar.svelte | 4 +- src/lib/constants/index.ts | 71 +++++ src/lib/constants/toolbar.ts | 9 - src/lib/definitions/columns.ts | 61 ++++ src/lib/definitions/index.ts | 3 + .../config.ts => definitions/settings.ts} | 11 +- .../index.ts => definitions/themes.ts} | 28 +- src/lib/stores/index.ts | 61 +--- src/lib/stores/processes.ts | 186 +++++++++++ src/lib/stores/{config.ts => settings.ts} | 8 +- src/lib/stores/theme.ts | 58 ++++ src/lib/types/index.ts | 73 +++++ src/lib/types/toolbar.ts | 28 -- src/lib/utils/index.ts | 110 +++++-- src/routes/+page.svelte | 294 ++++-------------- 33 files changed, 650 insertions(+), 575 deletions(-) create mode 100644 src/lib/components/index.ts create mode 100644 src/lib/constants/index.ts delete mode 100644 src/lib/constants/toolbar.ts create mode 100644 src/lib/definitions/columns.ts create mode 100644 src/lib/definitions/index.ts rename src/lib/{types/config.ts => definitions/settings.ts} (72%) rename src/lib/{styles/index.ts => definitions/themes.ts} (97%) create mode 100644 src/lib/stores/processes.ts rename src/lib/stores/{config.ts => settings.ts} (82%) create mode 100644 src/lib/stores/theme.ts delete mode 100644 src/lib/types/toolbar.ts diff --git a/src/lib/components/AppInfo.svelte b/src/lib/components/AppInfo.svelte index 9a4f77d..8fac7f6 100644 --- a/src/lib/components/AppInfo.svelte +++ b/src/lib/components/AppInfo.svelte @@ -1,31 +1,16 @@ @@ -54,9 +42,7 @@
Status: - - {@html formatStatus(currentProcess.status)} - + {currentProcess.status}
diff --git a/src/lib/components/process/ProcessRow.svelte b/src/lib/components/process/ProcessRow.svelte index 07ea55b..1f60e0e 100644 --- a/src/lib/components/process/ProcessRow.svelte +++ b/src/lib/components/process/ProcessRow.svelte @@ -1,11 +1,9 @@
diff --git a/src/lib/components/stats/ProgressBar.svelte b/src/lib/components/stats/ProgressBar.svelte index 0834121..830d21d 100644 --- a/src/lib/components/stats/ProgressBar.svelte +++ b/src/lib/components/stats/ProgressBar.svelte @@ -21,7 +21,7 @@
+ >
{Math.round(value)}%
diff --git a/src/lib/components/stats/StatsBar.svelte b/src/lib/components/stats/StatsBar.svelte index 949d770..b4fa191 100644 --- a/src/lib/components/stats/StatsBar.svelte +++ b/src/lib/components/stats/StatsBar.svelte @@ -1,10 +1,12 @@ diff --git a/src/lib/components/stats/StoragePanel.svelte b/src/lib/components/stats/StoragePanel.svelte index cf1fae1..9ed0431 100644 --- a/src/lib/components/stats/StoragePanel.svelte +++ b/src/lib/components/stats/StoragePanel.svelte @@ -1,26 +1,12 @@ diff --git a/src/lib/components/stats/SystemPanel.svelte b/src/lib/components/stats/SystemPanel.svelte index d8faf32..b5fd204 100644 --- a/src/lib/components/stats/SystemPanel.svelte +++ b/src/lib/components/stats/SystemPanel.svelte @@ -1,7 +1,6 @@ - {#if currentProcess} + {#if process}
@@ -26,23 +24,23 @@
Name: - {currentProcess.name} + {process.name}
PID: - {currentProcess.pid} + {process.pid}
Parent PID: - {currentProcess.ppid} + {process.ppid}
User: - {currentProcess.user} + {process.user}
Status: - {currentProcess.status} + {process.status}
@@ -61,12 +59,12 @@
50} - class:critical={currentProcess.cpu_usage > 80} + style="width: {process.cpu_usage}%" + class:high={process.cpu_usage > 50} + class:critical={process.cpu_usage > 80} >
- {currentProcess.cpu_usage.toFixed(1)}% + {process.cpu_usage.toFixed(1)}%
@@ -77,8 +75,8 @@ Memory Usage
-
Physical: {formatBytes(currentProcess.memory_usage)}
-
Virtual: {formatBytes(currentProcess.virtual_memory)}
+
Physical: {formatBytes(process.memory_usage)}
+
Virtual: {formatBytes(process.virtual_memory)}
@@ -89,8 +87,8 @@ Disk I/O
-
Read: {formatBytes(currentProcess.disk_usage[0])}
-
Written: {formatBytes(currentProcess.disk_usage[1])}
+
Read: {formatBytes(process.disk_usage[0])}
+
Written: {formatBytes(process.disk_usage[1])}
@@ -101,8 +99,8 @@ Time Information
-
Started: {formatDate(currentProcess.start_time)}
-
Running: {formatUptime(currentProcess.run_time)}
+
Started: {formatDate(process.start_time)}
+
Running: {formatUptime(process.run_time)}
@@ -114,17 +112,17 @@
Command: - {currentProcess.command} + {process.command}
Root: - {currentProcess.root} + {process.root}
- {#if currentProcess.environ.length > 0} + {#if process.environ.length > 0}
Environment:
- {#each currentProcess.environ as env} + {#each process.environ as env}
{env}
{/each}
diff --git a/src/lib/stores/processes.ts b/src/lib/stores/processes.ts index 96d34d9..ec263f0 100644 --- a/src/lib/stores/processes.ts +++ b/src/lib/stores/processes.ts @@ -47,139 +47,171 @@ const initialState: ProcessStore = { function createProcessStore() { const { subscribe, set, update } = writable(initialState); - return { - subscribe, - set, - update, + // Define all methods first + const setIsLoading = (isLoading: boolean) => + update((state) => ({ ...state, isLoading })); - setIsLoading: (isLoading: boolean) => - update((state) => ({ ...state, isLoading })), + const getProcesses = async () => { + try { + const result = await invoke<[Process[], SystemStats]>("get_processes"); + update((state) => { + let updatedSelectedProcess = state.selectedProcess; + if (state.selectedProcessPid) { + updatedSelectedProcess = + result[0].find((p) => p.pid === state.selectedProcessPid) || null; + } - async getProcesses() { - try { - const result = await invoke<[Process[], SystemStats]>("get_processes"); - update((state) => ({ + return { ...state, processes: result[0], systemStats: result[1], error: null, - })); - } catch (e: unknown) { - update((state) => ({ - ...state, - error: e instanceof Error ? e.message : String(e), - })); - } - }, - - async killProcess(pid: number) { - try { - const success = await invoke("kill_process", { pid }); - if (success) { - await this.getProcesses(); - } - } catch (e: unknown) { - update((state) => ({ - ...state, - error: e instanceof Error ? e.message : String(e), - })); - } - }, - - toggleSort(field: keyof Process) { - update((state) => ({ - ...state, - sortConfig: { - field, - direction: - state.sortConfig.field === field - ? state.sortConfig.direction === "asc" - ? "desc" - : "asc" - : "desc", - }, - })); - }, - - togglePin(command: string) { - update((state) => { - const newPinnedProcesses = new Set(state.pinnedProcesses); - if (newPinnedProcesses.has(command)) { - newPinnedProcesses.delete(command); - } else { - newPinnedProcesses.add(command); - } - return { ...state, pinnedProcesses: newPinnedProcesses }; + selectedProcess: updatedSelectedProcess, + }; }); - }, - - setSearchTerm: (searchTerm: string) => - update((state) => ({ ...state, searchTerm, currentPage: 1 })), - setIsFrozen: (isFrozen: boolean) => - update((state) => ({ ...state, isFrozen })), - setCurrentPage: (currentPage: number) => - update((state) => ({ ...state, currentPage })), - - showProcessDetails(process: Process) { + } catch (e: unknown) { update((state) => ({ ...state, - selectedProcessPid: process.pid, - selectedProcess: process, - showInfoModal: true, + error: e instanceof Error ? e.message : String(e), })); - }, + } + }; - closeProcessDetails() { + const killProcess = async (pid: number) => { + try { + update((state) => ({ ...state, isKilling: true })); + const success = await invoke("kill_process", { pid }); + if (success) { + await getProcesses(); + } else { + throw new Error("Failed to kill process"); + } + } catch (e: unknown) { update((state) => ({ ...state, - showInfoModal: false, - selectedProcess: null, - selectedProcessPid: null, + error: e instanceof Error ? e.message : String(e), })); - }, + } finally { + update((state) => ({ ...state, isKilling: false })); + } + }; - confirmKillProcess(process: Process) { - update((state) => ({ - ...state, - processToKill: process, - showConfirmModal: true, - })); - }, + const toggleSort = (field: keyof Process) => { + update((state) => ({ + ...state, + sortConfig: { + field, + direction: + state.sortConfig.field === field + ? state.sortConfig.direction === "asc" + ? "desc" + : "asc" + : "desc", + }, + })); + }; - closeConfirmKill() { + const togglePin = (command: string) => { + update((state) => { + const newPinnedProcesses = new Set(state.pinnedProcesses); + if (newPinnedProcesses.has(command)) { + newPinnedProcesses.delete(command); + } else { + newPinnedProcesses.add(command); + } + return { ...state, pinnedProcesses: newPinnedProcesses }; + }); + }; + + const setSearchTerm = (searchTerm: string) => + update((state) => ({ ...state, searchTerm, currentPage: 1 })); + + const setIsFrozen = (isFrozen: boolean) => + update((state) => ({ ...state, isFrozen })); + + const setCurrentPage = (currentPage: number) => + update((state) => ({ ...state, currentPage })); + + const showProcessDetails = (process: Process) => { + update((state) => ({ + ...state, + selectedProcessPid: process.pid, + selectedProcess: process, + showInfoModal: true, + })); + }; + + const closeProcessDetails = () => { + update((state) => ({ + ...state, + showInfoModal: false, + selectedProcess: null, + selectedProcessPid: null, + })); + }; + + const confirmKillProcess = (process: Process) => { + update((state) => ({ + ...state, + processToKill: process, + showConfirmModal: true, + })); + }; + + const closeConfirmKill = () => { + update((state) => ({ + ...state, + showConfirmModal: false, + processToKill: null, + })); + }; + + const handleConfirmKill = async () => { + let processToKill: Process | null = null; + + let currentState: ProcessStore | undefined; + const unsubscribe = subscribe((state) => { + currentState = state; + }); + unsubscribe(); + + if (currentState?.processToKill && "pid" in currentState.processToKill) { + processToKill = currentState.processToKill; + } + + if (!processToKill?.pid) { + return; + } + + try { + await killProcess(processToKill.pid); + } finally { update((state) => ({ ...state, showConfirmModal: false, processToKill: null, })); - }, + } + }; - async handleConfirmKill() { - update((state) => ({ ...state, isKilling: true })); - - try { - const pid = this.getState().processToKill?.pid; - if (pid) { - await this.killProcess(pid); - } - } finally { - update((state) => ({ - ...state, - isKilling: false, - showConfirmModal: false, - processToKill: null, - })); - } - }, - - // Helper to get current state - getState() { - let currentState: ProcessStore | undefined; - subscribe((state) => { - currentState = state; - })(); - return currentState!; - }, + // Return all methods + return { + subscribe, + set, + update, + setIsLoading, + getProcesses, + killProcess, + toggleSort, + togglePin, + setSearchTerm, + setIsFrozen, + setCurrentPage, + showProcessDetails, + closeProcessDetails, + confirmKillProcess, + closeConfirmKill, + handleConfirmKill, }; } diff --git a/src/lib/utils/index.ts b/src/lib/utils/index.ts index d3aecaa..80c1e4f 100644 --- a/src/lib/utils/index.ts +++ b/src/lib/utils/index.ts @@ -121,19 +121,16 @@ export function sortProcesses( sortConfig: SortConfig, pinnedProcesses: Set, ): Process[] { + // Clear the cache before sorting + isPinned.clear(); + return [...processes].sort((a, b) => { // Cache pinned status - let aPin = isPinned.get(a.command); - if (aPin === undefined) { - aPin = pinnedProcesses.has(a.command); - isPinned.set(a.command, aPin); - } + let aPin = pinnedProcesses.has(a.command); + isPinned.set(a.command, aPin); - let bPin = isPinned.get(b.command); - if (bPin === undefined) { - bPin = pinnedProcesses.has(b.command); - isPinned.set(b.command, bPin); - } + let bPin = pinnedProcesses.has(b.command); + isPinned.set(b.command, bPin); // Quick pin comparison if (aPin !== bPin) {
onToggleSort(column.id)}> -
- {column.label} - - {getSortIndicator(column.id)} - -
-
Actions
onToggleSort(column.id)}> +
+ {column.label} + + {getSortIndicator(column.id)} + +
+
Actions
+
+ + + +
+
+ {#if column.id === "name"} +
+ + {process.name} +
+ {:else if column.format} + {@html column.format(process[column.id])} + {:else} + {process[column.id]} + {/if} +