mirror of
https://github.com/kunkunsh/kunkun-ext-neohtop.git
synced 2025-07-20 21:41:29 +00:00
ProcessRow component
This commit is contained in:
parent
44cb0c965e
commit
5aa6a7a8e7
@ -0,0 +1,81 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import type { Process, Column } from "$lib/types";
|
||||||
|
import ProcessIcon from "./ProcessIcon.svelte";
|
||||||
|
import ActionButtons from "./ActionButtons.svelte";
|
||||||
|
|
||||||
|
export let process: Process;
|
||||||
|
export let columns: Column[];
|
||||||
|
export let systemStats: { memory_total: number } | null;
|
||||||
|
export let isPinned: boolean;
|
||||||
|
export let isHighUsage: boolean;
|
||||||
|
|
||||||
|
export let onTogglePin: (command: string) => void;
|
||||||
|
export let onShowDetails: (process: Process) => void;
|
||||||
|
export let onKillProcess: (process: Process) => void;
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<tr class:high-usage={isHighUsage} class:pinned={isPinned}>
|
||||||
|
{#each columns.filter((col) => col.visible) as column}
|
||||||
|
<td class="truncate">
|
||||||
|
{#if column.id === "name"}
|
||||||
|
<div class="name-cell">
|
||||||
|
<ProcessIcon processName={process.name} />
|
||||||
|
<span class="process-name">{process.name}</span>
|
||||||
|
</div>
|
||||||
|
{:else if column.format}
|
||||||
|
{@html column.format(process[column.id])}
|
||||||
|
{:else}
|
||||||
|
{process[column.id]}
|
||||||
|
{/if}
|
||||||
|
</td>
|
||||||
|
{/each}
|
||||||
|
<ActionButtons
|
||||||
|
{process}
|
||||||
|
{isPinned}
|
||||||
|
{onTogglePin}
|
||||||
|
{onShowDetails}
|
||||||
|
{onKillProcess}
|
||||||
|
/>
|
||||||
|
</tr>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
td {
|
||||||
|
padding: 6px 12px;
|
||||||
|
border-bottom: 1px solid var(--surface0);
|
||||||
|
color: var(--text);
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.truncate {
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
max-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
tr:hover {
|
||||||
|
background-color: var(--surface0);
|
||||||
|
}
|
||||||
|
|
||||||
|
.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);
|
||||||
|
}
|
||||||
|
|
||||||
|
.name-cell {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
</style>
|
@ -1,8 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import type { Process, Column } from "$lib/types";
|
import type { Process, Column } from "$lib/types";
|
||||||
import ProcessIcon from "./ProcessIcon.svelte";
|
|
||||||
import TableHeader from "./TableHeader.svelte";
|
import TableHeader from "./TableHeader.svelte";
|
||||||
import ActionButtons from "./ActionButtons.svelte";
|
import ProcessRow from "./ProcessRow.svelte";
|
||||||
|
|
||||||
export let processes: Process[];
|
export let processes: Process[];
|
||||||
export let columns: Column[];
|
export let columns: Column[];
|
||||||
@ -21,33 +20,17 @@
|
|||||||
<TableHeader {columns} {sortConfig} {onToggleSort} />
|
<TableHeader {columns} {sortConfig} {onToggleSort} />
|
||||||
<tbody>
|
<tbody>
|
||||||
{#each processes as process (process.pid)}
|
{#each processes as process (process.pid)}
|
||||||
<tr
|
<ProcessRow
|
||||||
class:high-usage={process.cpu_usage > 50 ||
|
{process}
|
||||||
|
{columns}
|
||||||
|
{systemStats}
|
||||||
|
isPinned={pinnedProcesses.has(process.command)}
|
||||||
|
isHighUsage={process.cpu_usage > 50 ||
|
||||||
process.memory_usage / (systemStats?.memory_total || 0) > 0.1}
|
process.memory_usage / (systemStats?.memory_total || 0) > 0.1}
|
||||||
class:pinned={pinnedProcesses.has(process.command)}
|
{onTogglePin}
|
||||||
>
|
{onShowDetails}
|
||||||
{#each columns.filter((col) => col.visible) as column}
|
{onKillProcess}
|
||||||
<td class="truncate">
|
/>
|
||||||
{#if column.id === "name"}
|
|
||||||
<div class="name-cell">
|
|
||||||
<ProcessIcon processName={process.name} />
|
|
||||||
<span class="process-name">{process.name}</span>
|
|
||||||
</div>
|
|
||||||
{:else if column.format}
|
|
||||||
{@html column.format(process[column.id])}
|
|
||||||
{:else}
|
|
||||||
{process[column.id]}
|
|
||||||
{/if}
|
|
||||||
</td>
|
|
||||||
{/each}
|
|
||||||
<ActionButtons
|
|
||||||
{process}
|
|
||||||
isPinned={pinnedProcesses.has(process.command)}
|
|
||||||
{onTogglePin}
|
|
||||||
{onShowDetails}
|
|
||||||
{onKillProcess}
|
|
||||||
/>
|
|
||||||
</tr>
|
|
||||||
{/each}
|
{/each}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
@ -58,13 +41,10 @@
|
|||||||
flex: 1;
|
flex: 1;
|
||||||
overflow-x: auto;
|
overflow-x: auto;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
|
scrollbar-width: thin;
|
||||||
/* Scrollbar styles */
|
scrollbar-color: var(--surface2) var(--mantle);
|
||||||
scrollbar-width: thin; /* Firefox */
|
|
||||||
scrollbar-color: var(--surface2) var(--mantle); /* Firefox */
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Webkit scrollbar styles (Chrome, Safari, Edge) */
|
|
||||||
.table-container::-webkit-scrollbar {
|
.table-container::-webkit-scrollbar {
|
||||||
width: 8px;
|
width: 8px;
|
||||||
height: 8px;
|
height: 8px;
|
||||||
@ -89,11 +69,6 @@
|
|||||||
background: var(--mantle);
|
background: var(--mantle);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* When both scrollbars are present, add some padding to prevent overlap */
|
|
||||||
.table-container::-webkit-scrollbar-corner {
|
|
||||||
background-color: var(--mantle);
|
|
||||||
}
|
|
||||||
|
|
||||||
table {
|
table {
|
||||||
width: max-content;
|
width: max-content;
|
||||||
min-width: 100%;
|
min-width: 100%;
|
||||||
@ -101,51 +76,4 @@
|
|||||||
border-collapse: collapse;
|
border-collapse: collapse;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
}
|
}
|
||||||
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
|
|
||||||
.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);
|
|
||||||
}
|
|
||||||
|
|
||||||
td:last-child {
|
|
||||||
width: 120px;
|
|
||||||
min-width: 120px;
|
|
||||||
max-width: 120px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.name-cell {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user