Compare commits
10 Commits
select-tar
...
9a5d64cc9e
| Author | SHA1 | Date | |
|---|---|---|---|
| 9a5d64cc9e | |||
| 1beb1009b3 | |||
| 9cd32a7aa6 | |||
| 473e6df77b | |||
| 088cd33b0a | |||
| 51fd4acd8b | |||
| 691e33a84e | |||
| e60d24d56c | |||
| e1a9d97596 | |||
| f07facd98b |
5
.gitignore
vendored
5
.gitignore
vendored
@@ -8,6 +8,11 @@ pnpm-debug.log*
|
|||||||
lerna-debug.log*
|
lerna-debug.log*
|
||||||
|
|
||||||
node_modules
|
node_modules
|
||||||
|
.history
|
||||||
|
.github
|
||||||
|
openspec
|
||||||
|
CLAUDE.md
|
||||||
|
dosc
|
||||||
.DS_Store
|
.DS_Store
|
||||||
dist
|
dist
|
||||||
*.local
|
*.local
|
||||||
|
|||||||
@@ -52,7 +52,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
function onDeviceShoot() {
|
function onDeviceShoot() {
|
||||||
audioManager.play("射箭声音")
|
// audioManager.play("射箭声音")
|
||||||
}
|
}
|
||||||
|
|
||||||
function onShootWsMsg(content) {
|
function onShootWsMsg(content) {
|
||||||
|
|||||||
30
src/apis.js
30
src/apis.js
@@ -179,12 +179,39 @@ export const loginAPI = async (phone, nickName, avatarData, code) => {
|
|||||||
return result;
|
return result;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const silentLoginAPI = async (code) => {
|
||||||
|
const result = await request("POST", "/index/code", {
|
||||||
|
appName: "shoot",
|
||||||
|
appId: "wxa8f5989dcd45cc23",
|
||||||
|
code,
|
||||||
|
});
|
||||||
|
uni.setStorageSync(
|
||||||
|
`${uni.getAccountInfoSync().miniProgram.envVersion}_token`,
|
||||||
|
result.token
|
||||||
|
);
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const checkUserBindAPI = async (code) => {
|
||||||
|
return request("POST", "/index/checkBind", {
|
||||||
|
appName: "shoot",
|
||||||
|
appId: "wxa8f5989dcd45cc23",
|
||||||
|
code,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
export const bindDeviceAPI = (device) => {
|
export const bindDeviceAPI = (device) => {
|
||||||
return request("POST", "/user/device/bindDevice", {
|
return request("POST", "/user/device/bindDevice", {
|
||||||
device,
|
device,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const bindDeviceAPIV2 = (token) => {
|
||||||
|
return request("POST", "/user/device/bindDevice/v2", {
|
||||||
|
token: token,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
export const unbindDeviceAPI = (deviceId) => {
|
export const unbindDeviceAPI = (deviceId) => {
|
||||||
return request("POST", "/user/device/unbindDevice", {
|
return request("POST", "/user/device/unbindDevice", {
|
||||||
deviceId,
|
deviceId,
|
||||||
@@ -262,7 +289,8 @@ export const matchGameAPI = (match, gameType, teamSize) => {
|
|||||||
match,
|
match,
|
||||||
gameType,
|
gameType,
|
||||||
teamSize,
|
teamSize,
|
||||||
readyTime: 1.5,
|
readyTime: 15,
|
||||||
|
targetType: 20,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import AppBackground from "@/components/AppBackground.vue";
|
|||||||
import Header from "@/components/Header.vue";
|
import Header from "@/components/Header.vue";
|
||||||
import ScreenHint from "@/components/ScreenHint.vue";
|
import ScreenHint from "@/components/ScreenHint.vue";
|
||||||
import BackToGame from "@/components/BackToGame.vue";
|
import BackToGame from "@/components/BackToGame.vue";
|
||||||
import { laserAimAPI, getBattleAPI } from "@/apis";
|
import {laserAimAPI, getBattleAPI, matchGameAPI} from "@/apis";
|
||||||
import { capsuleHeight, debounce } from "@/util";
|
import { capsuleHeight, debounce } from "@/util";
|
||||||
import AudioManager from "@/audioManager";
|
import AudioManager from "@/audioManager";
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
@@ -135,6 +135,10 @@ const goBack = () => {
|
|||||||
uni.navigateBack();
|
uni.navigateBack();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const cancelMatching = async () => {
|
||||||
|
uni.$emit("cancelMatching");
|
||||||
|
}
|
||||||
|
|
||||||
const goCalibration = async () => {
|
const goCalibration = async () => {
|
||||||
await laserAimAPI();
|
await laserAimAPI();
|
||||||
uni.navigateTo({
|
uni.navigateTo({
|
||||||
@@ -202,7 +206,7 @@ const goCalibration = async () => {
|
|||||||
<button hover-class="none" @click="() => (showHint = false)">
|
<button hover-class="none" @click="() => (showHint = false)">
|
||||||
取消
|
取消
|
||||||
</button>
|
</button>
|
||||||
<button hover-class="none" @click="goBack">确认</button>
|
<button hover-class="none" @click="cancelMatching">确认</button>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
<view v-if="hintType === 4" class="tip-content">
|
<view v-if="hintType === 4" class="tip-content">
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ async function onReceiveMessage(message) {
|
|||||||
currentRoundEnded.value = true;
|
currentRoundEnded.value = true;
|
||||||
audioManager.play("比赛开始");
|
audioManager.play("比赛开始");
|
||||||
} else if (type === MESSAGETYPESV2.BattleEnd) {
|
} else if (type === MESSAGETYPESV2.BattleEnd) {
|
||||||
audioManager.play("比赛结束");
|
audioManager.play("比赛结束", false);
|
||||||
} else if (type === MESSAGETYPESV2.ShootResult) {
|
} else if (type === MESSAGETYPESV2.ShootResult) {
|
||||||
if (melee.value && current.playerId !== user.value.id) return;
|
if (melee.value && current.playerId !== user.value.id) return;
|
||||||
if (current.playerId === user.value.id) currentShot.value++;
|
if (current.playerId === user.value.id) currentShot.value++;
|
||||||
|
|||||||
@@ -53,6 +53,7 @@ const currentRoundEnded = ref(false);
|
|||||||
const ended = ref(false);
|
const ended = ref(false);
|
||||||
const halfTime = ref(false);
|
const halfTime = ref(false);
|
||||||
const wait = ref(0);
|
const wait = ref(0);
|
||||||
|
const transitionStyle = ref("all 1s linear");
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => props.tips,
|
() => props.tips,
|
||||||
@@ -81,7 +82,19 @@ watch(
|
|||||||
|
|
||||||
const resetTimer = (count) => {
|
const resetTimer = (count) => {
|
||||||
if (timer.value) clearInterval(timer.value);
|
if (timer.value) clearInterval(timer.value);
|
||||||
remain.value = Math.round(count);
|
const newVal = Math.round(count);
|
||||||
|
|
||||||
|
// 如果剩余时间增加(如重置),瞬间变化无动画
|
||||||
|
if (newVal >= remain.value) {
|
||||||
|
transitionStyle.value = "none";
|
||||||
|
remain.value = newVal;
|
||||||
|
setTimeout(() => {
|
||||||
|
transitionStyle.value = "all 1s linear";
|
||||||
|
}, 50);
|
||||||
|
} else {
|
||||||
|
remain.value = newVal;
|
||||||
|
}
|
||||||
|
|
||||||
if (remain.value > 0) {
|
if (remain.value > 0) {
|
||||||
timer.value = setInterval(() => {
|
timer.value = setInterval(() => {
|
||||||
if (remain.value === 0) {
|
if (remain.value === 0) {
|
||||||
@@ -188,6 +201,7 @@ onBeforeUnmount(() => {
|
|||||||
width: `${(remain / total) * 100}%`,
|
width: `${(remain / total) * 100}%`,
|
||||||
backgroundColor: barColor,
|
backgroundColor: barColor,
|
||||||
right: tips.includes('红队') ? 0 : 'unset',
|
right: tips.includes('红队') ? 0 : 'unset',
|
||||||
|
transition: transitionStyle,
|
||||||
}"
|
}"
|
||||||
/>
|
/>
|
||||||
<text>剩余{{ remain }}秒</text>
|
<text>剩余{{ remain }}秒</text>
|
||||||
@@ -242,7 +256,6 @@ onBeforeUnmount(() => {
|
|||||||
height: 15px;
|
height: 15px;
|
||||||
border-radius: 15px;
|
border-radius: 15px;
|
||||||
z-index: -1;
|
z-index: -1;
|
||||||
transition: all 1s linear;
|
|
||||||
}
|
}
|
||||||
.container > view:last-child > text {
|
.container > view:last-child > text {
|
||||||
font-size: 10px;
|
font-size: 10px;
|
||||||
|
|||||||
@@ -20,12 +20,23 @@ const barColor = ref("");
|
|||||||
const remain = ref(15);
|
const remain = ref(15);
|
||||||
const timer = ref(null);
|
const timer = ref(null);
|
||||||
const laoding = ref(false);
|
const laoding = ref(false);
|
||||||
|
const transitionStyle = ref("all 1s linear");
|
||||||
|
|
||||||
const updateRemain = (value) => {
|
const updateRemain = (value) => {
|
||||||
// if (Math.ceil(value) === remain.value || Math.floor(value) === remain.value)
|
|
||||||
// return;
|
|
||||||
if (timer.value) clearInterval(timer.value);
|
if (timer.value) clearInterval(timer.value);
|
||||||
remain.value = Math.round(value);
|
const newVal = Math.round(value);
|
||||||
|
|
||||||
|
// 如果剩余时间增加(如轮次切换重置),瞬间变化无动画;否则保持动画
|
||||||
|
if (newVal >= remain.value) {
|
||||||
|
transitionStyle.value = "none";
|
||||||
|
remain.value = newVal;
|
||||||
|
setTimeout(() => {
|
||||||
|
transitionStyle.value = "all 1s linear";
|
||||||
|
}, 50);
|
||||||
|
} else {
|
||||||
|
remain.value = newVal;
|
||||||
|
}
|
||||||
|
|
||||||
timer.value = setInterval(() => {
|
timer.value = setInterval(() => {
|
||||||
laoding.value = remain.value === 0;
|
laoding.value = remain.value === 0;
|
||||||
if (remain.value > 0) remain.value--;
|
if (remain.value > 0) remain.value--;
|
||||||
@@ -72,6 +83,7 @@ onBeforeUnmount(() => {
|
|||||||
width: `${(remain / total) * 100}%`,
|
width: `${(remain / total) * 100}%`,
|
||||||
background: barColor,
|
background: barColor,
|
||||||
right: tips.includes('红队') ? 0 : 'unset',
|
right: tips.includes('红队') ? 0 : 'unset',
|
||||||
|
transition: transitionStyle,
|
||||||
}"
|
}"
|
||||||
/>
|
/>
|
||||||
<text v-if="!laoding">剩余{{ remain }}秒</text>
|
<text v-if="!laoding">剩余{{ remain }}秒</text>
|
||||||
@@ -106,7 +118,6 @@ onBeforeUnmount(() => {
|
|||||||
.container > view:last-child > view {
|
.container > view:last-child > view {
|
||||||
height: 24rpx;
|
height: 24rpx;
|
||||||
border-radius: 15px;
|
border-radius: 15px;
|
||||||
transition: all 1s linear;
|
|
||||||
}
|
}
|
||||||
.container > view:last-child > text {
|
.container > view:last-child > text {
|
||||||
font-size: 18rpx;
|
font-size: 18rpx;
|
||||||
|
|||||||
@@ -86,18 +86,17 @@ const handleLogin = async () => {
|
|||||||
icon: "none",
|
icon: "none",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
await doLogin();
|
||||||
|
};
|
||||||
|
|
||||||
|
async function doLogin() {
|
||||||
loading.value = true;
|
loading.value = true;
|
||||||
try {
|
try {
|
||||||
const wxResult = await wxLogin();
|
const wxResult = await wxLogin();
|
||||||
const fileManager = uni.getFileSystemManager();
|
const fileManager = uni.getFileSystemManager();
|
||||||
const avatarBase64 = fileManager.readFileSync(avatarUrl.value, "base64");
|
const avatarBase64 = fileManager.readFileSync(avatarUrl.value, "base64");
|
||||||
const base64Url = `data:image/png;base64,${avatarBase64}`;
|
const base64Url = `data:image/png;base64,${avatarBase64}`;
|
||||||
const result = await loginAPI(
|
await loginAPI(phone.value, nickName.value, base64Url, wxResult.code);
|
||||||
phone.value,
|
|
||||||
nickName.value,
|
|
||||||
base64Url,
|
|
||||||
wxResult.code
|
|
||||||
);
|
|
||||||
const data = await getHomeData();
|
const data = await getHomeData();
|
||||||
if (data.user) updateUser(data.user);
|
if (data.user) updateUser(data.user);
|
||||||
const devices = await getMyDevicesAPI();
|
const devices = await getMyDevicesAPI();
|
||||||
@@ -139,6 +138,10 @@ const openPrivacyLink = () => {
|
|||||||
|
|
||||||
onShow(() => {
|
onShow(() => {
|
||||||
loading.value = false;
|
loading.value = false;
|
||||||
|
agree.value = false;
|
||||||
|
phone.value = "";
|
||||||
|
avatarUrl.value = "";
|
||||||
|
nickName.value = "";
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name" : "shoot-miniprograms",
|
"name" : "shoot-miniprograms",
|
||||||
"appid": "",
|
"appid" : "__UNI__B03E251",
|
||||||
"description" : "",
|
"description" : "",
|
||||||
"versionName" : "1.0.0",
|
"versionName" : "1.0.0",
|
||||||
"versionCode" : "100",
|
"versionCode" : "100",
|
||||||
@@ -40,7 +40,9 @@
|
|||||||
"<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>"
|
"<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"ios": {},
|
"ios" : {
|
||||||
|
"dSYMs" : false
|
||||||
|
},
|
||||||
"sdkConfigs" : {}
|
"sdkConfigs" : {}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,25 +1,26 @@
|
|||||||
<script setup>
|
<script setup>
|
||||||
import { ref, onMounted } from "vue";
|
import {onMounted, ref} from "vue";
|
||||||
import { onShow, onShareAppMessage, onShareTimeline } from "@dcloudio/uni-app";
|
import {onShareAppMessage, onShareTimeline, onShow} from "@dcloudio/uni-app";
|
||||||
import Container from "@/components/Container.vue";
|
import Container from "@/components/Container.vue";
|
||||||
import AppFooter from "@/components/AppFooter.vue";
|
import AppFooter from "@/components/AppFooter.vue";
|
||||||
import AppBackground from "@/components/AppBackground.vue";
|
|
||||||
import UserHeader from "@/components/UserHeader.vue";
|
import UserHeader from "@/components/UserHeader.vue";
|
||||||
import Signin from "@/components/Signin.vue";
|
import Signin from "@/components/Signin.vue";
|
||||||
import BubbleTip from "@/components/BubbleTip.vue";
|
import BubbleTip from "@/components/BubbleTip.vue";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
|
checkUserBindAPI,
|
||||||
getAppConfig,
|
getAppConfig,
|
||||||
getRankListAPI,
|
getDeviceBatteryAPI,
|
||||||
getHomeData,
|
getHomeData,
|
||||||
getMyDevicesAPI,
|
getMyDevicesAPI,
|
||||||
getDeviceBatteryAPI,
|
getRankListAPI,
|
||||||
|
silentLoginAPI,
|
||||||
} from "@/apis";
|
} from "@/apis";
|
||||||
import {topThreeColors} from "@/constants";
|
import {topThreeColors} from "@/constants";
|
||||||
import { canEenter } from "@/util";
|
|
||||||
|
|
||||||
import useStore from "@/store";
|
import useStore from "@/store";
|
||||||
import {storeToRefs} from "pinia";
|
import {storeToRefs} from "pinia";
|
||||||
|
|
||||||
const store = useStore();
|
const store = useStore();
|
||||||
const {
|
const {
|
||||||
updateConfig,
|
updateConfig,
|
||||||
@@ -27,6 +28,7 @@ const {
|
|||||||
updateDevice,
|
updateDevice,
|
||||||
updateRank,
|
updateRank,
|
||||||
getLvlName,
|
getLvlName,
|
||||||
|
getLvlNameByScore,
|
||||||
updateOnline,
|
updateOnline,
|
||||||
} = store;
|
} = store;
|
||||||
const {user, device, rankData, online, game} = storeToRefs(store);
|
const {user, device, rankData, online, game} = storeToRefs(store);
|
||||||
@@ -54,12 +56,36 @@ const toRankListPage = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
onShow(async () => {
|
onShow(async () => {
|
||||||
const token = uni.getStorageSync(
|
const env = uni.getAccountInfoSync().miniProgram.envVersion;
|
||||||
`${uni.getAccountInfoSync().miniProgram.envVersion}_token`
|
const token = uni.getStorageSync(`${env}_token`);
|
||||||
|
|
||||||
|
if (!user.value.id && !token) {
|
||||||
|
try {
|
||||||
|
const wxResult = await uni.login({provider: "weixin"});
|
||||||
|
const bindResult = await checkUserBindAPI(wxResult.code);
|
||||||
|
if (bindResult.binded) {
|
||||||
|
const newResult = await uni.login({provider: "weixin"});
|
||||||
|
const silentResult = await silentLoginAPI(newResult.code);
|
||||||
|
if (silentResult.user) updateUser(silentResult.user);
|
||||||
|
const devices = await getMyDevicesAPI();
|
||||||
|
if (devices.bindings && devices.bindings.length) {
|
||||||
|
updateDevice(
|
||||||
|
devices.bindings[0].deviceId,
|
||||||
|
devices.bindings[0].deviceName
|
||||||
);
|
);
|
||||||
|
const data = await getDeviceBatteryAPI();
|
||||||
|
updateOnline(data.online);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showModal.value = true;
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.log("检查绑定状态失败", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const promises = [getRankListAPI()];
|
const promises = [getRankListAPI()];
|
||||||
if (token) {
|
if (token || user.value.id) {
|
||||||
promises.push(getHomeData());
|
promises.push(getHomeData());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -221,8 +247,9 @@ onShareTimeline(() => {
|
|||||||
<view>
|
<view>
|
||||||
<text>段位</text>
|
<text>段位</text>
|
||||||
<text>{{
|
<text>{{
|
||||||
user.rankLvl ? getLvlName(user.rankLvl) : "暂无"
|
user.lvlName || "暂无"
|
||||||
}}</text>
|
}}
|
||||||
|
</text>
|
||||||
</view>
|
</view>
|
||||||
<view>
|
<view>
|
||||||
<text>赛季平均环数</text>
|
<text>赛季平均环数</text>
|
||||||
@@ -234,7 +261,8 @@ onShareTimeline(() => {
|
|||||||
user.avg_win
|
user.avg_win
|
||||||
? Number((user.avg_win * 100).toFixed(2)) + "%"
|
? Number((user.avg_win * 100).toFixed(2)) + "%"
|
||||||
: "暂无"
|
: "暂无"
|
||||||
}}</text>
|
}}
|
||||||
|
</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
@@ -363,6 +391,7 @@ onShareTimeline(() => {
|
|||||||
width: 32rpx;
|
width: 32rpx;
|
||||||
height: 32rpx;
|
height: 32rpx;
|
||||||
}
|
}
|
||||||
|
|
||||||
.player-avatar > view:first-child {
|
.player-avatar > view:first-child {
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
background: #777777;
|
background: #777777;
|
||||||
@@ -373,6 +402,7 @@ onShareTimeline(() => {
|
|||||||
height: 18px;
|
height: 18px;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
}
|
}
|
||||||
|
|
||||||
.player-avatar > image:last-child {
|
.player-avatar > image:last-child {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@@ -391,18 +421,22 @@ onShareTimeline(() => {
|
|||||||
margin-left: 2px;
|
margin-left: 2px;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
}
|
}
|
||||||
|
|
||||||
.my-data {
|
.my-data {
|
||||||
display: flex;
|
display: flex;
|
||||||
margin-top: 20px;
|
margin-top: 20px;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
}
|
}
|
||||||
|
|
||||||
.my-data > view:first-child {
|
.my-data > view:first-child {
|
||||||
width: 28%;
|
width: 28%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.my-data > view:first-child > image {
|
.my-data > view:first-child > image {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
transform: translateX(-8px);
|
transform: translateX(-8px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.my-data > view:nth-child(2) {
|
.my-data > view:nth-child(2) {
|
||||||
width: 68%;
|
width: 68%;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
@@ -410,9 +444,11 @@ onShareTimeline(() => {
|
|||||||
display: flex;
|
display: flex;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
}
|
}
|
||||||
|
|
||||||
.my-data > view:nth-child(2) > view:nth-child(2) {
|
.my-data > view:nth-child(2) > view:nth-child(2) {
|
||||||
width: 38%;
|
width: 38%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.my-data > view:nth-child(2) > view {
|
.my-data > view:nth-child(2) > view {
|
||||||
width: 28%;
|
width: 28%;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
@@ -422,11 +458,13 @@ onShareTimeline(() => {
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.my-data > view:nth-child(2) > view > text:last-child {
|
.my-data > view:nth-child(2) > view > text:last-child {
|
||||||
color: #fff;
|
color: #fff;
|
||||||
line-height: 25px;
|
line-height: 25px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
|
|
||||||
.top-theme {
|
.top-theme {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -436,6 +474,7 @@ onShareTimeline(() => {
|
|||||||
height: 60px;
|
height: 60px;
|
||||||
z-index: -1;
|
z-index: -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.top-theme > image {
|
.top-theme > image {
|
||||||
width: 300rpx;
|
width: 300rpx;
|
||||||
transform: translate(-4%, -14%);
|
transform: translate(-4%, -14%);
|
||||||
|
|||||||
@@ -3,24 +3,29 @@ import { ref, onMounted, onBeforeUnmount } from "vue";
|
|||||||
import { onLoad, onShow, onHide } from "@dcloudio/uni-app";
|
import { onLoad, onShow, onHide } from "@dcloudio/uni-app";
|
||||||
import Container from "@/components/Container.vue";
|
import Container from "@/components/Container.vue";
|
||||||
import Matching from "@/components/Matching.vue";
|
import Matching from "@/components/Matching.vue";
|
||||||
import RoundEndTip from "@/components/RoundEndTip.vue";
|
|
||||||
import TestDistance from "@/components/TestDistance.vue";
|
|
||||||
import { matchGameAPI } from "@/apis";
|
import { matchGameAPI } from "@/apis";
|
||||||
import { MESSAGETYPESV2 } from "@/constants";
|
import { MESSAGETYPESV2 } from "@/constants";
|
||||||
|
|
||||||
const gameType = ref(0);
|
const gameType = ref(0);
|
||||||
const teamSize = ref(0);
|
const teamSize = ref(0);
|
||||||
const onComplete = ref(null);
|
const onComplete = ref(null);
|
||||||
const matchSuccess = ref(false);
|
|
||||||
|
|
||||||
async function stopMatch() {
|
async function stopMatch() {
|
||||||
uni.$showHint(3);
|
uni.$showHint(3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function cancelMatch() {
|
||||||
|
if (gameType.value && teamSize.value) {
|
||||||
|
await matchGameAPI(false, gameType.value, teamSize.value);
|
||||||
|
}
|
||||||
|
uni.navigateBack()
|
||||||
|
}
|
||||||
|
|
||||||
async function onReceiveMessage(msg) {
|
async function onReceiveMessage(msg) {
|
||||||
if (msg.type === MESSAGETYPESV2.MatchSuccess) {
|
if (msg.type === MESSAGETYPESV2.MatchSuccess) {
|
||||||
matchSuccess.value = true;
|
onComplete.value = () => {}
|
||||||
onComplete.value = () => {
|
}
|
||||||
|
if (msg.type === MESSAGETYPESV2.AboutToStart) {
|
||||||
if (gameType.value == 1) {
|
if (gameType.value == 1) {
|
||||||
uni.redirectTo({
|
uni.redirectTo({
|
||||||
url: `/pages/team-battle?battleId=${msg.id}&gameMode=2`,
|
url: `/pages/team-battle?battleId=${msg.id}&gameMode=2`,
|
||||||
@@ -30,7 +35,6 @@ async function onReceiveMessage(msg) {
|
|||||||
url: `/pages/melee-battle?battleId=${msg.id}&gameMode=2`,
|
url: `/pages/melee-battle?battleId=${msg.id}&gameMode=2`,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
};
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -46,6 +50,7 @@ onMounted(() => {
|
|||||||
keepScreenOn: true,
|
keepScreenOn: true,
|
||||||
});
|
});
|
||||||
uni.$on("socket-inbox", onReceiveMessage);
|
uni.$on("socket-inbox", onReceiveMessage);
|
||||||
|
uni.$on("cancelMatching", cancelMatch);
|
||||||
});
|
});
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
onBeforeUnmount(() => {
|
||||||
@@ -53,9 +58,7 @@ onBeforeUnmount(() => {
|
|||||||
keepScreenOn: false,
|
keepScreenOn: false,
|
||||||
});
|
});
|
||||||
uni.$off("socket-inbox", onReceiveMessage);
|
uni.$off("socket-inbox", onReceiveMessage);
|
||||||
if (gameType.value && teamSize.value && !matchSuccess.value) {
|
uni.$off("cancelMatching", cancelMatch);
|
||||||
matchGameAPI(false, gameType.value, teamSize.value);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
onShow(async () => {
|
onShow(async () => {
|
||||||
@@ -65,9 +68,7 @@ onShow(async () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
onHide(() => {
|
onHide(() => {
|
||||||
if (gameType.value && teamSize.value && !matchSuccess.value) {
|
|
||||||
matchGameAPI(false, gameType.value, teamSize.value);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|||||||
@@ -28,21 +28,37 @@ const playersScores = ref([]);
|
|||||||
const halfTimeTip = ref(false);
|
const halfTimeTip = ref(false);
|
||||||
const halfRest = ref(false);
|
const halfRest = ref(false);
|
||||||
|
|
||||||
|
const navigateToResult = () => {
|
||||||
|
uni.redirectTo({ url: `/pages/battle-result?battleId=${battleId.value}` });
|
||||||
|
};
|
||||||
|
|
||||||
function recoverData(battleInfo, { force = false } = {}) {
|
function recoverData(battleInfo, { force = false } = {}) {
|
||||||
|
if (!battleInfo) return;
|
||||||
|
try {
|
||||||
if (battleInfo.way === 1) title.value = "好友约战 - 大乱斗";
|
if (battleInfo.way === 1) title.value = "好友约战 - 大乱斗";
|
||||||
if (battleInfo.way === 2) title.value = "排位赛 - 大乱斗";
|
if (battleInfo.way === 2) title.value = "排位赛 - 大乱斗";
|
||||||
players.value = battleInfo.teams[0].players;
|
|
||||||
|
// 优先使用接口数据,否则使用缓存
|
||||||
|
if (battleInfo.teams?.[0]?.players) {
|
||||||
|
players.value = [...battleInfo.teams[0].players];
|
||||||
|
} else {
|
||||||
|
// 大乱斗可能存的是 players 列表
|
||||||
|
// 这里的缓存逻辑根据 AboutToStart 消息结构可能不同,假设也是 teams[0]
|
||||||
|
// 如果是从 match-page 过来的,match-page 只存了 teams[1] 和 [2] 给对抗模式
|
||||||
|
// 大乱斗的匹配逻辑可能不同,暂时保持原样,只做安全保护
|
||||||
|
players.value = [];
|
||||||
|
}
|
||||||
|
|
||||||
start.value = battleInfo.status !== 0;
|
start.value = battleInfo.status !== 0;
|
||||||
|
|
||||||
if (battleInfo.status === 0) {
|
if (battleInfo.status === 0) {
|
||||||
const readyRemain = (Date.now() - battleInfo.createTime) / 1000;
|
const readyRemain = (Date.now() - (battleInfo.createTime || Date.now())) / 1000;
|
||||||
console.log(`对局已进行${readyRemain}秒`);
|
|
||||||
if (readyRemain > 0 && readyRemain < 15) {
|
if (readyRemain > 0 && readyRemain < 15) {
|
||||||
setTimeout(() => {
|
setTimeout(() => uni.$emit("update-timer", 15 - readyRemain - 0.2), 200);
|
||||||
uni.$emit("update-timer", 15 - readyRemain - 0.2);
|
|
||||||
}, 200);
|
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
tips.value =
|
tips.value =
|
||||||
(battleInfo.rounds.length !== 2 ? "上" : "下") + "半场:请先射6箭";
|
(battleInfo.rounds.length !== 2 ? "上" : "下") + "半场:请先射6箭";
|
||||||
playersScores.value = battleInfo.rounds.map((r) => ({ ...r.shoots }));
|
playersScores.value = battleInfo.rounds.map((r) => ({ ...r.shoots }));
|
||||||
@@ -69,7 +85,7 @@ function recoverData(battleInfo, { force = false } = {}) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (force) {
|
if (force) {
|
||||||
const remain = (Date.now() - battleInfo.current.startTime) / 1000;
|
const remain = (Date.now() - (battleInfo.current?.startTime || Date.now())) / 1000;
|
||||||
console.log(`当前轮已进行${remain}秒`);
|
console.log(`当前轮已进行${remain}秒`);
|
||||||
if (remain > 0 && remain < 90) {
|
if (remain > 0 && remain < 90) {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
@@ -77,6 +93,9 @@ function recoverData(battleInfo, { force = false } = {}) {
|
|||||||
}, 200);
|
}, 200);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.error("recoverData error:", err);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
onLoad(async (options) => {
|
onLoad(async (options) => {
|
||||||
@@ -127,6 +146,7 @@ onBeforeUnmount(() => {
|
|||||||
onShow(async () => {
|
onShow(async () => {
|
||||||
if (battleId.value) {
|
if (battleId.value) {
|
||||||
const result = await getBattleAPI(battleId.value);
|
const result = await getBattleAPI(battleId.value);
|
||||||
|
if (!result) return;
|
||||||
if (result.status === 2) {
|
if (result.status === 2) {
|
||||||
uni.showToast({
|
uni.showToast({
|
||||||
title: "比赛已结束",
|
title: "比赛已结束",
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import {
|
|||||||
bindDeviceAPI,
|
bindDeviceAPI,
|
||||||
getMyDevicesAPI,
|
getMyDevicesAPI,
|
||||||
unbindDeviceAPI,
|
unbindDeviceAPI,
|
||||||
laserAimAPI,
|
laserAimAPI, bindDeviceAPIV2,
|
||||||
} from "@/apis";
|
} from "@/apis";
|
||||||
import useStore from "@/store";
|
import useStore from "@/store";
|
||||||
import { storeToRefs } from "pinia";
|
import { storeToRefs } from "pinia";
|
||||||
@@ -20,6 +20,7 @@ const { updateDevice } = store;
|
|||||||
const { user, device } = storeToRefs(store);
|
const { user, device } = storeToRefs(store);
|
||||||
const justBind = ref(false);
|
const justBind = ref(false);
|
||||||
const calibration = ref(false);
|
const calibration = ref(false);
|
||||||
|
const token = ref(null);
|
||||||
|
|
||||||
// 扫描二维码方法
|
// 扫描二维码方法
|
||||||
const handleScan = () => {
|
const handleScan = () => {
|
||||||
@@ -30,13 +31,14 @@ const handleScan = () => {
|
|||||||
scanType: ["qrCode"],
|
scanType: ["qrCode"],
|
||||||
success: async (res) => {
|
success: async (res) => {
|
||||||
try {
|
try {
|
||||||
const base64Decode = (str) => {
|
// const base64Decode = (str) => {
|
||||||
// 将 base64 转换为 utf8 字符串
|
// // 将 base64 转换为 utf8 字符串
|
||||||
const bytes = wx.base64ToArrayBuffer(str);
|
// const bytes = wx.base64ToArrayBuffer(str);
|
||||||
return String.fromCharCode.apply(null, new Uint8Array(bytes));
|
// return String.fromCharCode.apply(null, new Uint8Array(bytes));
|
||||||
};
|
// };
|
||||||
|
//
|
||||||
addDevice.value = JSON.parse(base64Decode(res.result));
|
// addDevice.value = JSON.parse(base64Decode(res.result));
|
||||||
|
token.value = res.result;
|
||||||
confirmBindTip.value = true;
|
confirmBindTip.value = true;
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
uni.showToast({
|
uni.showToast({
|
||||||
@@ -57,8 +59,8 @@ const handleScan = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const confirmBind = async () => {
|
const confirmBind = async () => {
|
||||||
if (!justBind.value && addDevice.value.id) {
|
if (!justBind.value && token.value) {
|
||||||
const result = await bindDeviceAPI(addDevice.value);
|
const result = await bindDeviceAPIV2(token.value);
|
||||||
confirmBindTip.value = false;
|
confirmBindTip.value = false;
|
||||||
if (result.binded) {
|
if (result.binded) {
|
||||||
return uni.showToast({
|
return uni.showToast({
|
||||||
@@ -66,7 +68,7 @@ const confirmBind = async () => {
|
|||||||
icon: "none",
|
icon: "none",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
updateDevice(addDevice.value.id, addDevice.value.name);
|
updateDevice(result.deviceId, result.name);
|
||||||
justBind.value = true;
|
justBind.value = true;
|
||||||
uni.showToast({
|
uni.showToast({
|
||||||
title: "绑定成功",
|
title: "绑定成功",
|
||||||
|
|||||||
@@ -110,7 +110,7 @@ onShow(async () => {
|
|||||||
const result = await getHomeData();
|
const result = await getHomeData();
|
||||||
rankData.value = result;
|
rankData.value = result;
|
||||||
handleSelect(selectedIndex.value);
|
handleSelect(selectedIndex.value);
|
||||||
seasonData.value = result.seasonList;
|
seasonData.value = result.seasonList || [];
|
||||||
if (seasonData.value[0]) {
|
if (seasonData.value[0]) {
|
||||||
seasonName.value = seasonData.value[0].seasonName;
|
seasonName.value = seasonData.value[0].seasonName;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,16 +35,32 @@ const redPoints = ref(0);
|
|||||||
const bluePoints = ref(0);
|
const bluePoints = ref(0);
|
||||||
const showRoundTip = ref(false);
|
const showRoundTip = ref(false);
|
||||||
const isFinalShoot = ref(false);
|
const isFinalShoot = ref(false);
|
||||||
|
const matchStatus = ref(undefined);
|
||||||
|
|
||||||
const recoverData = (battleInfo, { force = false, arrowOnly = false } = {}) => {
|
const recoverData = (battleInfo, { force = false, arrowOnly = false } = {}) => {
|
||||||
try {
|
try {
|
||||||
battleId.value = battleInfo.matchId;
|
battleId.value = battleInfo.matchId;
|
||||||
blueTeam.value = battleInfo.teams[1].players || [];
|
|
||||||
redTeam.value = battleInfo.teams[2].players || [];
|
// 优先使用接口返回的队伍数据,如果没有则尝试从缓存读取(应对匹配刚完成接口未就绪的情况)
|
||||||
|
const t1 = battleInfo.teams?.[1] || {};
|
||||||
|
const t2 = battleInfo.teams?.[2] || {};
|
||||||
|
|
||||||
|
if (t1.players) blueTeam.value = [...t1.players];
|
||||||
|
else {
|
||||||
|
const cached = uni.getStorageSync("blue-team");
|
||||||
|
if (cached && cached.length) blueTeam.value = cached;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (t2.players) redTeam.value = [...t2.players];
|
||||||
|
else {
|
||||||
|
const cached = uni.getStorageSync("red-team");
|
||||||
|
if (cached && cached.length) redTeam.value = cached;
|
||||||
|
}
|
||||||
|
|
||||||
start.value = battleInfo.status !== 0;
|
start.value = battleInfo.status !== 0;
|
||||||
|
|
||||||
if (battleInfo.status === 0) {
|
if (battleInfo.status === 0) {
|
||||||
const readyRemain = (Date.now() - battleInfo.createTime) / 1000;
|
const readyRemain = (Date.now() - (battleInfo.createTime || Date.now())) / 1000;
|
||||||
console.log(`对局已进行${readyRemain}秒`);
|
|
||||||
if (readyRemain > 0 && readyRemain < 15) {
|
if (readyRemain > 0 && readyRemain < 15) {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
uni.$emit("update-timer", 15 - readyRemain - 0.2);
|
uni.$emit("update-timer", 15 - readyRemain - 0.2);
|
||||||
@@ -95,6 +111,15 @@ const recoverData = (battleInfo, { force = false, arrowOnly = false } = {}) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function onBattleEnd(){
|
||||||
|
showRoundTip.value = false
|
||||||
|
console.log("onBattleEnd ended...")
|
||||||
|
if (matchStatus.value === 2) {
|
||||||
|
uni.redirectTo({
|
||||||
|
url: `/pages/battle-result?battleId=${battleId.value}`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
async function onReceiveMessage(msg) {
|
async function onReceiveMessage(msg) {
|
||||||
if (Array.isArray(msg)) return;
|
if (Array.isArray(msg)) return;
|
||||||
if (msg.type === MESSAGETYPESV2.BattleStart) {
|
if (msg.type === MESSAGETYPESV2.BattleStart) {
|
||||||
@@ -118,6 +143,7 @@ async function onReceiveMessage(msg) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (msg.type === MESSAGETYPESV2.BattleEnd) {
|
} else if (msg.type === MESSAGETYPESV2.BattleEnd) {
|
||||||
|
matchStatus.value = msg.status;
|
||||||
if (msg.status === 4) {
|
if (msg.status === 4) {
|
||||||
showRoundTip.value = true;
|
showRoundTip.value = true;
|
||||||
currentBluePoint.value = 0;
|
currentBluePoint.value = 0;
|
||||||
@@ -125,13 +151,7 @@ async function onReceiveMessage(msg) {
|
|||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
uni.navigateBack();
|
uni.navigateBack();
|
||||||
}, 2000);
|
}, 2000);
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
setTimeout(() => {
|
|
||||||
uni.redirectTo({
|
|
||||||
url: "/pages/battle-result?battleId=" + msg.matchId,
|
|
||||||
});
|
|
||||||
}, 1000);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -162,6 +182,7 @@ const refreshTimer = ref(null);
|
|||||||
onShow(async () => {
|
onShow(async () => {
|
||||||
if (battleId.value) {
|
if (battleId.value) {
|
||||||
const result = await getBattleAPI(battleId.value);
|
const result = await getBattleAPI(battleId.value);
|
||||||
|
if (!result) return;
|
||||||
if (result.status === 2) {
|
if (result.status === 2) {
|
||||||
uni.showToast({
|
uni.showToast({
|
||||||
title: "比赛已结束",
|
title: "比赛已结束",
|
||||||
@@ -227,7 +248,7 @@ onShow(async () => {
|
|||||||
:roundData="
|
:roundData="
|
||||||
roundResults[currentRound - 1] ? roundResults[currentRound - 1] : []
|
roundResults[currentRound - 1] ? roundResults[currentRound - 1] : []
|
||||||
"
|
"
|
||||||
:onAutoClose="() => (showRoundTip = false)"
|
:onAutoClose="onBattleEnd"
|
||||||
/>
|
/>
|
||||||
</ScreenHint>
|
</ScreenHint>
|
||||||
</view>
|
</view>
|
||||||
|
|||||||
40
src/store.js
40
src/store.js
@@ -22,6 +22,19 @@ const getLvlName = (rankLvl, rankList = []) => {
|
|||||||
return lvlName;
|
return lvlName;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getLvlNameByScore = (score, rankList = []) => {
|
||||||
|
if (!rankList) return;
|
||||||
|
let lvlName = "";
|
||||||
|
rankList.some((r, index) => {
|
||||||
|
if (r.upgrade_scores && r.upgrade_scores < score) {
|
||||||
|
lvlName = rankList[index].name;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
return lvlName;
|
||||||
|
};
|
||||||
|
|
||||||
const getLvlImage = (rankLvl, rankList = []) => {
|
const getLvlImage = (rankLvl, rankList = []) => {
|
||||||
if (!rankList) return;
|
if (!rankList) return;
|
||||||
let lvlImage = "";
|
let lvlImage = "";
|
||||||
@@ -35,6 +48,19 @@ const getLvlImage = (rankLvl, rankList = []) => {
|
|||||||
return lvlImage;
|
return lvlImage;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getLvlImageByScore = (score, rankList = []) => {
|
||||||
|
if (!rankList) return;
|
||||||
|
let lvlImage = "";
|
||||||
|
rankList.some((r, index) => {
|
||||||
|
if (r.upgrade_scores && r.upgrade_scores < score) {
|
||||||
|
lvlImage = rankList[index].icoin;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
return lvlImage;
|
||||||
|
};
|
||||||
|
|
||||||
// 定义游戏相关的 store
|
// 定义游戏相关的 store
|
||||||
export default defineStore("store", {
|
export default defineStore("store", {
|
||||||
// 状态
|
// 状态
|
||||||
@@ -65,6 +91,9 @@ export default defineStore("store", {
|
|||||||
|
|
||||||
// 方法
|
// 方法
|
||||||
actions: {
|
actions: {
|
||||||
|
getLvlNameByScore(score) {
|
||||||
|
return getLvlNameByScore(score, this.config.randInfos);
|
||||||
|
},
|
||||||
getLvlName(rankLvl) {
|
getLvlName(rankLvl) {
|
||||||
return getLvlName(rankLvl, this.config.randInfos);
|
return getLvlName(rankLvl, this.config.randInfos);
|
||||||
},
|
},
|
||||||
@@ -79,14 +108,9 @@ export default defineStore("store", {
|
|||||||
},
|
},
|
||||||
async updateUser(user = {}) {
|
async updateUser(user = {}) {
|
||||||
this.user = { ...defaultUser, ...user };
|
this.user = { ...defaultUser, ...user };
|
||||||
if (this.user.rankLvl !== undefined) {
|
this.user.lvlName = getLvlNameByScore(this.user.scores, this.config.randInfos)
|
||||||
this.user.lvlName = getLvlName(
|
this.user.lvlImage = getLvlImageByScore(
|
||||||
this.user.rankLvl,
|
this.user.scores,
|
||||||
this.config.randInfos
|
|
||||||
);
|
|
||||||
}
|
|
||||||
this.user.lvlImage = getLvlImage(
|
|
||||||
this.user.rankLvl,
|
|
||||||
this.config.randInfos
|
this.config.randInfos
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
|||||||
Reference in New Issue
Block a user