mirror of
https://github.com/PhotonVision/photonvision
synced 2026-06-21 01:01:41 +00:00
## Description Upgrades to Vue 3 and necessary associated dependencies. Also fixes some issues with the layout and adds validation for object detection models. Closes #885, closes #1943, closes #1449. ## Meta Merge checklist: - [x] Pull Request title is [short, imperative summary](https://cbea.ms/git-commit/) of proposed changes - [x] The description documents the _what_ and _why_ - [ ] If this PR changes behavior or adds a feature, user documentation is updated - [ ] If this PR touches photon-serde, all messages have been regenerated and hashes have not changed unexpectedly - [ ] If this PR touches configuration, this is backwards compatible with settings back to v2024.3.1 - [ ] If this PR touches pipeline settings or anything related to data exchange, the frontend typing is updated - [ ] If this PR addresses a bug, a regression test for it is added --------- Co-authored-by: Matt M <matthew.morley.ca@gmail.com> Co-authored-by: Gold856 <117957790+Gold856@users.noreply.github.com> Co-authored-by: samfreund <techguy763@gmail.com>
194 lines
6.0 KiB
Vue
194 lines
6.0 KiB
Vue
<script setup lang="ts">
|
|
import { computed, inject, ref, watch } from "vue";
|
|
import { LogLevel, type LogMessage } from "@/types/SettingTypes";
|
|
import { useStateStore } from "@/stores/StateStore";
|
|
import LogEntry from "@/components/app/photon-log-entry.vue";
|
|
import VirtualList from "vue3-virtual-scroll-list";
|
|
|
|
const backendHost = inject<string>("backendHost");
|
|
|
|
const searchQuery = ref("");
|
|
const timeInput = ref<string>();
|
|
const autoScroll = ref(true);
|
|
const logList = ref();
|
|
const logKeeps = ref(40);
|
|
const exportLogFile = ref();
|
|
const selectedLogLevels = ref({
|
|
[LogLevel.ERROR]: true,
|
|
[LogLevel.WARN]: true,
|
|
[LogLevel.INFO]: true,
|
|
[LogLevel.DEBUG]: false
|
|
});
|
|
|
|
const logs = computed<LogMessage[]>(() =>
|
|
useStateStore()
|
|
.logMessages.filter(
|
|
(message) =>
|
|
selectedLogLevels.value[message.level] &&
|
|
message.message.toLowerCase().includes(searchQuery.value?.toLowerCase() || "") &&
|
|
(timeInput.value === undefined ||
|
|
message.timestamp.getTime() >=
|
|
new Date().setHours(
|
|
parseInt(timeInput.value.substring(0, 2)),
|
|
parseInt(timeInput.value.substring(3, 5)),
|
|
parseInt(timeInput.value.substring(6, 8))
|
|
))
|
|
)
|
|
.map((item, index) => ({ ...item, index: index }))
|
|
);
|
|
|
|
watch(logs, () => {
|
|
if (!logList.value) return;
|
|
|
|
// Dynamic list render size based on console size
|
|
logKeeps.value = Math.ceil(logList.value.$el.clientHeight / 17.5) + 20;
|
|
|
|
const bottomOffset = Math.abs(
|
|
logList.value.$el.scrollHeight - logList.value.$el.scrollTop - logList.value.$el.clientHeight
|
|
);
|
|
autoScroll.value = bottomOffset < 50;
|
|
|
|
if (autoScroll.value) logList.value.scrollToBottom();
|
|
});
|
|
|
|
const getLogLevelFromIndex = (index: number): string => {
|
|
return LogLevel[index];
|
|
};
|
|
|
|
const handleLogExport = () => {
|
|
exportLogFile.value.click();
|
|
};
|
|
|
|
const handleLogClear = () => {
|
|
useStateStore().logMessages = [];
|
|
};
|
|
|
|
document.addEventListener("keydown", (e) => {
|
|
switch (e.key) {
|
|
case "`":
|
|
useStateStore().$patch((state) => (state.showLogModal = !state.showLogModal));
|
|
break;
|
|
}
|
|
});
|
|
</script>
|
|
|
|
<template>
|
|
<v-dialog v-model="useStateStore().showLogModal" width="1500" dark>
|
|
<v-card class="dialog-container pa-6" color="primary" flat>
|
|
<!-- Logs header -->
|
|
<v-row class="pb-3">
|
|
<v-col cols="4">
|
|
<v-card-title class="pa-0">Program Logs</v-card-title>
|
|
</v-col>
|
|
<v-col class="align-self-center pl-3" style="text-align: right">
|
|
<v-btn variant="text" color="white" @click="handleLogExport">
|
|
<v-icon start class="menu-icon"> mdi-download </v-icon>
|
|
<span class="menu-label">Download</span>
|
|
|
|
<!-- Special hidden link that gets 'clicked' when the user exports journalctl logs -->
|
|
<a
|
|
ref="exportLogFile"
|
|
style="color: black; text-decoration: none; display: none"
|
|
:href="`http://${backendHost}/api/utils/photonvision-journalctl.txt`"
|
|
download="photonvision-journalctl.txt"
|
|
target="_blank"
|
|
/>
|
|
</v-btn>
|
|
<v-btn variant="text" color="white" @click="handleLogClear">
|
|
<v-icon start class="menu-icon"> mdi-trash-can-outline </v-icon>
|
|
<span class="menu-label">Clear Client Logs</span>
|
|
</v-btn>
|
|
<v-btn variant="text" color="white" @click="() => (useStateStore().showLogModal = false)">
|
|
<v-icon start class="menu-icon"> mdi-close </v-icon>
|
|
<span class="menu-label">Close</span>
|
|
</v-btn>
|
|
</v-col>
|
|
</v-row>
|
|
|
|
<v-divider />
|
|
|
|
<div class="dialog-data">
|
|
<!-- Log view options -->
|
|
<v-row no-gutters class="pt-4 pt-md-0" style="display: flex; justify-content: space-between">
|
|
<v-col cols="12" md="7" style="display: flex; align-items: center">
|
|
<v-text-field
|
|
v-model="searchQuery"
|
|
density="compact"
|
|
clearable
|
|
hide-details="auto"
|
|
prepend-icon="mdi-magnify"
|
|
color="accent"
|
|
label="Search"
|
|
variant="underlined"
|
|
/>
|
|
<input v-model="timeInput" type="time" step="1" class="text-white pl-3" />
|
|
<v-btn icon variant="flat" @click="timeInput = undefined">
|
|
<v-icon>mdi-close-circle-outline</v-icon>
|
|
</v-btn>
|
|
</v-col>
|
|
<v-col v-for="level in [0, 1, 2, 3]" :key="level" class="pr-3">
|
|
<div class="pb-0 pt-0" style="display: flex; align-items: center; flex: min-content">
|
|
{{ getLogLevelFromIndex(level)
|
|
}}<v-switch v-model="selectedLogLevels[level]" class="pl-2" hide-details color="#ffd843"></v-switch>
|
|
</div>
|
|
</v-col>
|
|
</v-row>
|
|
|
|
<!-- Log entry list display -->
|
|
<div class="log-display">
|
|
<v-card-text v-if="!logs.length" style="font-size: 18px; font-weight: 150; height: 100%; text-align: center">
|
|
No available logs
|
|
</v-card-text>
|
|
<virtual-list
|
|
v-else
|
|
ref="logList"
|
|
style="height: 100%; overflow-y: auto"
|
|
data-key="index"
|
|
:data-sources="logs"
|
|
:data-component="LogEntry"
|
|
:estimate-size="35"
|
|
:keeps="logKeeps"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</v-card>
|
|
</v-dialog>
|
|
</template>
|
|
|
|
<style scoped lang="scss">
|
|
.dialog-container {
|
|
height: 90vh;
|
|
min-height: 300px !important;
|
|
}
|
|
|
|
.dialog-data {
|
|
/* Dialog size - dialog padding - header - divider */
|
|
height: calc(max(90vh, 300px) - 48px - 48px - 1px);
|
|
}
|
|
|
|
.log-display {
|
|
/* Dialog data size - options */
|
|
height: calc(100% - 66px);
|
|
padding: 10px;
|
|
background-color: #232c37 !important;
|
|
border-radius: 5px;
|
|
}
|
|
|
|
@media only screen and (max-width: 960px) {
|
|
.log-display {
|
|
/* Dialog data size - options */
|
|
height: calc(100% - 118px);
|
|
}
|
|
}
|
|
|
|
@media only screen and (max-width: 700px) {
|
|
.menu-label {
|
|
display: none;
|
|
}
|
|
|
|
.menu-icon {
|
|
margin: 0 !important;
|
|
}
|
|
}
|
|
</style>
|