mirror of https://github.com/rancher/dashboard.git
317 lines
6.1 KiB
Vue
317 lines
6.1 KiB
Vue
<script>
|
|
import { debounce } from 'lodash';
|
|
import { _EDIT, _VIEW } from '@/config/query-params';
|
|
import { removeAt } from '@/utils/array';
|
|
import TextAreaAutoGrow from '@/components/form/TextAreaAutoGrow';
|
|
|
|
/*
|
|
@TODO
|
|
- Paste
|
|
- Read from file
|
|
- Multiline
|
|
- Concealed value
|
|
*/
|
|
|
|
export default {
|
|
components: { TextAreaAutoGrow },
|
|
|
|
props: {
|
|
value: {
|
|
type: Array,
|
|
default: null,
|
|
},
|
|
mode: {
|
|
type: String,
|
|
default: _EDIT,
|
|
},
|
|
initialEmptyRow: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
|
|
title: {
|
|
type: String,
|
|
default: ''
|
|
},
|
|
protip: {
|
|
type: [String, Boolean],
|
|
default: 'ProTip: Paste lines of <code>key=value</code> or <code>key: value</code> into any key field for easy bulk entry',
|
|
},
|
|
|
|
showHeader: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
|
|
padLeft: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
|
|
valueLabel: {
|
|
type: String,
|
|
default: 'Value',
|
|
},
|
|
valuePlaceholder: {
|
|
type: String,
|
|
default: 'e.g. bar'
|
|
},
|
|
valueMultiline: {
|
|
type: Boolean,
|
|
default: true,
|
|
},
|
|
valueConcealed: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
|
|
addLabel: {
|
|
type: String,
|
|
default: 'Add',
|
|
},
|
|
addIcon: {
|
|
type: String,
|
|
default: 'icon-plus',
|
|
},
|
|
addAllowed: {
|
|
type: Boolean,
|
|
default: true,
|
|
},
|
|
|
|
removeLabel: {
|
|
type: String,
|
|
default: '',
|
|
},
|
|
removeIcon: {
|
|
type: String,
|
|
default: 'icon-minus',
|
|
},
|
|
removeAllowed: {
|
|
type: Boolean,
|
|
default: true,
|
|
},
|
|
defaultAddValue: {
|
|
type: [String, Number, Object, Array],
|
|
default: ''
|
|
}
|
|
},
|
|
|
|
data() {
|
|
const input = (this.value || []).slice();
|
|
const rows = [];
|
|
|
|
for ( const value of input ) {
|
|
rows.push({ value });
|
|
}
|
|
|
|
if ( !rows.length && this.initialEmptyRow ) {
|
|
rows.push({ value: '' });
|
|
}
|
|
|
|
return { rows };
|
|
},
|
|
|
|
computed: {
|
|
isView() {
|
|
return this.mode === _VIEW;
|
|
},
|
|
|
|
showAdd() {
|
|
return !this.isView && this.addAllowed;
|
|
},
|
|
|
|
showRead() {
|
|
return !this.isView && this.readAllowed;
|
|
},
|
|
|
|
showRemove() {
|
|
return !this.isView && this.removeAllowed;
|
|
},
|
|
},
|
|
|
|
watch: {
|
|
value() {
|
|
this.rows = (this.value || []).map(v => ({ value: v }));
|
|
}
|
|
},
|
|
|
|
created() {
|
|
this.queueUpdate = debounce(this.update, 100);
|
|
},
|
|
|
|
methods: {
|
|
add() {
|
|
this.rows.push({ value: this.defaultAddValue });
|
|
this.queueUpdate();
|
|
|
|
this.$nextTick(() => {
|
|
const inputs = this.$refs.value;
|
|
|
|
if ( inputs && inputs.length > 0 ) {
|
|
inputs[inputs.length - 1].focus();
|
|
}
|
|
});
|
|
},
|
|
|
|
remove(idx) {
|
|
removeAt(this.rows, idx);
|
|
this.queueUpdate();
|
|
},
|
|
|
|
update() {
|
|
if ( this.isView ) {
|
|
return;
|
|
}
|
|
|
|
const out = [];
|
|
|
|
for ( const row of this.rows ) {
|
|
const value = (typeof row.value === 'string') ? row.value.trim() : row.value;
|
|
|
|
if ( typeof value !== 'undefined' ) {
|
|
out.push(value);
|
|
}
|
|
}
|
|
|
|
this.$emit('input', out);
|
|
}
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div>
|
|
<div v-if="title" class="title clearfix">
|
|
<h4>{{ title }} <i v-if="protip" v-tooltip="protip" class="icon icon-info" style="font-size: 12px" /></h4>
|
|
</div>
|
|
|
|
<table v-if="rows.length" class="fixed">
|
|
<thead v-if="showHeader">
|
|
<tr>
|
|
<th v-if="padLeft" class="left"></th>
|
|
<slot name="thead-columns">
|
|
<th class="value">
|
|
{{ valueLabel }}
|
|
</th>
|
|
</slot>
|
|
<th v-if="showRemove" class="remove"></th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr
|
|
v-for="(row, idx) in rows"
|
|
:key="idx"
|
|
class="box"
|
|
>
|
|
<td v-if="padLeft" class="left"></td>
|
|
<slot
|
|
name="columns"
|
|
:queueUpdate="queueUpdate"
|
|
:i="idx"
|
|
:rows="rows"
|
|
:row="row"
|
|
:mode="mode"
|
|
:isView="isView"
|
|
>
|
|
<td class="value">
|
|
<slot
|
|
name="value"
|
|
:row="row"
|
|
:mode="mode"
|
|
:isView="isView"
|
|
>
|
|
<span v-if="isView">{{ row.value }}</span>
|
|
<TextAreaAutoGrow
|
|
v-else-if="valueMultiline"
|
|
ref="value"
|
|
v-model="row.value"
|
|
:placeholder="valuePlaceholder"
|
|
@input="queueUpdate"
|
|
/>
|
|
<input
|
|
v-else
|
|
ref="value"
|
|
v-model="row.value"
|
|
:placeholder="valuePlaceholder"
|
|
@input="queueUpdate"
|
|
/>
|
|
</slot>
|
|
</td>
|
|
</slot>
|
|
<td v-if="showRemove" class="remove">
|
|
<button type="button" class="btn role-link" @click="remove(idx)">
|
|
Remove
|
|
{{ removeLabel }}
|
|
</button>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
<div v-else-if="mode==='view'">
|
|
n/a
|
|
</div>
|
|
<div v-if="showAdd || showRead" class="footer">
|
|
<slot v-if="showAdd" name="add">
|
|
<button type="button" class="btn role-tertiary add" @click="add()">
|
|
{{ addLabel }}
|
|
</button>
|
|
<slot name="moreAdd" :rows="rows" />
|
|
</slot>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style lang="scss" scoped>
|
|
$remove: 75;
|
|
|
|
.title {
|
|
margin-bottom: 10px;
|
|
|
|
.read-from-file {
|
|
float: right;
|
|
}
|
|
}
|
|
|
|
TABLE {
|
|
width: 100%;
|
|
}
|
|
|
|
TH {
|
|
text-align: left;
|
|
font-size: 12px;
|
|
font-weight: normal;
|
|
color: var(--input-label);
|
|
}
|
|
|
|
.box {
|
|
border-radius: var(--border-radius);
|
|
}
|
|
|
|
.api-groups, .kinds {
|
|
padding: 40px 20px 20px 20px;
|
|
}
|
|
|
|
.left {
|
|
width: #{$remove}px;
|
|
}
|
|
|
|
.value {
|
|
vertical-align: top;
|
|
}
|
|
|
|
.remove {
|
|
vertical-align: middle;
|
|
text-align: right;
|
|
width: #{$remove}px;
|
|
}
|
|
|
|
.footer {
|
|
margin-top: 10px;
|
|
|
|
.protip {
|
|
float: right;
|
|
padding: 5px 0;
|
|
}
|
|
}
|
|
</style>
|