Files
legado/modules/web/src/components/ToolBar.vue
2023-07-25 21:35:32 +08:00

353 lines
8.5 KiB
Vue

<template>
<div class="menu flex-column-center">
<el-button
v-for="button in buttons"
size="large"
:key="button.name"
@click="button.action"
>
{{ button.name }}
</el-button>
<el-button size="large" @click="() => (hotkeysDialogVisible = true)"
>快捷键</el-button
>
</div>
<el-dialog
v-model="hotkeysDialogVisible"
:show-close="false"
:before-close="stopRecordKeyDown"
>
<template #header="{ titleClass, titleId }">
<div class="hotkeys-header flex-space-between">
<div :id="titleId" :class="titleClass">
快捷键设置
<span v-if="recordKeyDowning">
<el-text> / 录入中 </el-text>
</span>
</div>
<el-button
:disabled="recordKeyDowning"
@click="saveHotKeys"
:icon="CircleCheckFilled"
>保存</el-button
>
</div>
</template>
<div class="hotkeys-settings flex-column-center">
<div
v-for="(button, index) in buttons"
:key="button.name"
class="hotkeys-item flex-space-between"
>
<span class="title"
><el-text>{{ button.name }}</el-text></span
>
<div class="hotkeys-item__content">
<div v-for="(key, index) in button.hotKeys" :key="key">
<kbd>{{ key }}</kbd>
<span v-if="index + 1 < button.hotKeys.length">
<el-text>+</el-text>
</span>
</div>
<span v-if="button.hotKeys.length == 0">未设置</span>
</div>
<el-button
:disabled="recordKeyDowning"
text
:icon="Edit"
@click="recordKeyDown(index)"
>编辑</el-button
>
</div>
</div>
</el-dialog>
</template>
<script setup>
import API from "@api";
import { CircleCheckFilled, Edit } from "@element-plus/icons-vue";
import hotkeys from "hotkeys-js";
import { isInvaildSource } from "../utils/souce";
const store = useSourceStore();
const pull = () => {
const loadingMsg = ElMessage({
message: "加载中……",
showClose: true,
duration: 0,
});
API.getSources()
.then(({ data }) => {
if (data.isSuccess) {
store.changeTabName("editList");
store.saveSources(data.data);
ElMessage({
message: `成功拉取${data.data.length}条源`,
type: "success",
});
} else {
ElMessage({
message: data.errorMsg ?? "后端错误",
type: "error",
});
}
})
.finally(() => loadingMsg.close());
};
const push = () => {
let sources = store.sources;
store.changeTabName("editList");
if (sources.length === 0) {
return ElMessage({
message: "空空如也",
type: "info",
});
}
ElMessage({
message: "正在推送中",
type: "info",
});
API.saveSources(sources).then(({ data }) => {
if (data.isSuccess) {
let okData = data.data;
if (Array.isArray(okData)) {
let failMsg = ``;
if (sources.length > okData.length) {
failMsg = "\n推送失败的源将用红色字体标注!";
store.setPushReturnSources(okData);
}
ElMessage({
message: `批量推送源到「阅读3.0APP」\n共计: ${
sources.length
}\n成功: ${okData.length}\n失败: ${
sources.length - okData.length
}${failMsg}`,
type: "success",
});
}
} else {
ElMessage({
message: `批量推送源失败!\nErrorMsg: ${data.errorMsg}`,
type: "error",
});
}
});
};
const conver2Tab = () => {
store.changeTabName("editTab");
store.changeEditTabSource(store.currentSource);
};
const conver2Source = () => {
store.changeCurrentSource(store.editTabSource);
};
const undo = () => {
store.editHistoryUndo();
};
const clearEdit = () => {
store.clearEdit();
ElMessage({
message: "已清除",
type: "success",
});
};
const redo = () => {
store.clearEdit();
store.clearAllHistory();
ElMessage({
message: "已清除所有历史记录",
type: "success",
});
};
const saveSource = () => {
let isBookSource = /bookSource/.test(location.href),
/** @type {import("@/source.js").Source} */
source = store.currentSource;
if (isInvaildSource(source)) {
API.saveSource(source).then(({ data }) => {
if (data.isSuccess) {
ElMessage({
message: `源《${
isBookSource ? source.bookSourceName : source.sourceName
}》已成功保存到「阅读3.0APP」`,
type: "success",
});
//save to store
store.saveCurrentSource();
} else {
ElMessage({
message: `源《${
isBookSource ? source.bookSourceName : source.sourceName
}》保存失败!\nErrorMsg: ${data.errorMsg}`,
type: "error",
});
}
});
} else {
ElMessage({
message: `请检查<必填>项是否全部填写`,
type: "error",
});
}
};
const debug = () => {
store.startDebug();
};
const buttons = ref(
Array.of(
{ name: "⇈推送源", hotKeys: [], action: push },
{ name: "⇊拉取源", hotKeys: [], action: pull },
{ name: "⋙生成源", hotKeys: [], action: conver2Tab },
{ name: "⋘编辑源", hotKeys: [], action: conver2Source },
{ name: "✗清空表单", hotKeys: [], action: clearEdit },
{ name: "↶撤销操作", hotKeys: [], action: undo },
{ name: "↷重做操作", hotKeys: [], action: redo },
{ name: "⇏调试源", hotKeys: [], action: debug },
{ name: "✓保存源", hotKeys: [], action: saveSource },
),
);
const hotkeysDialogVisible = ref(true);
const recordKeyDowning = ref(false);
const recordKeyDownIndex = ref(-1);
const stopRecordKeyDown = () => {
if (!recordKeyDowning.value) {
hotkeysDialogVisible.value = false;
}
recordKeyDowning.value = false;
};
watch(
hotkeysDialogVisible,
(visibale) => {
if (!visibale) {
hotkeys.unbind("*");
readHotkeysConfig();
bindHotKeys();
return;
}
readHotkeysConfig();
hotkeys.unbind();
/**监听按键 */
hotkeys("*", (event) => {
event.preventDefault();
let pressedKeys = hotkeys.getPressedKeyString();
if (pressedKeys.length == 1 && pressedKeys[0] == "esc") {
//单独按下esc 不录入
return;
}
if (recordKeyDowning.value && recordKeyDownIndex.value > -1)
buttons.value[recordKeyDownIndex.value].hotKeys = pressedKeys;
});
},
{ immediate: true },
);
const recordKeyDown = (index) => {
recordKeyDowning.value = true;
ElMessage({
message: "按ESC键或者点击空白处结束录入",
type: "info",
});
buttons.value[index].hotKeys = [];
recordKeyDownIndex.value = index;
};
const saveHotKeys = () => {
const hotKeysConfig = [];
buttons.value.forEach(({ hotKeys }) => {
hotKeysConfig.push(hotKeys);
});
saveHotkeysConfig(hotKeysConfig);
hotkeysDialogVisible.value = false;
};
const bindHotKeys = () => {
// hotkeys默认过滤INPUT SELECT TEXTAREA
hotkeys.filter = () => true;
buttons.value.forEach(({ hotKeys, action }) => {
if (hotKeys.length == 0) return;
hotkeys(hotKeys.join("+"), (event) => {
event.preventDefault();
action.call(null);
});
});
};
const saveHotkeysConfig = (config) => {
localStorage.setItem("legado_web_hotkeys", JSON.stringify(config));
};
/**
* 读取快捷键配置
* @return 是否成功读取配置
*/
function readHotkeysConfig() {
try {
const config = JSON.parse(localStorage.getItem("legado_web_hotkeys"));
if (!Array.isArray(config) || config.length == 0) return false;
buttons.value.forEach((button, index) => (button.hotKeys = config[index]));
return true;
} catch {
ElMessage({ message: "快捷键配置错误", type: "error" });
localStorage.removeItem("legado_web_hotkeys");
}
return false;
}
onMounted(() => {
/**读取热键配置 */
if (readHotkeysConfig()) {
hotkeysDialogVisible.value = false;
}
});
</script>
<style lang="scss" scoped>
.flex-space-between {
display: flex;
justify-content: space-between;
align-items: baseline;
}
.flex-column-center {
display: flex;
flex-direction: column;
justify-content: center;
}
.menu > .el-button {
margin: 4px;
padding: 1em;
width: 6em;
}
.hotkeys-item {
.title {
width: 5em;
display: flex;
justify-content: flex-end;
margin-right: 1em;
}
&__content {
display: flex;
flex-wrap: wrap;
flex: 1;
div {
margin-bottom: 1em;
}
span {
margin: 0.5em;
}
}
}
</style>