-
Notifications
You must be signed in to change notification settings - Fork 270
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(xo-lite): add host network tab and pif table / status component …
…and pif store
- Loading branch information
Showing
9 changed files
with
328 additions
and
4 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
43 changes: 43 additions & 0 deletions
43
@xen-orchestra/lite/src/components/host/network/HostPifStatus.vue
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,43 @@ | ||
<template> | ||
<UiInfo class="pif-status text-ellipsis" :accent="getStatusProps(status).accent"> | ||
<p class="text-ellipsis" :class="{ card }">{{ getStatusProps(status).text }}</p> | ||
</UiInfo> | ||
</template> | ||
|
||
<script setup lang="ts"> | ||
import UiInfo from '@core/components/ui/info/UiInfo.vue' | ||
import type { IconDefinition } from '@fortawesome/fontawesome-common-types' | ||
import { faCheck, faExclamation } from '@fortawesome/free-solid-svg-icons' | ||
import { computed } from 'vue' | ||
import { useI18n } from 'vue-i18n' | ||
const props = defineProps<{ | ||
pif: any | ||
card?: boolean | ||
}>() | ||
const { t } = useI18n() | ||
type NetworkStatus = 'connected' | 'disconnected' | 'partial' | ||
type NetworkAccent = 'success' | 'warning' | 'danger' | ||
const states = computed<Record<NetworkStatus, { text: string; icon: IconDefinition; accent: NetworkAccent }>>(() => ({ | ||
connected: { text: t('connected'), icon: faCheck, accent: 'success' }, | ||
disconnected: { text: t('disconnected'), icon: faCheck, accent: 'danger' }, | ||
partial: { text: t('disconnected-from-physical-device'), icon: faExclamation, accent: 'warning' }, | ||
})) | ||
const status = computed(() => { | ||
if (props.pif.currently_attached) { | ||
return 'connected' | ||
} | ||
if (props.pif.currently_attached) { | ||
return 'partial' | ||
} | ||
return 'disconnected' | ||
}) | ||
const getStatusProps = (status: NetworkStatus) => states.value[status as NetworkStatus] | ||
</script> | ||
|
||
<style scoped lang="postcss"> | ||
p:not(.card) { | ||
font-size: 1.4rem !important; | ||
} | ||
</style> |
219 changes: 219 additions & 0 deletions
219
@xen-orchestra/lite/src/components/host/network/HostPifTable.vue
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,219 @@ | ||
<template> | ||
<div class="host-pif-table"> | ||
<UiTitle> | ||
{{ $t('pifs') }} | ||
<template #actions> | ||
<UiButton :left-icon="faPlus" variant="secondary" accent="info" size="medium">{{ $t('scan-pifs') }}</UiButton> | ||
</template> | ||
</UiTitle> | ||
<div class="container"> | ||
<UiQuerySearchBar class="table-query" @search="(value: string) => (searchQuery = value)" /> | ||
<UiTableActions title="Table actions"> | ||
<UiButton :left-icon="faEdit" variant="tertiary" accent="info" size="medium">{{ $t('edit') }}</UiButton> | ||
<UiButton :left-icon="faTrash" variant="tertiary" accent="danger" size="medium">{{ $t('delete') }}</UiButton> | ||
</UiTableActions> | ||
<UiTopBottomTable | ||
:selected-items="selected.length" | ||
:total-items="pifsUuid.length" | ||
@toggle-select-all="toggleSelect" | ||
/> | ||
<div class="table"> | ||
<VtsTable vertical-border> | ||
<thead> | ||
<tr> | ||
<template v-for="column of visibleColumns" :key="column.id"> | ||
<th v-if="column.id === 'checkbox'" class="checkbox"> | ||
<UiCheckbox :v-model="areAllSelected" accent="info" @update:model-value="toggleSelect" /> | ||
</th> | ||
<th v-else-if="column.id === 'more'" class="more"> | ||
<UiButtonIcon size="small" accent="info" :icon="getHeaderIcon(column.id)" /> | ||
{{ column.label }} | ||
</th> | ||
<ColumnTitle v-else id="networks" :icon="getHeaderIcon(column.id)"> {{ column.label }}</ColumnTitle> | ||
</template> | ||
</tr> | ||
</thead> | ||
<tbody> | ||
<tr v-for="row of rows" :key="row.id"> | ||
<td v-for="column of row.visibleColumns" :key="column.id" class="typo p2-regular"> | ||
<UiCheckbox v-if="column.id === 'checkbox'" v-model="selected" accent="info" :value="row.id" /> | ||
<div v-if="column.id === 'network'" class="network"> | ||
<UiComplexIcon size="medium" class="icon"> | ||
<VtsIcon :icon="faNetworkWired" accent="current" /> | ||
<VtsIcon accent="success" :icon="faCircle" :overlay-icon="faCheck" /> | ||
</UiComplexIcon> | ||
<a v-tooltip class="text-ellipsis name">{{ getNetworkName(column.value) }}</a> | ||
</div> | ||
<div v-if="column.id === 'device'"> | ||
{{ column.value }} | ||
</div> | ||
<div v-if="column.id === 'status'" v-tooltip> | ||
<HostPifStatus class="status" :pif="row.value" /> | ||
</div> | ||
<div v-if="column.id === 'VLAN'"> | ||
{{ column.value === -1 ? $t('none') : column.value }} | ||
</div> | ||
<div v-if="column.id === 'IP'" v-tooltip="{ placement: 'bottom-end' }" class="text-ellipsis"> | ||
{{ column.value }} | ||
</div> | ||
<div v-if="column.id === 'MAC'" v-tooltip="{ placement: 'bottom-end' }" class="text-ellipsis"> | ||
{{ column.value }} | ||
</div> | ||
<div v-if="column.id === 'ip_configuration_mode'"> | ||
{{ column.value }} | ||
</div> | ||
<div v-if="column.id === 'more'"> | ||
<VtsIcon accent="info" :icon="faEllipsis" /> | ||
</div> | ||
</td> | ||
</tr> | ||
</tbody> | ||
</VtsTable> | ||
</div> | ||
<UiTopBottomTable | ||
:selected-items="selected.length" | ||
:total-items="pifsUuid.length" | ||
@toggle-select-all="toggleSelect" | ||
/> | ||
</div> | ||
</div> | ||
<UiCardSpinner v-if="!isReady" /> | ||
</template> | ||
|
||
<script lang="ts" setup> | ||
import HostPifStatus from '@/components/host/network/HostPifStatus.vue' | ||
import UiCardSpinner from '@/components/ui/UiCardSpinner.vue' | ||
import useMultiSelect from '@/composables/multi-select.composable' | ||
import type { XenApiNetwork, XenApiPif } from '@/libs/xen-api/xen-api.types' | ||
import { useNetworkStore } from '@/stores/xen-api/network.store' | ||
import VtsIcon from '@core/components/icon/VtsIcon.vue' | ||
import ColumnTitle from '@core/components/table/ColumnTitle.vue' | ||
import VtsTable from '@core/components/table/VtsTable.vue' | ||
import UiButton from '@core/components/ui/button/UiButton.vue' | ||
import UiButtonIcon from '@core/components/ui/button-icon/UiButtonIcon.vue' | ||
import UiCheckbox from '@core/components/ui/checkbox/UiCheckbox.vue' | ||
import UiComplexIcon from '@core/components/ui/complex-icon/UiComplexIcon.vue' | ||
import UiQuerySearchBar from '@core/components/ui/query-search-bar/UiQuerySearchBar.vue' | ||
import UiTableActions from '@core/components/ui/table-actions/UiTableActions.vue' | ||
import UiTitle from '@core/components/ui/title/UiTitle.vue' | ||
import UiTopBottomTable from '@core/components/ui/top-bottom-table/UiTopBottomTable.vue' | ||
import { useTable } from '@core/composables/table.composable' | ||
import { vTooltip } from '@core/directives/tooltip.directive' | ||
import type { IconDefinition } from '@fortawesome/fontawesome-common-types' | ||
import { | ||
faAlignLeft, | ||
faAt, | ||
faCaretDown, | ||
faCheck, | ||
faCircle, | ||
faEdit, | ||
faEllipsis, | ||
faNetworkWired, | ||
faPlus, | ||
faPowerOff, | ||
faTrash, | ||
} from '@fortawesome/free-solid-svg-icons' | ||
import { computed, ref } from 'vue' | ||
const { pifs } = defineProps<{ | ||
pifs: XenApiPif[] | ||
isReady: boolean | ||
}>() | ||
const { getByOpaqueRef } = useNetworkStore().subscribe() | ||
const getNetworkName = (pif: string) => { | ||
const network: XenApiNetwork = getByOpaqueRef(pif as XenApiNetwork['$ref'])! | ||
return network?.name_label ? network.name_label : '' | ||
} | ||
const searchQuery = ref('') | ||
const filteredNetworks = computed(() => { | ||
return searchQuery.value | ||
? pifs.filter(pif => | ||
Object.values(pif).some(value => String(value).toLowerCase().includes(searchQuery.value.toLowerCase())) | ||
) | ||
: pifs | ||
}) | ||
const pifsUuid = computed(() => pifs.map(pif => pif.uuid)) | ||
const { selected, areAllSelected } = useMultiSelect(pifsUuid) | ||
const toggleSelect = () => { | ||
selected.value = selected.value.length === 0 ? pifsUuid.value : [] | ||
} | ||
const { visibleColumns, rows } = useTable('pifs', filteredNetworks, { | ||
rowId: record => record.uuid, | ||
columns: define => [ | ||
define('checkbox', () => '', { label: '', isHideable: false }), | ||
define('network', { label: 'Network', isHideable: true }), | ||
define('device', { label: 'Device', isHideable: true }), | ||
define('status', () => '', { label: 'Status', isHideable: true }), | ||
define('VLAN', { label: 'Vlan', isHideable: true }), | ||
define('IP', { label: 'IP', isHideable: true }), | ||
define('MAC', { label: 'MAC', isHideable: true }), | ||
define('ip_configuration_mode', { label: 'Mode', isHideable: true }), | ||
define('more', () => '', { label: '', isHideable: false }), | ||
], | ||
}) | ||
type pifHeader = 'network' | 'device' | 'status' | 'VLAN' | 'IP' | 'MAC' | 'ip_configuration_mode' | 'more' | ||
const headerIcon: Record<pifHeader, { icon: IconDefinition }> = { | ||
network: { icon: faAlignLeft }, | ||
device: { icon: faAlignLeft }, | ||
status: { icon: faPowerOff }, | ||
VLAN: { icon: faAlignLeft }, | ||
IP: { icon: faAt }, | ||
MAC: { icon: faAt }, | ||
ip_configuration_mode: { icon: faCaretDown }, | ||
more: { icon: faEllipsis }, | ||
} | ||
const getHeaderIcon = (status: pifHeader) => headerIcon[status].icon | ||
</script> | ||
|
||
<style scoped lang="postcss"> | ||
.host-pif-table, | ||
.container { | ||
display: flex; | ||
flex-direction: column; | ||
} | ||
.host-pif-table { | ||
gap: 2.4rem; | ||
.container { | ||
gap: 0.8rem; | ||
.table { | ||
overflow-x: auto; | ||
.network { | ||
display: flex; | ||
align-items: center; | ||
gap: 1.8rem; | ||
.name { | ||
cursor: pointer; | ||
text-decoration: underline solid #6b63bf; | ||
} | ||
} | ||
tr:last-child { | ||
border-bottom: 1px solid var(--color-neutral-border); | ||
} | ||
} | ||
.checkbox, | ||
.more { | ||
width: 4.8rem; | ||
} | ||
} | ||
@media (max-width: 1440px) { | ||
.table table { | ||
width: 160rem; | ||
} | ||
} | ||
} | ||
</style> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.