Andy Lee 9ca9fdb521
feat: support for subnets and VPCs from UI (#374)
Signed-off-by: Andy Lee <andy.lee@suse.com>
2025-07-16 17:51:37 +08:00

188 lines
4.1 KiB
Vue

<script>
import debounce from 'lodash/debounce';
import { _EDIT, _VIEW } from '@shell/config/query-params';
import { removeAt } from '@shell/utils/array';
import { HCI } from '../../types';
import LabeledSelect from '@shell/components/form/LabeledSelect';
export default {
name: 'VpcPeerings',
emits: ['update:value'],
components: { LabeledSelect },
props: {
value: {
type: Array,
default: null,
},
mode: {
type: String,
default: _EDIT,
},
vpc: {
type: Object,
default: () => ({}),
},
},
async fetch() {
await this.$store.dispatch('harvester/findAll', { type: HCI.VPC });
},
data() {
const rows = (this.value || []).map((row) => {
return {
localConnectIP: row.localConnectIP || '',
remoteVpc: row.remoteVpc || '',
};
});
return { rows };
},
computed: {
isView() {
return this.mode === _VIEW;
},
isEdit() {
return this.mode === _EDIT;
},
showAdd() {
return !this.isView;
},
showRemove() {
return !this.isView;
},
remoteVpcOptions() {
const allVpcs = this.$store.getters['harvester/all'](HCI.VPC) || [];
// filter self vpc if editing
const vpcs = this.isEdit ? allVpcs.filter((v) => v.id !== this.vpc.id) : allVpcs;
return vpcs.map((n) => ({
label: n.id,
value: n.id,
}));
},
localConnectIPTooltip() {
return this.t('harvester.vpc.vpcPeerings.localConnectIP.tooltip');
},
},
created() {
this.queueUpdate = debounce(this.update, 100);
},
methods: {
add() {
this.rows.push({
localConnectIP: '',
remoteVpc: '',
});
this.queueUpdate();
},
remove(idx) {
removeAt(this.rows, idx);
this.queueUpdate();
},
update() {
if (this.isView) {
return;
}
this.$emit('update:value', this.rows);
}
},
};
</script>
<template>
<div>
<div
v-if="rows.length"
class="static-route-row"
>
<div
v-for="(row, idx) in rows"
:key="idx"
>
<div class="pool-headers localConnectIP">
<span class="pool-localConnectIP">
<t k="harvester.vpc.vpcPeerings.localConnectIP.label" />
<i
v-clean-tooltip="{content: localConnectIPTooltip, triggers: ['hover', 'touch', 'focus'] }"
v-stripped-aria-label="localConnectPTooltip"
class="icon icon-info"
tabindex="0"
/>
</span>
<span class="pool-remoteVpc">
<t k="harvester.vpc.vpcPeerings.remoteVpc.label" />
</span>
</div>
<div class="pool-row localConnectIP">
<div class="pool-localConnectIP">
<span v-if="isView">
{{ row.localConnectIP }}
</span>
<input
v-else
v-model="row.localConnectIP"
type="text"
:placeholder="t('harvester.vpc.vpcPeerings.localConnectIP.placeholder')"
@input="queueUpdate"
/>
</div>
<div class="pool-remoteVpc">
<span v-if="isView">
{{ row.remoteVpc }}
</span>
<LabeledSelect
v-model:value="row.remoteVpc"
:options="remoteVpcOptions"
:mode="mode"
@update:value="queueUpdate"
/>
</div>
<button
v-if="showRemove"
type="button"
class="btn role-link pl-0"
@click="remove(idx)"
>
<t k="generic.remove" />
</button>
</div>
</div>
</div>
<button
v-if="showAdd"
type="button"
class="btn role-tertiary add"
@click="add()"
>
<t k="generic.add" />
</button>
</div>
</template>
<style lang="scss" scoped>
.pool-headers, .pool-row {
display: grid;
grid-column-gap: $column-gutter;
margin-bottom: 10px;
align-items: center;
&.localConnectIP {
grid-template-columns: 40%+$column-gutter 40%+$column-gutter 15%;
}
}
</style>