<template> <div class="shadow-box mb-3" :style="boxStyle"> <div class="list-header"> <div class="search-wrapper float-start" style="margin-left: 5px;"> <font-awesome-icon icon="filter" /> <select v-model="selectedList" class="form-control" style="margin-left: 5px;"> <option value="monitor" selected>{{ $t('Monitors') }}</option> <option value="maintenance">{{ $t('Maintenance') }}</option> </select> </div> <div class="search-wrapper"> <a v-if="searchText == ''" class="search-icon"> <font-awesome-icon icon="search" /> </a> <a v-if="searchText != ''" class="search-icon" @click="clearSearchText"> <font-awesome-icon icon="times" /> </a> <form> <input v-model="searchText" class="form-control search-input" :placeholder="$t('Search...')" autocomplete="off" /> </form> </div> </div> <div class="monitor-list" :class="{ scrollbar: scrollbar }"> <div v-if="Object.keys($root.monitorList).length === 0 && selectedList === 'monitor'" class="text-center mt-3"> {{ $t("No Monitors, please") }} <router-link to="/add">{{ $t("add one") }}</router-link> </div> <div v-if="Object.keys($root.maintenanceList).length === 0 && selectedList === 'maintenance'" class="text-center mt-3"> {{ $t("No Maintenance, please") }} <router-link to="/addMaintenance">{{ $t("add one") }}</router-link> </div> <template v-if="selectedList === 'maintenance'"> <router-link v-for="(item, index) in sortedMaintenanceList" :key="index" :to="maintenanceURL(item.id)" class="item" :class="{ 'disabled': !$root.isActiveMaintenance(item.end_date) }" > <div class="row"> <div class="col-9 col-md-8 small-padding"> <div class="info"> <Uptime :monitor="null" type="maintenance" :pill="true" /> {{ item.title }} </div> </div> </div> </router-link> </template> <template v-if="selectedList === 'monitor'"> <router-link v-for="(item, index) in sortedMonitorList" :key="index" :to="monitorURL(item.id)" class="item" :class="{ 'disabled': ! item.active }" > <div class="row"> <div class="col-9 col-md-8 small-padding" :class="{ 'monitor-item': $root.userHeartbeatBar == 'bottom' || $root.userHeartbeatBar == 'none' }" > <div class="info"> <Uptime :monitor="item" type="24" :pill="true" /> {{ item.name }} </div> <div class="tags"> <Tag v-for="tag in item.tags" :key="tag" :item="tag" :size="'sm'" /> </div> </div> <div v-show="$root.userHeartbeatBar == 'normal'" :key="$root.userHeartbeatBar" class="col-3 col-md-4" > <HeartbeatBar size="small" :monitor-id="item.id" /> </div> </div> <div v-if="$root.userHeartbeatBar == 'bottom'" class="row"> <div class="col-12 bottom-style"> <HeartbeatBar size="small" :monitor-id="item.id" /> </div> </div> </router-link> </template> </div> </div> </template> <script> import HeartbeatBar from "../components/HeartbeatBar.vue"; import Tag from "../components/Tag.vue"; import Uptime from "../components/Uptime.vue"; import { getMaintenanceRelativeURL, getMonitorRelativeURL } from "../util.ts"; export default { components: { Uptime, HeartbeatBar, Tag, }, props: { /** Should the scrollbar be shown */ scrollbar: { type: Boolean, }, }, data() { return { searchText: "", selectedList: "monitor", windowTop: 0, }; }, computed: { /** * Improve the sticky appearance of the list by increasing its * height as user scrolls down. * Not used on mobile. */ boxStyle() { if (window.innerWidth > 550) { return { height: `calc(100vh - 160px + ${this.windowTop}px)`, }; } else { return { height: "calc(100vh - 160px)", }; } }, sortedMaintenanceList() { let result = Object.values(this.$root.maintenanceList); result.sort((m1, m2) => { if (this.$root.isActiveMaintenance(m1.end_date) !== this.$root.isActiveMaintenance(m2.end_date)) { if (!this.$root.isActiveMaintenance(m2.end_date)) { return -1; } if (!this.$root.isActiveMaintenance(m1.end_date)) { return 1; } } if (this.$root.isActiveMaintenance(m1.end_date) && this.$root.isActiveMaintenance(m2.end_date)) { if (Date.parse(m1.end_date) < Date.parse(m2.end_date)) { return -1; } if (Date.parse(m2.end_date) < Date.parse(m1.end_date)) { return 1; } } if (!this.$root.isActiveMaintenance(m1.end_date) && !this.$root.isActiveMaintenance(m2.end_date)) { if (Date.parse(m1.end_date) < Date.parse(m2.end_date)) { return 1; } if (Date.parse(m2.end_date) < Date.parse(m1.end_date)) { return -1; } } return m1.title.localeCompare(m2.title); }); // Simple filter by search text // finds maintenance name if (this.searchText !== "") { const loweredSearchText = this.searchText.toLowerCase(); result = result.filter(maintenance => { return maintenance.title.toLowerCase().includes(loweredSearchText) || maintenance.description.toLowerCase().includes(loweredSearchText); }); } return result; }, sortedMonitorList() { let result = Object.values(this.$root.monitorList); result.sort((m1, m2) => { if (m1.active !== m2.active) { if (m1.active === 0) { return 1; } if (m2.active === 0) { return -1; } } if (m1.weight !== m2.weight) { if (m1.weight > m2.weight) { return -1; } if (m1.weight < m2.weight) { return 1; } } return m1.name.localeCompare(m2.name); }); // Simple filter by search text // finds monitor name, tag name or tag value if (this.searchText !== "") { const loweredSearchText = this.searchText.toLowerCase(); result = result.filter(monitor => { return monitor.name.toLowerCase().includes(loweredSearchText) || monitor.tags.find(tag => tag.name.toLowerCase().includes(loweredSearchText) || tag.value?.toLowerCase().includes(loweredSearchText)); }); } return result; }, }, mounted() { window.addEventListener("scroll", this.onScroll); }, beforeUnmount() { window.removeEventListener("scroll", this.onScroll); }, methods: { /** Handle user scroll */ onScroll() { if (window.top.scrollY <= 133) { this.windowTop = window.top.scrollY; } else { this.windowTop = 133; } }, /** * Get URL of monitor * @param {number} id ID of monitor * @returns {string} Relative URL of monitor */ monitorURL(id) { return getMonitorRelativeURL(id); }, maintenanceURL(id) { return getMaintenanceRelativeURL(id); }, /** Clear the search bar */ clearSearchText() { this.searchText = ""; } }, }; </script> <style lang="scss" scoped> @import "../assets/vars.scss"; .shadow-box { height: calc(100vh - 150px); position: sticky; top: 10px; } .small-padding { padding-left: 5px !important; padding-right: 5px !important; } .list-header { border-bottom: 1px solid #dee2e6; border-radius: 10px 10px 0 0; margin: -10px; margin-bottom: 10px; padding: 10px; display: flex; justify-content: space-between; .dark & { background-color: $dark-header-bg; border-bottom: 0; } } @media (max-width: 770px) { .list-header { margin: -20px; margin-bottom: 10px; padding: 5px; } } .search-wrapper { display: flex; align-items: center; } .search-icon { padding: 10px; color: #c0c0c0; } .search-input { max-width: 15em; } .monitor-item { width: 100%; } .tags { margin-top: 4px; padding-left: 67px; display: flex; flex-wrap: wrap; gap: 0; } .bottom-style { padding-left: 67px; margin-top: 5px; } .bg-maintenance { background-color: $maintenance; } select { text-align: center; } </style>