260 lines
8.5 KiB
Vue
260 lines
8.5 KiB
Vue
<template>
|
||
<div class="table col column no-wrap">
|
||
<!-- header -->
|
||
<div class="table-row row">
|
||
<div class="desc q-pa-sm bg-header-3">
|
||
Команда
|
||
</div>
|
||
<div class="hotKeys col q-pa-sm bg-header-3 row no-wrap">
|
||
<div style="width: 80px">
|
||
Сочетание клавиш
|
||
</div>
|
||
<q-input
|
||
ref="input"
|
||
v-model="search"
|
||
class="q-ml-sm col"
|
||
outlined dense
|
||
bg-color="input"
|
||
placeholder="Найти"
|
||
@click.stop
|
||
/>
|
||
<div v-show="!readonly" class="q-ml-sm column justify-center">
|
||
<q-btn class="bg-grey-4 text-grey-6" style="height: 35px; width: 35px" rounded flat icon="la la-broom" @click="defaultHotKeyAll">
|
||
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||
Установить все сочетания по умолчанию
|
||
</q-tooltip>
|
||
</q-btn>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- body -->
|
||
<div v-for="(action, index) in tableData" :key="index" class="table-row row">
|
||
<div class="desc q-pa-sm">
|
||
{{ rstore.readerActions[action] }}
|
||
</div>
|
||
<div class="hotKeys col q-pa-sm">
|
||
<q-chip
|
||
v-for="(code, index2) in modelValue[action]" :key="index2"
|
||
:color="collisions[code] ? 'red' : 'grey-7'"
|
||
:removable="!readonly" :clickable="collisions[code] ? true : false"
|
||
text-color="white" @remove="removeCode(action, code)"
|
||
@click="collisionWarning(code)"
|
||
>
|
||
{{ code }}
|
||
</q-chip>
|
||
</div>
|
||
<div v-show="!readonly" class="column q-pa-xs">
|
||
<q-icon
|
||
v-ripple
|
||
:disabled="(modelValue[action].length >= maxCodesLength) || null"
|
||
name="la la-plus-circle"
|
||
class="button bg-green-8 text-white"
|
||
@click="addHotKey(action)"
|
||
>
|
||
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||
Добавить сочетание клавиш
|
||
</q-tooltip>
|
||
</q-icon>
|
||
<q-icon
|
||
v-ripple
|
||
name="la la-broom"
|
||
class="button text-grey-5"
|
||
@click="defaultHotKey(action)"
|
||
>
|
||
<q-tooltip :delay="1000" anchor="top middle" self="bottom middle" content-style="font-size: 80%">
|
||
По умолчанию
|
||
</q-tooltip>
|
||
</q-icon>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script>
|
||
//-----------------------------------------------------------------------------
|
||
import vueComponent from '../../../../vueComponent.js';
|
||
|
||
import rstore from '../../../../../store/modules/reader';
|
||
|
||
const componentOptions = {
|
||
watch: {
|
||
search: function() {
|
||
this.updateTableData();
|
||
},
|
||
modelValue: function() {
|
||
this.checkCollisions();
|
||
this.updateTableData();
|
||
}
|
||
},
|
||
};
|
||
class UserHotKeys {
|
||
_options = componentOptions;
|
||
_props = {
|
||
modelValue: Object,
|
||
readonly: Boolean,
|
||
};
|
||
|
||
search = '';
|
||
rstore = {};
|
||
tableData = [];
|
||
collisions = {};
|
||
maxCodesLength = 10;
|
||
|
||
created() {
|
||
this.rstore = rstore;
|
||
}
|
||
|
||
mounted() {
|
||
this.checkCollisions();
|
||
this.updateTableData();
|
||
}
|
||
|
||
get mode() {
|
||
return this.$store.state.config.mode;
|
||
}
|
||
|
||
updateTableData() {
|
||
let result = rstore.hotKeys.map(hk => hk.name);
|
||
|
||
const search = this.search.toLowerCase();
|
||
const codesIncludeSearch = (action) => {
|
||
for (const code of this.modelValue[action]) {
|
||
if (code.toLowerCase().includes(search))
|
||
return true;
|
||
}
|
||
return false;
|
||
};
|
||
|
||
result = result.filter(item => {
|
||
return !search ||
|
||
rstore.readerActions[item].toLowerCase().includes(search) ||
|
||
codesIncludeSearch(item)
|
||
});
|
||
|
||
this.tableData = result;
|
||
}
|
||
|
||
checkCollisions() {
|
||
const cols = {};
|
||
for (const [action, codes] of Object.entries(this.modelValue)) {
|
||
codes.forEach(code => {
|
||
if (!cols[code])
|
||
cols[code] = [];
|
||
if (cols[code].indexOf(action) < 0)
|
||
cols[code].push(action);
|
||
});
|
||
}
|
||
|
||
const result = {};
|
||
for (const [code, actions] of Object.entries(cols)) {
|
||
if (actions.length > 1)
|
||
result[code] = actions;
|
||
}
|
||
|
||
this.collisions = result;
|
||
}
|
||
|
||
collisionWarning(code) {
|
||
if (this.collisions[code]) {
|
||
const descs = this.collisions[code].map(action => `<b>${rstore.readerActions[action]}</b>`);
|
||
this.$root.stdDialog.alert(`Сочетание '${code}' одновременно назначено<br>следующим командам:<br>${descs.join('<br>')}<br><br>
|
||
Возможно неожиданное поведение.`, 'Предупреждение');
|
||
}
|
||
}
|
||
|
||
removeCode(action, code) {
|
||
let codes = Array.from(this.modelValue[action]);
|
||
const index = codes.indexOf(code);
|
||
if (index >= 0) {
|
||
codes.splice(index, 1);
|
||
const newValue = Object.assign({}, this.modelValue, {[action]: codes});
|
||
this.$emit('update:modelValue', newValue);
|
||
}
|
||
}
|
||
|
||
async addHotKey(action) {
|
||
if (this.modelValue[action].length >= this.maxCodesLength)
|
||
return;
|
||
try {
|
||
const result = await this.$root.stdDialog.getHotKey(`Добавить сочетание для:<br><b>${rstore.readerActions[action]}</b>`, '');
|
||
if (result) {
|
||
let codes = Array.from(this.modelValue[action]);
|
||
if (codes.indexOf(result) < 0) {
|
||
codes.push(result);
|
||
const newValue = Object.assign({}, this.modelValue, {[action]: codes});
|
||
this.$emit('update:modelValue', newValue);
|
||
this.$nextTick(() => {
|
||
this.collisionWarning(result);
|
||
});
|
||
}
|
||
}
|
||
} catch (e) {
|
||
//
|
||
}
|
||
}
|
||
|
||
async defaultHotKey(action) {
|
||
try {
|
||
if (await this.$root.stdDialog.confirm(`Подтвердите сброс сочетаний клавиш<br>в значения по умолчанию для команды:<br><b>${rstore.readerActions[action]}</b>`, ' ')) {
|
||
const codes = Array.from(rstore.settingDefaults.userHotKeys[action]);
|
||
const newValue = Object.assign({}, this.modelValue, {[action]: codes});
|
||
this.$emit('update:modelValue', newValue);
|
||
}
|
||
} catch (e) {
|
||
//
|
||
}
|
||
}
|
||
|
||
async defaultHotKeyAll() {
|
||
try {
|
||
if (await this.$root.stdDialog.confirm('Подтвердите сброс сочетаний клавиш<br>для ВСЕХ команд в значения по умолчанию:', ' ')) {
|
||
const newValue = Object.assign({}, rstore.settingDefaults.userHotKeys);
|
||
this.$emit('update:modelValue', newValue);
|
||
}
|
||
} catch (e) {
|
||
//
|
||
}
|
||
}
|
||
}
|
||
|
||
export default vueComponent(UserHotKeys);
|
||
//-----------------------------------------------------------------------------
|
||
</script>
|
||
|
||
<style scoped>
|
||
.table {
|
||
border-left: 1px solid grey;
|
||
border-top: 1px solid grey;
|
||
}
|
||
|
||
.table-row {
|
||
border-right: 1px solid grey;
|
||
border-bottom: 1px solid grey;
|
||
}
|
||
|
||
.table-row:nth-child(even) {
|
||
background-color: var(--bg-menu-color1);
|
||
}
|
||
|
||
.table-row:hover {
|
||
background-color: var(--bg-menu-color2);
|
||
}
|
||
|
||
.desc {
|
||
width: 130px;
|
||
overflow-wrap: break-word;
|
||
word-wrap: break-word;
|
||
white-space: normal;
|
||
}
|
||
|
||
.hotKeys {
|
||
border-left: 1px solid grey;
|
||
}
|
||
|
||
.button {
|
||
font-size: 25px;
|
||
border-radius: 25px;
|
||
cursor: pointer;
|
||
}
|
||
</style> |