|
| 1 | +<template> |
| 2 | + <v-card> |
| 3 | + <div class="d-flex align-center justify-space-between mb-8"> |
| 4 | + <h3 class="d-flex align-center ga-2"> |
| 5 | + <v-icon :icon="icon" size="x-small" color="primary" /> |
| 6 | + {{ nodeType }} Nodes |
| 7 | + </h3> |
| 8 | + |
| 9 | + <v-btn |
| 10 | + variant="text" |
| 11 | + :text="`Add ${nodeType}`" |
| 12 | + color="primary" |
| 13 | + prepend-icon="mdi-plus" |
| 14 | + border |
| 15 | + @click="$emit('addNode')" |
| 16 | + /> |
| 17 | + </div> |
| 18 | + |
| 19 | + <div v-if="nodes.length === 0" cols="12" class="border border-dashed rounded-lg d-flex justify-center align-center"> |
| 20 | + <div class="text-center my-12" :style="{ color: 'rgba(var(--v-border-color), 0.5)' }"> |
| 21 | + <v-icon icon="mdi-sitemap-outline" size="50" class="mb-4" /> |
| 22 | + <p class="text-body-1"> |
| 23 | + No {{ nodeType }} nodes configured |
| 24 | + </p> |
| 25 | + </div> |
| 26 | + </div> |
| 27 | + |
| 28 | + <div v-else> |
| 29 | + <v-tabs |
| 30 | + v-model="tab" |
| 31 | + align-tabs="center" |
| 32 | + color="primary" |
| 33 | + class="mb-8" |
| 34 | + > |
| 35 | + <v-tab v-for="node in nodes" :key="node.id" :value="node.id" :class="{ 'text-error': !isValidClusterNode(node) }"> |
| 36 | + {{ node.name || '*Unnamed' }} |
| 37 | + |
| 38 | + <template v-if="!node.permanent" #append> |
| 39 | + <v-btn |
| 40 | + icon |
| 41 | + size="x-small" |
| 42 | + variant="plain" |
| 43 | + color="error" |
| 44 | + :style="{ borderRadius: '50% !important' }" |
| 45 | + @click.stop="onRemoveNode(node.id)" |
| 46 | + > |
| 47 | + <v-icon icon="mdi-close" size="small" /> |
| 48 | + </v-btn> |
| 49 | + </template> |
| 50 | + </v-tab> |
| 51 | + </v-tabs> |
| 52 | + |
| 53 | + <DefineVMNodeForm v-if="activeNode" :ssh-keys="sshKeys" :node="activeNode" /> |
| 54 | + </div> |
| 55 | + </v-card> |
| 56 | +</template> |
| 57 | + |
| 58 | +<script setup lang="ts"> |
| 59 | +import type { ModelsSSHKey } from "~/generated/api" |
| 60 | +
|
| 61 | +const props = defineProps<{ |
| 62 | + icon: string |
| 63 | + nodeType: string |
| 64 | + sshKeys: ModelsSSHKey[] |
| 65 | + nodes: ClusterNode[] |
| 66 | +}>() |
| 67 | +
|
| 68 | +const emit = defineEmits<{ |
| 69 | + (e: "addNode"): void |
| 70 | + (e: "removeNode", id: string): void |
| 71 | +}>() |
| 72 | +
|
| 73 | +const tab = ref<string>() |
| 74 | +const activeNode = computed(() => props.nodes.find(node => node.id === tab.value)) |
| 75 | +
|
| 76 | +function onRemoveNode(id: string) { |
| 77 | + if (tab.value === id) { |
| 78 | + const nodes = props.nodes.filter(node => node.id !== id) |
| 79 | + tab.value = nodes[0]?.id |
| 80 | + } |
| 81 | +
|
| 82 | + emit("removeNode", id) |
| 83 | +} |
| 84 | +</script> |
0 commit comments