mirror of
https://github.com/amir20/dozzle.git
synced 2025-12-21 13:23:07 +01:00
222 lines
6.3 KiB
Vue
222 lines
6.3 KiB
Vue
!
|
|
<template>
|
|
<div class="flex flex-row">
|
|
<div v-if="Object.keys(hosts).length > 1" class="flex-1">
|
|
<div role="tablist" class="tabs-boxed tabs block" v-if="Object.keys(hosts).length < 4">
|
|
<input
|
|
type="radio"
|
|
name="host"
|
|
role="tab"
|
|
class="tab !rounded"
|
|
aria-label="Show All"
|
|
v-model="selectedHost"
|
|
:value="null"
|
|
/>
|
|
<input
|
|
type="radio"
|
|
name="host"
|
|
role="tab"
|
|
class="tab !rounded"
|
|
:aria-label="host.name"
|
|
v-for="host in hosts"
|
|
:value="host.id"
|
|
:key="host.id"
|
|
v-model="selectedHost"
|
|
/>
|
|
</div>
|
|
<dropdown-menu
|
|
class="btn-sm"
|
|
v-model="selectedHost"
|
|
:options="[
|
|
{ label: 'Show All', value: null },
|
|
...Object.values(hosts).map((host) => ({ label: host.name, value: host.id })),
|
|
]"
|
|
v-else
|
|
/>
|
|
</div>
|
|
<div class="flex-1 text-right" v-show="containers.length > pageSizes[0]">
|
|
{{ $t("label.per-page") }}
|
|
<dropdown-menu
|
|
class="dropdown-left btn-xs md:btn-sm"
|
|
v-model="perPage"
|
|
:options="pageSizes.map((i) => ({ label: i.toLocaleString(), value: i }))"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<table class="table table-lg bg-base">
|
|
<thead>
|
|
<tr :data-direction="direction > 0 ? 'asc' : 'desc'">
|
|
<th
|
|
v-for="(value, key) in fields"
|
|
:key="key"
|
|
@click.prevent="sort(key)"
|
|
:class="{ 'selected-sort': key === sortField }"
|
|
v-show="isVisible(key)"
|
|
>
|
|
<a class="inline-flex cursor-pointer gap-2 text-sm uppercase">
|
|
<span>{{ $t(value.label) }}</span>
|
|
<span class="h-4" data-icon>
|
|
<mdi:arrow-up />
|
|
</span>
|
|
</a>
|
|
</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr v-for="container in paginated" :key="container.id">
|
|
<td v-if="isVisible('name')">
|
|
<router-link :to="{ name: 'container-id', params: { id: container.id } }" :title="container.name">
|
|
{{ container.name }}
|
|
</router-link>
|
|
</td>
|
|
<td v-if="isVisible('host')">{{ container.hostLabel }}</td>
|
|
<td v-if="isVisible('state')">{{ container.state }}</td>
|
|
<td v-if="isVisible('created')">
|
|
<distance-time :date="container.created" strict :suffix="false"></distance-time>
|
|
</td>
|
|
<td v-if="isVisible('cpu')">
|
|
<progress
|
|
class="progress progress-primary"
|
|
:value="container.movingAverage.cpu"
|
|
:max="100 * hosts[container.host].nCPU"
|
|
></progress>
|
|
</td>
|
|
<td v-if="isVisible('mem')">
|
|
<progress class="progress progress-primary" :value="container.movingAverage.memory" max="100"></progress>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
<div class="p-4 text-center">
|
|
<nav class="join" v-if="isPaginated">
|
|
<input
|
|
class="btn btn-square join-item"
|
|
type="radio"
|
|
v-model="currentPage"
|
|
:aria-label="`${i}`"
|
|
:value="i"
|
|
v-for="i in totalPages"
|
|
/>
|
|
</nav>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { Container } from "@/models/Container";
|
|
import { toRefs } from "@vueuse/core";
|
|
|
|
const { hosts } = useHosts();
|
|
const selectedHost = ref(null);
|
|
|
|
const fields = {
|
|
name: {
|
|
label: "label.container-name",
|
|
sortFunc: (a: Container, b: Container) => a.name.localeCompare(b.name) * direction.value,
|
|
mobileVisible: true,
|
|
},
|
|
host: {
|
|
label: "label.host",
|
|
sortFunc: (a: Container, b: Container) => a.hostLabel.localeCompare(b.hostLabel) * direction.value,
|
|
mobileVisible: false,
|
|
},
|
|
state: {
|
|
label: "label.status",
|
|
sortFunc: (a: Container, b: Container) => a.state.localeCompare(b.state) * direction.value,
|
|
mobileVisible: false,
|
|
},
|
|
created: {
|
|
label: "label.created",
|
|
sortFunc: (a: Container, b: Container) => (a.created.getTime() - b.created.getTime()) * direction.value,
|
|
mobileVisible: true,
|
|
},
|
|
cpu: {
|
|
label: "label.avg-cpu",
|
|
sortFunc: (a: Container, b: Container) => (a.movingAverage.cpu - b.movingAverage.cpu) * direction.value,
|
|
mobileVisible: false,
|
|
},
|
|
mem: {
|
|
label: "label.avg-mem",
|
|
sortFunc: (a: Container, b: Container) => (a.movingAverage.memory - b.movingAverage.memory) * direction.value,
|
|
mobileVisible: false,
|
|
},
|
|
};
|
|
|
|
const { containers } = defineProps<{
|
|
containers: Container[];
|
|
}>();
|
|
type keys = keyof typeof fields;
|
|
|
|
const perPage = useStorage("DOZZLE_TABLE_PAGE_SIZE", 15);
|
|
const pageSizes = [15, 30, 50, 100];
|
|
|
|
const storage = useStorage<{ column: keys; direction: 1 | -1 }>("DOZZLE_TABLE_CONTAINERS_SORT", {
|
|
column: "created",
|
|
direction: -1,
|
|
});
|
|
const { column: sortField, direction } = toRefs(storage);
|
|
const counter = useInterval(10000);
|
|
const filteredContainers = computed(() =>
|
|
containers.filter((c) => selectedHost.value === null || c.host === selectedHost.value),
|
|
);
|
|
const sortedContainers = computedWithControl(
|
|
() => [filteredContainers.value.length, sortField.value, direction.value, counter.value],
|
|
() => filteredContainers.value.sort((a, b) => fields[sortField.value].sortFunc(a, b)),
|
|
);
|
|
|
|
const totalPages = computed(() => Math.ceil(sortedContainers.value.length / perPage.value));
|
|
const isPaginated = computed(() => totalPages.value > 1);
|
|
const currentPage = ref(1);
|
|
watch(perPage, () => (currentPage.value = 1));
|
|
const paginated = computed(() => {
|
|
const start = (currentPage.value - 1) * perPage.value;
|
|
const end = start + perPage.value;
|
|
|
|
return sortedContainers.value.slice(start, end);
|
|
});
|
|
|
|
function sort(field: keys) {
|
|
if (sortField.value === field) {
|
|
direction.value *= -1;
|
|
} else {
|
|
sortField.value = field;
|
|
direction.value = 1;
|
|
}
|
|
}
|
|
function isVisible(field: keys) {
|
|
return fields[field].mobileVisible || !isMobile.value;
|
|
}
|
|
</script>
|
|
|
|
<style lang="postcss" scoped>
|
|
[data-icon] {
|
|
display: none;
|
|
transition: transform 0.2s ease-in-out;
|
|
[data-direction="desc"] & {
|
|
transform: rotate(180deg);
|
|
}
|
|
}
|
|
|
|
th {
|
|
@apply border-b-2 border-base-lighter;
|
|
&.selected-sort {
|
|
font-weight: bold;
|
|
@apply border-primary;
|
|
[data-icon] {
|
|
display: inline-block;
|
|
}
|
|
}
|
|
}
|
|
|
|
tbody td {
|
|
max-width: 0;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
white-space: nowrap;
|
|
}
|
|
|
|
a {
|
|
@apply hover:text-primary;
|
|
}
|
|
</style>
|