2020-12-01 16:48:20 +01:00
|
|
|
<template>
|
|
|
|
<ConfirmModal
|
|
|
|
:confirm-text="$t('word.confirm')"
|
|
|
|
size="medium"
|
2021-04-03 11:21:58 +02:00
|
|
|
class="options-modal"
|
2020-12-01 16:48:20 +01:00
|
|
|
@confirm="confirmIndexesChange"
|
|
|
|
@hide="$emit('hide')"
|
|
|
|
>
|
2021-12-10 17:34:44 +01:00
|
|
|
<template #header>
|
2020-12-01 16:48:20 +01:00
|
|
|
<div class="d-flex">
|
2021-06-13 11:16:21 +02:00
|
|
|
<i class="mdi mdi-24px mdi-key mdi-rotate-45 mr-1" />
|
|
|
|
<span class="cut-text">{{ $t('word.indexes') }} "{{ table }}"</span>
|
2020-12-01 16:48:20 +01:00
|
|
|
</div>
|
|
|
|
</template>
|
2021-12-10 17:34:44 +01:00
|
|
|
<template #body>
|
2020-12-01 16:48:20 +01:00
|
|
|
<div class="columns col-gapless">
|
|
|
|
<div class="column col-5">
|
|
|
|
<div class="panel" :style="{ height: modalInnerHeight + 'px'}">
|
|
|
|
<div class="panel-header pt-0 pl-0">
|
|
|
|
<div class="d-flex">
|
|
|
|
<button class="btn btn-dark btn-sm d-flex" @click="addIndex">
|
2021-07-09 15:51:02 +02:00
|
|
|
<i class="mdi mdi-24px mdi-key-plus mr-1" />
|
2020-12-01 16:48:20 +01:00
|
|
|
<span>{{ $t('word.add') }}</span>
|
|
|
|
</button>
|
|
|
|
<button
|
|
|
|
class="btn btn-dark btn-sm d-flex ml-2 mr-0"
|
|
|
|
:title="$t('message.clearChanges')"
|
|
|
|
:disabled="!isChanged"
|
|
|
|
@click.prevent="clearChanges"
|
|
|
|
>
|
2021-07-09 15:51:02 +02:00
|
|
|
<i class="mdi mdi-24px mdi-delete-sweep mr-1" />
|
2020-12-01 16:48:20 +01:00
|
|
|
<span>{{ $t('word.clear') }}</span>
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div ref="indexesPanel" class="panel-body p-0 pr-1">
|
|
|
|
<div
|
|
|
|
v-for="index in indexesProxy"
|
2021-11-08 15:12:37 +01:00
|
|
|
:key="index._antares_id"
|
2020-12-01 16:48:20 +01:00
|
|
|
class="tile tile-centered c-hand mb-1 p-1"
|
2021-11-08 15:12:37 +01:00
|
|
|
:class="{'selected-element': selectedIndexID === index._antares_id}"
|
|
|
|
@click="selectIndex($event, index._antares_id)"
|
2020-12-01 16:48:20 +01:00
|
|
|
>
|
|
|
|
<div class="tile-icon">
|
|
|
|
<div>
|
|
|
|
<i class="mdi mdi-key mdi-24px column-key" :class="`key-${index.type}`" />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<div class="tile-content">
|
|
|
|
<div class="tile-title">
|
|
|
|
{{ index.name }}
|
|
|
|
</div>
|
|
|
|
<small class="tile-subtitle text-gray">{{ index.type }} · {{ index.fields.length }} {{ $tc('word.field', index.fields.length) }}</small>
|
|
|
|
</div>
|
|
|
|
<div class="tile-action">
|
|
|
|
<button
|
|
|
|
class="btn btn-link remove-field p-0 mr-2"
|
|
|
|
:title="$t('word.delete')"
|
2021-11-08 15:12:37 +01:00
|
|
|
@click.prevent="removeIndex(index._antares_id)"
|
2020-12-01 16:48:20 +01:00
|
|
|
>
|
|
|
|
<i class="mdi mdi-close" />
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="column col-7 pl-2 editor-col">
|
2020-12-15 17:08:36 +01:00
|
|
|
<form
|
|
|
|
v-if="selectedIndexObj"
|
|
|
|
:style="{ height: modalInnerHeight + 'px'}"
|
|
|
|
class="form-horizontal"
|
|
|
|
>
|
2020-12-01 16:48:20 +01:00
|
|
|
<div class="form-group">
|
2020-12-15 17:08:36 +01:00
|
|
|
<label class="form-label col-3">
|
2020-12-01 16:48:20 +01:00
|
|
|
{{ $t('word.name') }}
|
|
|
|
</label>
|
2020-12-15 17:08:36 +01:00
|
|
|
<div class="column">
|
|
|
|
<input
|
|
|
|
v-model="selectedIndexObj.name"
|
|
|
|
class="form-input"
|
|
|
|
type="text"
|
|
|
|
>
|
|
|
|
</div>
|
2020-12-01 16:48:20 +01:00
|
|
|
</div>
|
|
|
|
<div class="form-group">
|
2020-12-15 17:08:36 +01:00
|
|
|
<label class="form-label col-3">
|
2020-12-01 16:48:20 +01:00
|
|
|
{{ $t('word.type') }}
|
|
|
|
</label>
|
2020-12-15 17:08:36 +01:00
|
|
|
<div class="column">
|
|
|
|
<select v-model="selectedIndexObj.type" class="form-select">
|
|
|
|
<option
|
|
|
|
v-for="index in indexTypes"
|
|
|
|
:key="index"
|
|
|
|
:value="index"
|
|
|
|
:disabled="index === 'PRIMARY' && hasPrimary"
|
|
|
|
>
|
|
|
|
{{ index }}
|
|
|
|
</option>
|
|
|
|
</select>
|
|
|
|
</div>
|
2020-12-01 16:48:20 +01:00
|
|
|
</div>
|
|
|
|
<div class="form-group">
|
2020-12-15 17:08:36 +01:00
|
|
|
<label class="form-label col-3">
|
2020-12-01 16:48:20 +01:00
|
|
|
{{ $tc('word.field', fields.length) }}
|
|
|
|
</label>
|
2020-12-15 17:08:36 +01:00
|
|
|
<div class="fields-list column pt-1">
|
2020-12-01 16:48:20 +01:00
|
|
|
<label
|
|
|
|
v-for="(field, i) in fields"
|
|
|
|
:key="`${field.name}-${i}`"
|
|
|
|
class="form-checkbox m-0"
|
|
|
|
@click.prevent="toggleField(field.name)"
|
|
|
|
>
|
|
|
|
<input type="checkbox" :checked="selectedIndexObj.fields.some(f => f === field.name)">
|
|
|
|
<i class="form-icon" /> {{ field.name }}
|
|
|
|
</label>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</form>
|
2020-12-11 18:09:17 +01:00
|
|
|
<div v-if="!indexesProxy.length" class="empty">
|
|
|
|
<div class="empty-icon">
|
|
|
|
<i class="mdi mdi-key-outline mdi-48px" />
|
|
|
|
</div>
|
|
|
|
<p class="empty-title h5">
|
|
|
|
{{ $t('message.thereAreNoIndexes') }}
|
|
|
|
</p>
|
|
|
|
<div class="empty-action">
|
|
|
|
<button class="btn btn-primary" @click="addIndex">
|
|
|
|
{{ $t('message.createNewIndex') }}
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
</div>
|
2020-12-01 16:48:20 +01:00
|
|
|
</div>
|
|
|
|
</div>
|
2021-12-10 17:34:44 +01:00
|
|
|
</template>
|
2020-12-01 16:48:20 +01:00
|
|
|
</ConfirmModal>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
import { uidGen } from 'common/libs/uidGen';
|
|
|
|
import ConfirmModal from '@/components/BaseConfirmModal';
|
|
|
|
|
|
|
|
export default {
|
2021-08-12 09:54:13 +02:00
|
|
|
name: 'WorkspaceTabPropsTableIndexesModal',
|
2020-12-01 16:48:20 +01:00
|
|
|
components: {
|
|
|
|
ConfirmModal
|
|
|
|
},
|
|
|
|
props: {
|
|
|
|
localIndexes: Array,
|
|
|
|
table: String,
|
|
|
|
fields: Array,
|
|
|
|
workspace: Object,
|
|
|
|
indexTypes: Array
|
|
|
|
},
|
2022-04-22 12:16:02 +02:00
|
|
|
emits: ['hide', 'indexes-update'],
|
2020-12-01 16:48:20 +01:00
|
|
|
data () {
|
|
|
|
return {
|
|
|
|
indexesProxy: [],
|
|
|
|
isOptionsChanging: false,
|
|
|
|
selectedIndexID: '',
|
|
|
|
modalInnerHeight: 400
|
|
|
|
};
|
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
selectedIndexObj () {
|
2021-11-08 15:12:37 +01:00
|
|
|
return this.indexesProxy.find(index => index._antares_id === this.selectedIndexID);
|
2020-12-01 16:48:20 +01:00
|
|
|
},
|
|
|
|
isChanged () {
|
|
|
|
return JSON.stringify(this.localIndexes) !== JSON.stringify(this.indexesProxy);
|
|
|
|
},
|
|
|
|
hasPrimary () {
|
|
|
|
return this.indexesProxy.some(index => index.type === 'PRIMARY');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
mounted () {
|
|
|
|
this.indexesProxy = JSON.parse(JSON.stringify(this.localIndexes));
|
|
|
|
|
|
|
|
if (this.indexesProxy.length)
|
|
|
|
this.resetSelectedID();
|
|
|
|
|
|
|
|
this.getModalInnerHeight();
|
|
|
|
window.addEventListener('resize', this.getModalInnerHeight);
|
|
|
|
},
|
2022-04-21 14:39:24 +02:00
|
|
|
unmounted () {
|
2020-12-01 16:48:20 +01:00
|
|
|
window.removeEventListener('resize', this.getModalInnerHeight);
|
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
confirmIndexesChange () {
|
2021-07-17 09:59:45 +02:00
|
|
|
this.indexesProxy = this.indexesProxy.filter(index => index.fields.length);
|
2020-12-01 16:48:20 +01:00
|
|
|
this.$emit('indexes-update', this.indexesProxy);
|
|
|
|
},
|
|
|
|
selectIndex (event, id) {
|
|
|
|
if (this.selectedIndexID !== id && !event.target.classList.contains('remove-field'))
|
|
|
|
this.selectedIndexID = id;
|
|
|
|
},
|
|
|
|
getModalInnerHeight () {
|
|
|
|
const modalBody = document.querySelector('.modal-body');
|
|
|
|
if (modalBody)
|
|
|
|
this.modalInnerHeight = modalBody.clientHeight - (parseFloat(getComputedStyle(modalBody).paddingTop) + parseFloat(getComputedStyle(modalBody).paddingBottom));
|
|
|
|
},
|
|
|
|
addIndex () {
|
|
|
|
this.indexesProxy = [...this.indexesProxy, {
|
2021-11-08 15:12:37 +01:00
|
|
|
_antares_id: uidGen(),
|
2020-12-01 16:48:20 +01:00
|
|
|
name: 'NEW_INDEX',
|
|
|
|
fields: [],
|
|
|
|
type: 'INDEX',
|
|
|
|
comment: '',
|
|
|
|
indexType: 'BTREE',
|
|
|
|
indexComment: '',
|
|
|
|
cardinality: 0
|
|
|
|
}];
|
|
|
|
|
|
|
|
if (this.indexesProxy.length === 1)
|
|
|
|
this.resetSelectedID();
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
this.$refs.indexesPanel.scrollTop = this.$refs.indexesPanel.scrollHeight + 60;
|
|
|
|
}, 20);
|
|
|
|
},
|
|
|
|
removeIndex (id) {
|
2021-11-08 15:12:37 +01:00
|
|
|
this.indexesProxy = this.indexesProxy.filter(index => index._antares_id !== id);
|
2020-12-01 16:48:20 +01:00
|
|
|
|
|
|
|
if (this.selectedIndexID === id && this.indexesProxy.length)
|
|
|
|
this.resetSelectedID();
|
|
|
|
},
|
|
|
|
clearChanges () {
|
|
|
|
this.indexesProxy = JSON.parse(JSON.stringify(this.localIndexes));
|
2021-11-08 15:12:37 +01:00
|
|
|
if (!this.indexesProxy.some(index => index._antares_id === this.selectedIndexID))
|
2020-12-01 16:48:20 +01:00
|
|
|
this.resetSelectedID();
|
|
|
|
},
|
|
|
|
toggleField (field) {
|
|
|
|
this.indexesProxy = this.indexesProxy.map(index => {
|
2021-11-08 15:12:37 +01:00
|
|
|
if (index._antares_id === this.selectedIndexID) {
|
2020-12-01 16:48:20 +01:00
|
|
|
if (index.fields.includes(field))
|
|
|
|
index.fields = index.fields.filter(f => f !== field);
|
|
|
|
else
|
|
|
|
index.fields.push(field);
|
|
|
|
}
|
|
|
|
return index;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
resetSelectedID () {
|
2021-11-08 15:12:37 +01:00
|
|
|
this.selectedIndexID = this.indexesProxy.length ? this.indexesProxy[0]._antares_id : '';
|
2020-12-01 16:48:20 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
.tile {
|
2021-06-24 21:49:46 +02:00
|
|
|
border-radius: $border-radius;
|
2020-12-01 16:48:20 +01:00
|
|
|
opacity: 0.5;
|
|
|
|
transition: background 0.2s;
|
|
|
|
transition: opacity 0.2s;
|
|
|
|
|
|
|
|
.tile-action {
|
|
|
|
opacity: 0;
|
|
|
|
transition: opacity 0.2s;
|
|
|
|
}
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
.tile-action {
|
|
|
|
opacity: 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-03 11:21:58 +02:00
|
|
|
&.selected-element {
|
2020-12-01 16:48:20 +01:00
|
|
|
opacity: 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.fields-list {
|
2020-12-15 17:08:36 +01:00
|
|
|
max-height: 300px;
|
2020-12-01 16:48:20 +01:00
|
|
|
overflow: auto;
|
|
|
|
}
|
|
|
|
|
|
|
|
.remove-field .mdi {
|
|
|
|
pointer-events: none;
|
|
|
|
}
|
|
|
|
</style>
|