mirror of
https://github.com/amir20/dozzle.git
synced 2025-12-21 21:33:18 +01:00
113 lines
2.2 KiB
Vue
113 lines
2.2 KiB
Vue
<template>
|
|
<ul class="events group pt-4" :class="{ 'disable-wrap': !softWrap, [size]: true, compact }">
|
|
<li
|
|
v-for="item in messages"
|
|
ref="list"
|
|
:key="item.id"
|
|
:data-key="item.id"
|
|
:data-time="item.date.getTime()"
|
|
class="group/entry"
|
|
>
|
|
<component :is="item.getComponent()" :log-entry="item" />
|
|
</li>
|
|
</ul>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { type JSONObject, LogEntry } from "@/models/LogEntry";
|
|
|
|
const { loading, progress, currentDate } = useScrollContext();
|
|
|
|
const { messages } = defineProps<{
|
|
messages: LogEntry<string | JSONObject>[];
|
|
}>();
|
|
|
|
watchEffect(() => {
|
|
loading.value = messages.length === 0;
|
|
});
|
|
|
|
const { containers } = useLoggingContext();
|
|
|
|
const list = ref<HTMLElement[]>([]);
|
|
|
|
useIntersectionObserver(
|
|
list,
|
|
(entries) => {
|
|
if (containers.value.length != 1) return;
|
|
const container = containers.value[0];
|
|
for (const entry of entries) {
|
|
if (entry.isIntersecting) {
|
|
const time = entry.target.getAttribute("data-time");
|
|
if (time) {
|
|
const date = new Date(parseInt(time));
|
|
const diff = new Date().getTime() - container.created.getTime();
|
|
progress.value = (date.getTime() - container.created.getTime()) / diff;
|
|
currentDate.value = date;
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{
|
|
rootMargin: "-10% 0px -10% 0px",
|
|
threshold: 1,
|
|
},
|
|
);
|
|
</script>
|
|
<style scoped>
|
|
@reference "@/main.css";
|
|
.events {
|
|
font-family:
|
|
ui-monospace,
|
|
SFMono-Regular,
|
|
SF Mono,
|
|
Consolas,
|
|
Liberation Mono,
|
|
monaco,
|
|
Menlo,
|
|
monospace;
|
|
|
|
> li {
|
|
@apply has-[.clickable]:hover:bg-primary/10 flex px-2 py-1 break-words last:snap-end odd:bg-gray-400/[0.07] has-[.clickable]:cursor-pointer md:px-4;
|
|
&:last-child {
|
|
scroll-margin-block-end: 5rem;
|
|
}
|
|
}
|
|
|
|
&.small {
|
|
@apply text-[0.7em];
|
|
}
|
|
|
|
&.medium {
|
|
@apply text-[0.8em];
|
|
}
|
|
|
|
&.large {
|
|
@apply text-lg;
|
|
}
|
|
|
|
&.compact {
|
|
> li {
|
|
@apply py-0;
|
|
}
|
|
|
|
:deep(.tag) {
|
|
@apply rounded-none;
|
|
}
|
|
}
|
|
|
|
:deep(mark) {
|
|
@apply bg-secondary inline-block rounded-xs;
|
|
animation: pops 200ms ease-out;
|
|
}
|
|
}
|
|
|
|
@keyframes pops {
|
|
0% {
|
|
transform: scale(1.5);
|
|
}
|
|
100% {
|
|
transform: scale(1.05);
|
|
}
|
|
}
|
|
</style>
|