mirror of
https://github.com/AkiChase/scrcpy-mask
synced 2025-04-21 04:25:13 +08:00
433 lines
11 KiB
Vue
433 lines
11 KiB
Vue
<script setup lang="ts">
|
|
import { h, nextTick, onActivated, onMounted, ref } from "vue";
|
|
import { NDialog, NInput, useDialog, useMessage } from "naive-ui";
|
|
import { useGlobalStore } from "../store/global";
|
|
import { onBeforeRouteLeave, useRouter } from "vue-router";
|
|
import {
|
|
applyShortcuts,
|
|
clearShortcuts,
|
|
listenToEvent,
|
|
unlistenToEvent,
|
|
updateScreenSizeAndMaskArea,
|
|
} from "../hotkey";
|
|
import { KeyMappingConfig, KeySteeringWheel } from "../keyMappingConfig";
|
|
import { getVersion } from "@tauri-apps/api/app";
|
|
import { fetch } from "@tauri-apps/plugin-http";
|
|
import { open } from "@tauri-apps/plugin-shell";
|
|
import {
|
|
sendInjectKeycode,
|
|
sendSetClipboard,
|
|
} from "../frontcommand/controlMsg";
|
|
import { getCurrent, PhysicalSize } from "@tauri-apps/api/window";
|
|
import {
|
|
AndroidKeyEventAction,
|
|
AndroidKeycode,
|
|
AndroidMetastate,
|
|
} from "../frontcommand/android";
|
|
import { Store } from "@tauri-apps/plugin-store";
|
|
import { useI18n } from "vue-i18n";
|
|
|
|
const { t } = useI18n();
|
|
const store = useGlobalStore();
|
|
const router = useRouter();
|
|
const message = useMessage();
|
|
const dialog = useDialog();
|
|
|
|
const showInputBoxRef = ref(false);
|
|
const inputBoxVal = ref("");
|
|
const inputInstRef = ref<HTMLInputElement | null>(null);
|
|
|
|
onBeforeRouteLeave(() => {
|
|
if (store.controledDevice) {
|
|
unlistenToEvent();
|
|
clearShortcuts();
|
|
}
|
|
});
|
|
|
|
onActivated(async () => {
|
|
cleanAfterimage();
|
|
const maskElement = document.getElementById("maskElement") as HTMLElement;
|
|
|
|
if (store.controledDevice) {
|
|
updateScreenSizeAndMaskArea(
|
|
[store.screenSizeW, store.screenSizeH],
|
|
[maskElement.clientWidth, maskElement.clientHeight]
|
|
);
|
|
|
|
if (
|
|
applyShortcuts(
|
|
maskElement,
|
|
store.keyMappingConfigList[store.curKeyMappingIndex]
|
|
)
|
|
) {
|
|
listenToEvent();
|
|
} else {
|
|
message.error(t("pages.Mask.keyconfigException"));
|
|
}
|
|
}
|
|
});
|
|
|
|
onMounted(async () => {
|
|
await loadLocalStore();
|
|
store.checkUpdate = checkUpdate;
|
|
store.showInputBox = showInputBox;
|
|
if (store.checkUpdateAtStart) checkUpdate();
|
|
});
|
|
|
|
async function loadLocalStore() {
|
|
const localStore = new Store("store.bin");
|
|
// loading screenSize from local store
|
|
const screenSize = await localStore.get<{ sizeW: number; sizeH: number }>(
|
|
"screenSize"
|
|
);
|
|
if (screenSize !== null) {
|
|
store.screenSizeW = screenSize.sizeW;
|
|
store.screenSizeH = screenSize.sizeH;
|
|
}
|
|
|
|
// loading keyMappingConfigList from local store
|
|
let keyMappingConfigList = await localStore.get<KeyMappingConfig[]>(
|
|
"keyMappingConfigList"
|
|
);
|
|
if (keyMappingConfigList === null || keyMappingConfigList.length === 0) {
|
|
// add empty key mapping config
|
|
// unable to get mask element when app is not ready
|
|
// so we use the stored mask area to get relative size
|
|
const maskArea = await localStore.get<{
|
|
posX: number;
|
|
posY: number;
|
|
sizeW: number;
|
|
sizeH: number;
|
|
}>("maskArea");
|
|
let relativeSize = { w: 800, h: 600 };
|
|
if (maskArea !== null) {
|
|
relativeSize = {
|
|
w: maskArea.sizeW,
|
|
h: maskArea.sizeH,
|
|
};
|
|
}
|
|
keyMappingConfigList = [
|
|
{
|
|
relativeSize,
|
|
title: t("pages.Mask.blankConfig"),
|
|
list: [],
|
|
},
|
|
];
|
|
await localStore.set("keyMappingConfigList", keyMappingConfigList);
|
|
}
|
|
store.keyMappingConfigList = keyMappingConfigList;
|
|
|
|
// loading curKeyMappingIndex from local store
|
|
let curKeyMappingIndex = await localStore.get<number>("curKeyMappingIndex");
|
|
if (
|
|
curKeyMappingIndex === null ||
|
|
curKeyMappingIndex >= keyMappingConfigList.length
|
|
) {
|
|
curKeyMappingIndex = 0;
|
|
localStore.set("curKeyMappingIndex", curKeyMappingIndex);
|
|
}
|
|
store.curKeyMappingIndex = curKeyMappingIndex;
|
|
|
|
// loading maskButton from local store
|
|
let maskButton = await localStore.get<{
|
|
show: boolean;
|
|
transparency: number;
|
|
}>("maskButton");
|
|
store.maskButton = maskButton ?? {
|
|
show: true,
|
|
transparency: 0.5,
|
|
};
|
|
|
|
// loading checkUpdateAtStart from local store
|
|
let checkUpdateAtStart = await localStore.get<boolean>("checkUpdateAtStart");
|
|
store.checkUpdateAtStart = checkUpdateAtStart ?? true;
|
|
}
|
|
|
|
async function cleanAfterimage() {
|
|
const appWindow = getCurrent();
|
|
const oldSize = await appWindow.outerSize();
|
|
const newSize = new PhysicalSize(oldSize.width, oldSize.height + 1);
|
|
await appWindow.setSize(newSize);
|
|
await appWindow.setSize(oldSize);
|
|
}
|
|
|
|
function handleInputBoxClick(event: MouseEvent) {
|
|
if (event.target === document.getElementById("input-box")) {
|
|
showInputBox(false);
|
|
}
|
|
}
|
|
|
|
function handleInputKeyUp(event: KeyboardEvent) {
|
|
if (event.key === "Enter") {
|
|
pasteText();
|
|
} else if (event.key === "Escape") {
|
|
showInputBox(false);
|
|
}
|
|
}
|
|
|
|
function showInputBox(flag: boolean) {
|
|
if (flag) {
|
|
unlistenToEvent();
|
|
inputBoxVal.value = "";
|
|
showInputBoxRef.value = true;
|
|
document.addEventListener("keyup", handleInputKeyUp);
|
|
nextTick(() => {
|
|
inputInstRef.value?.focus();
|
|
});
|
|
} else {
|
|
document.removeEventListener("keyup", handleInputKeyUp);
|
|
inputInstRef.value?.blur();
|
|
showInputBoxRef.value = false;
|
|
listenToEvent();
|
|
nextTick(() => {
|
|
cleanAfterimage();
|
|
});
|
|
}
|
|
}
|
|
|
|
function sleep(time: number) {
|
|
return new Promise<void>((resolve) => {
|
|
setTimeout(() => {
|
|
resolve();
|
|
}, time);
|
|
});
|
|
}
|
|
|
|
async function pasteText() {
|
|
showInputBox(false);
|
|
if (!inputBoxVal.value) return;
|
|
sendSetClipboard({
|
|
sequence: new Date().getTime() % 100000,
|
|
text: inputBoxVal.value,
|
|
paste: true,
|
|
});
|
|
await sleep(300);
|
|
// send enter
|
|
await sendInjectKeycode({
|
|
action: AndroidKeyEventAction.AKEY_EVENT_ACTION_DOWN,
|
|
keycode: AndroidKeycode.AKEYCODE_ENTER,
|
|
repeat: 0,
|
|
metastate: AndroidMetastate.AMETA_NONE,
|
|
});
|
|
await sleep(50);
|
|
await sendInjectKeycode({
|
|
action: AndroidKeyEventAction.AKEY_EVENT_ACTION_UP,
|
|
keycode: AndroidKeycode.AKEYCODE_ENTER,
|
|
repeat: 0,
|
|
metastate: AndroidMetastate.AMETA_NONE,
|
|
});
|
|
}
|
|
|
|
function toStartServer() {
|
|
router.replace({ name: "device" });
|
|
}
|
|
|
|
function renderUpdateInfo(content: string) {
|
|
const pList = content.split("\r\n").map((line: string) => h("p", line));
|
|
return h("div", { style: "margin: 20px 0" }, pList);
|
|
}
|
|
|
|
async function checkUpdate() {
|
|
try {
|
|
const curVersion = await getVersion();
|
|
const res = await fetch(
|
|
"https://api.github.com/repos/AkiChase/scrcpy-mask/releases/latest",
|
|
{
|
|
connectTimeout: 5000,
|
|
}
|
|
);
|
|
if (res.status !== 200) {
|
|
message.error(t("pages.Mask.checkUpdate.failed"));
|
|
} else {
|
|
const data = await res.json();
|
|
const latestVersion = (data.tag_name as string).slice(1);
|
|
if (latestVersion <= curVersion) {
|
|
message.success(t("pages.Mask.checkUpdate.isLatest", [latestVersion]));
|
|
return;
|
|
}
|
|
const body = data.body as string;
|
|
dialog.info({
|
|
title: t("pages.Mask.checkUpdate.notLatest.title", [latestVersion]),
|
|
content: () => renderUpdateInfo(body),
|
|
positiveText: t("pages.Mask.checkUpdate.notLatest.positiveText"),
|
|
negativeText: t("pages.Mask.checkUpdate.notLatest.negativeText"),
|
|
onPositiveClick: () => {
|
|
open(data.html_url);
|
|
},
|
|
});
|
|
}
|
|
} catch (e) {
|
|
console.error(e);
|
|
message.error(t("pages.Mask.checkUpdate.failed"));
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<div v-show="!store.controledDevice" class="notice">
|
|
<div class="content">
|
|
<NDialog
|
|
:closable="false"
|
|
:title="$t('pages.Mask.noControledDevice.title')"
|
|
:content="$t('pages.Mask.noControledDevice.content')"
|
|
:positive-text="$t('pages.Mask.noControledDevice.positiveText')"
|
|
type="warning"
|
|
@positive-click="toStartServer"
|
|
/>
|
|
</div>
|
|
</div>
|
|
<template v-if="store.keyMappingConfigList.length">
|
|
<div @contextmenu.prevent class="mask" id="maskElement"></div>
|
|
<div
|
|
v-show="showInputBoxRef"
|
|
class="input-box"
|
|
id="input-box"
|
|
@click="handleInputBoxClick"
|
|
>
|
|
<NInput
|
|
ref="inputInstRef"
|
|
v-model:value="inputBoxVal"
|
|
type="text"
|
|
:placeholder="$t('pages.Mask.inputBoxPlaceholder')"
|
|
/>
|
|
</div>
|
|
<div
|
|
v-if="store.maskButton.show"
|
|
:style="'--transparency: ' + store.maskButton.transparency"
|
|
class="button-layer"
|
|
>
|
|
<!-- <div style="position: absolute;height: 100%;width: 1px;background-color: red;left: 50%;"></div>
|
|
<div style="position: absolute;width: 100%;height: 1px;background-color: red;top: 56.6%;"></div> -->
|
|
<template
|
|
v-for="button in store.keyMappingConfigList[store.curKeyMappingIndex]
|
|
.list"
|
|
>
|
|
<div
|
|
v-if="button.type === 'SteeringWheel'"
|
|
class="mask-steering-wheel"
|
|
:style="{
|
|
left: button.posX - 75 + 'px',
|
|
top: button.posY - 75 + 'px',
|
|
}"
|
|
>
|
|
<div class="wheel-container">
|
|
<i />
|
|
<span>{{ (button as KeySteeringWheel).key.up }}</span>
|
|
<i />
|
|
<span>{{ (button as KeySteeringWheel).key.left }}</span>
|
|
<i />
|
|
<span>{{ (button as KeySteeringWheel).key.right }}</span>
|
|
<i />
|
|
<span>{{ (button as KeySteeringWheel).key.down }}</span>
|
|
<i />
|
|
</div>
|
|
</div>
|
|
<div
|
|
v-else
|
|
class="mask-button"
|
|
:style="{
|
|
left: button.posX + 'px',
|
|
top: button.posY - 14 + 'px',
|
|
}"
|
|
>
|
|
{{ button.key }}
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</template>
|
|
</template>
|
|
|
|
<style scoped lang="scss">
|
|
.mask {
|
|
background-color: transparent;
|
|
overflow: hidden;
|
|
cursor: pointer;
|
|
position: relative;
|
|
border-right: 1px solid var(--bg-color);
|
|
border-bottom: 1px solid var(--bg-color);
|
|
border-radius: 0 0 5px 0;
|
|
user-select: none;
|
|
-webkit-user-select: none;
|
|
z-index: 2;
|
|
}
|
|
|
|
.input-box {
|
|
z-index: 4;
|
|
position: absolute;
|
|
left: 70px;
|
|
top: 30px;
|
|
right: 0;
|
|
bottom: 0;
|
|
background-color: rgba(0, 0, 0, 0.5);
|
|
|
|
.n-input {
|
|
width: 50%;
|
|
position: absolute;
|
|
left: 0;
|
|
right: 0;
|
|
bottom: 15%;
|
|
margin: auto;
|
|
background-color: var(--content-bg-color);
|
|
}
|
|
}
|
|
|
|
.button-layer {
|
|
position: absolute;
|
|
left: 70px;
|
|
top: 30px;
|
|
right: 0;
|
|
bottom: 0;
|
|
background-color: transparent;
|
|
user-select: none;
|
|
-webkit-user-select: none;
|
|
z-index: 1;
|
|
|
|
& > .mask-button {
|
|
opacity: var(--transparency);
|
|
position: absolute;
|
|
background-color: black;
|
|
color: white;
|
|
border-radius: 5px;
|
|
padding: 5px;
|
|
font-size: 12px;
|
|
}
|
|
|
|
& > .mask-steering-wheel {
|
|
opacity: var(--transparency);
|
|
position: absolute;
|
|
background-color: black;
|
|
color: white;
|
|
border-radius: 50%;
|
|
width: 150px;
|
|
height: 150px;
|
|
font-size: 12px;
|
|
|
|
.wheel-container {
|
|
display: grid;
|
|
grid-template-columns: repeat(3, 50px);
|
|
grid-template-rows: repeat(3, 50px);
|
|
justify-items: center;
|
|
align-items: center;
|
|
}
|
|
}
|
|
}
|
|
|
|
.notice {
|
|
background-color: rgba(0, 0, 0, 0.5);
|
|
display: flex;
|
|
justify-content: center;
|
|
align-items: center;
|
|
position: absolute;
|
|
left: 70px;
|
|
top: 30px;
|
|
right: 0;
|
|
bottom: 0;
|
|
z-index: 3;
|
|
|
|
.content {
|
|
width: 80%;
|
|
}
|
|
}
|
|
</style>
|
|
h, import { getVersion } from "@tauri-apps/api/app";
|