41 Commits

Author SHA1 Message Date
e830a63deb Merge branch 'new-race-mode' into test 2026-05-19 15:16:07 +08:00
a885ce0dce Merge branch 'new-race-mode' into test 2026-05-19 14:51:13 +08:00
de7b44aa34 Merge branch 'new-race-mode' into test 2026-05-19 13:36:18 +08:00
7cbdfdceab Merge branch 'new-race-mode' into test 2026-05-19 10:03:18 +08:00
1ade861aae pref: 删除新年主题 2026-05-19 09:45:33 +08:00
3293356668 pref: 删除静默登录 2026-05-18 11:14:54 +08:00
fefdb3527d Merge branch 'new-race-mode' into test 2026-05-18 09:37:00 +08:00
076769c4d8 Merge branch 'feat-zhangyi' into test 2026-05-15 09:58:52 +08:00
7d503c09e0 update:优化赛季切换 2026-05-15 09:58:25 +08:00
15a3e4120d Merge branch 'new-race-mode' into test 2026-05-14 18:22:19 +08:00
192db06ac5 Merge branch 'new-race-mode' into test 2026-05-14 17:58:17 +08:00
162a6820e9 Merge branch 'new-race-mode' into test 2026-05-14 17:15:07 +08:00
852b2ce799 Merge branch 'new-race-mode' into test 2026-05-14 16:26:49 +08:00
2b14274453 Merge branch 'new-race-mode' into test 2026-05-14 15:43:50 +08:00
33d28c6a45 Merge branch 'new-race-mode' into test 2026-05-14 14:34:10 +08:00
301fc4ee42 Merge branch 'new-race-mode' into test 2026-05-14 14:23:25 +08:00
6596ea977d Merge branch 'new-race-mode' into test 2026-05-14 13:34:41 +08:00
f272df912b Merge branch 'new-race-mode' into test 2026-05-14 11:00:36 +08:00
cbb7bce746 Merge branch 'new-race-mode' into test 2026-05-14 10:28:59 +08:00
a5c6afe647 合并代码并解决冲突 2026-05-14 10:07:26 +08:00
0b93d99688 合并解决冲突 2026-05-13 18:32:56 +08:00
3bc2e6c14b Merge branch 'new-race-mode' into test 2026-05-13 17:19:04 +08:00
ff64fab323 Merge branch 'new-race-mode' into test 2026-05-13 16:34:07 +08:00
43123df18f Merge branch 'new-race-mode' into test 2026-05-13 15:15:04 +08:00
860b01d5eb Merge branch 'new-race-mode' into test 2026-05-13 14:58:45 +08:00
9c554a9366 Merge branch 'new-race-mode' into test 2026-05-13 13:42:51 +08:00
8d918b008e fix:解决合并冲突 2026-05-12 17:40:03 +08:00
a66d8a03c6 Merge branch 'new-race-mode' into test 2026-05-12 17:36:57 +08:00
b168fd54a4 fix:比赛页面新增设备已离线弹窗提示 2026-05-12 17:00:47 +08:00
da023c60f5 Merge branch 'new-race-mode' into test 2026-05-12 16:11:46 +08:00
cb4027418d Merge branch 'new-race-mode' into test 2026-05-12 09:20:55 +08:00
96cc69f041 Merge branch 'new-race-mode' into test 2026-05-11 15:17:06 +08:00
860cdbe332 update:优化多端登录 2026-05-11 14:11:52 +08:00
8b9c862b96 update:优化多端登录 2026-05-11 14:11:25 +08:00
19391808ef Merge branch 'feat-zhangyi' 2026-05-11 09:23:19 +08:00
9cc6ef4b88 Merge branch 'new-race-mode' into test 2026-05-09 17:41:54 +08:00
fc14489ab3 Merge branch 'new-race-mode' into test 2026-05-09 16:01:21 +08:00
aba2da56d7 Merge branch 'feat-zhangyi' into test 2026-05-09 15:25:52 +08:00
24314e5ec8 update:36箭倒计时优化 2026-05-09 15:25:37 +08:00
4dcfdeda68 feat: 添加分支管理说明 2026-05-07 21:04:10 +08:00
a6becf67ff fix: 个人练习报环完成之后再弹练习结果弹窗 2026-05-07 18:30:46 +08:00
6 changed files with 166 additions and 82 deletions

View File

@@ -21,7 +21,10 @@
} = storeToRefs(store); } = storeToRefs(store);
const { const {
updateUser, updateUser,
updateOnline updateOnline,
updateDevice,
updateGame,
updateRoomNumber
} = store; } = store;
watch( watch(
@@ -46,6 +49,22 @@
updateUser(value); updateUser(value);
} }
function onSessionKickedOut() {
uni.removeStorageSync(
`${uni.getAccountInfoSync().miniProgram.envVersion}_token`
);
updateUser();
updateDevice("", "");
updateOnline(false);
updateGame(false, "");
updateRoomNumber("");
uni.showModal({
title: "提示",
content: "账号已在其他设备登录",
showCancel: false,
});
}
async function emitUpdateOnline() { async function emitUpdateOnline() {
const data = await getDeviceBatteryAPI(); const data = await getDeviceBatteryAPI();
updateOnline(data.online); updateOnline(data.online);
@@ -65,6 +84,7 @@
onShow(() => { onShow(() => {
uni.$on("update-user", emitUpdateUser); uni.$on("update-user", emitUpdateUser);
uni.$on("update-online", emitUpdateOnline); uni.$on("update-online", emitUpdateOnline);
uni.$on("session-kicked-out", onSessionKickedOut);
const token = uni.getStorageSync( const token = uni.getStorageSync(
`${uni.getAccountInfoSync().miniProgram.envVersion}_token` `${uni.getAccountInfoSync().miniProgram.envVersion}_token`
); );
@@ -77,6 +97,7 @@
onHide(() => { onHide(() => {
uni.$off("update-user", emitUpdateUser); uni.$off("update-user", emitUpdateUser);
uni.$off("update-online", emitUpdateOnline); uni.$off("update-online", emitUpdateOnline);
uni.$off("session-kicked-out", onSessionKickedOut);
websocket.closeWebSocket(); websocket.closeWebSocket();
}); });
</script> </script>

View File

@@ -46,6 +46,8 @@ function request(method, url, data = {}) {
`${uni.getAccountInfoSync().miniProgram.envVersion}_token` `${uni.getAccountInfoSync().miniProgram.envVersion}_token`
); );
uni.$emit("update-user"); uni.$emit("update-user");
reject({ type: "AUTH_INVALID", message });
return;
} }
if (message === "ROOM_FULL") { if (message === "ROOM_FULL") {
resolve({full: true}); resolve({full: true});

View File

@@ -68,28 +68,29 @@ onShow(async () => {
const token = uni.getStorageSync(`${env}_token`); const token = uni.getStorageSync(`${env}_token`);
if (!user.value.id && !token) { if (!user.value.id && !token) {
try { // showModal.value = true;
const wxResult = await uni.login({provider: "weixin"}); // try {
const bindResult = await checkUserBindAPI(wxResult.code); // const wxResult = await uni.login({provider: "weixin"});
if (bindResult.binded) { // const bindResult = await checkUserBindAPI(wxResult.code);
const newResult = await uni.login({provider: "weixin"}); // if (bindResult.binded) {
const silentResult = await silentLoginAPI(newResult.code); // const newResult = await uni.login({provider: "weixin"});
if (silentResult.user) updateUser(silentResult.user); // const silentResult = await silentLoginAPI(newResult.code);
const devices = await getMyDevicesAPI(); // if (silentResult.user) updateUser(silentResult.user);
if (devices.bindings && devices.bindings.length) { // const devices = await getMyDevicesAPI();
updateDevice( // if (devices.bindings && devices.bindings.length) {
devices.bindings[0].deviceId, // updateDevice(
devices.bindings[0].deviceName // devices.bindings[0].deviceId,
); // devices.bindings[0].deviceName
const data = await getDeviceBatteryAPI(); // );
updateOnline(data.online); // const data = await getDeviceBatteryAPI();
} // updateOnline(data.online);
} else { // }
showModal.value = true; // } else {
} // showModal.value = true;
} catch (e) { // }
console.log("检查绑定状态失败", e); // } catch (e) {
} // console.log("检查绑定状态失败", e);
// }
} }
const promises = [getScoreRankList(undefined, 1, 10)]; const promises = [getScoreRankList(undefined, 1, 10)];
@@ -475,7 +476,7 @@ onShareTimeline(() => {
.top-theme { .top-theme {
position: absolute; position: absolute;
display: flex; display: none;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
width: 100%; width: 100%;

View File

@@ -89,7 +89,7 @@ async function onComplete() {
practiseResult.value = {}; practiseResult.value = {};
start.value = false; start.value = false;
scores.value = []; scores.value = [];
const result = await createPractiseAPI(total, 360); const result = await createPractiseAPI(total, 3600);
if (result) practiseId.value = result.id; if (result) practiseId.value = result.id;
} }
} }
@@ -105,7 +105,7 @@ onMounted(async () => {
}); });
uni.$on("socket-inbox", onReceiveMessage); uni.$on("socket-inbox", onReceiveMessage);
uni.$on("share-image", onClickShare); uni.$on("share-image", onClickShare);
const result = await createPractiseAPI(total, 360, targetType.value); const result = await createPractiseAPI(total, 3600, targetType.value);
if (result) practiseId.value = result.id; if (result) practiseId.value = result.id;
}); });
@@ -132,7 +132,7 @@ onBeforeUnmount(() => {
<ShootProgress <ShootProgress
:tips="`请连续射${total}支箭`" :tips="`请连续射${total}支箭`"
:start="start" :start="start"
:total="360" :total="3600"
:onStop="onOver" :onStop="onOver"
/> />
<view class="user-row"> <view class="user-row">

View File

@@ -323,6 +323,7 @@ onShow(async () => {
> >
<text>{{ seasonName }}</text> <text>{{ seasonName }}</text>
<image <image
class="triangle-icon"
v-show="seasonData.length > 1" v-show="seasonData.length > 1"
src="../static/rank/triangle.png" src="../static/rank/triangle.png"
mode="widthFix" mode="widthFix"
@@ -592,6 +593,7 @@ onShow(async () => {
.ranking-season > image { .ranking-season > image {
width: 12px; width: 12px;
height: 12px; height: 12px;
transform: rotateX(180deg);
} }
.ranking-season > text { .ranking-season > text {
@@ -807,18 +809,18 @@ onShow(async () => {
} }
.season-list { .season-list {
background-color: #000c; background-color: rgba(0, 0, 0, 0.8);
border-radius: 15px; border-radius: 15px;
color: #fff; color: #fff;
font-size: 12px; font-size: 12px;
padding: 5px 0; padding: 12px 0;
position: absolute; position: absolute;
width: 220rpx; width: 220rpx;
height: auto; height: auto;
max-height: 400rpx; max-height: 400rpx;
overflow: hidden; overflow: hidden;
overflow-y: auto; overflow-y: auto;
top: -44rpx; top: -42rpx;
right: -30rpx; right: -30rpx;
letter-spacing: 2px; letter-spacing: 2px;
z-index: 10; z-index: 10;
@@ -829,7 +831,7 @@ onShow(async () => {
display: flex; display: flex;
align-items: center; align-items: center;
word-break: keep-all; word-break: keep-all;
padding: 20rpx 0; padding: 20rpx 10rpx;
} }
.season-list > view > text { .season-list > view > text {

View File

@@ -1,12 +1,22 @@
import { MESSAGETYPES, getMessageTypeName } from "@/constants"; import { MESSAGETYPES, getMessageTypeName } from "@/constants";
import { getUserGameState } from "@/apis";
let socket = null; let socket = null;
let heartbeatInterval = null; let heartbeatInterval = null;
let reconnectTimer = null; let reconnectTimer = null;
let manualClose = false;
let checkingSession = false;
let kickedOut = false;
let isConnecting = false;
/**
* 建立 WebSocket 连接
*/
function createWebSocket(token, onMessage) { function createWebSocket(token, onMessage) {
if (!token) return;
if (kickedOut) kickedOut = false;
if (socket || isConnecting) return;
manualClose = false;
isConnecting = true;
let url = "wss://api.shelingxingqiu.com/socket"; let url = "wss://api.shelingxingqiu.com/socket";
try { try {
const accountInfo = uni.getAccountInfoSync(); const accountInfo = uni.getAccountInfoSync();
@@ -20,42 +30,65 @@ function createWebSocket(token, onMessage) {
case "trial": // 体验版 case "trial": // 体验版
url = "wss://apitest.shelingxingqiu.com/socket"; url = "wss://apitest.shelingxingqiu.com/socket";
break; break;
case "release": // 正式版 case "trial":
url = "wss://apitest.shelingxingqiu.com/socket";
break;
case "release":
url = "wss://api.shelingxingqiu.com/socket"; url = "wss://api.shelingxingqiu.com/socket";
break; break;
default: default:
// 保持默认值
break; break;
} }
} catch (e) { } catch (e) {
console.error("获取环境信息失败,使用默认正式环境", e); console.error("获取 WebSocket 环境信息失败,使用默认正式环境", e);
} }
url += `?authorization=${token}`; url += `?authorization=${token}`;
socket = uni.connectSocket({ const socketTask = uni.connectSocket({
url, url,
success: () => { success: () => {
console.log("websocket 连接成功"); console.log("WebSocket 已发起连接");
// 启动心跳
startHeartbeat(onMessage);
}, },
fail: () => { fail: (err) => {
if (socket !== socketTask) return;
console.error("WebSocket 连接失败", err);
socket = null;
isConnecting = false;
reconnect(onMessage); reconnect(onMessage);
}, },
}); });
// 接收消息 socket = socketTask;
uni.onSocketMessage((res) => {
socketTask.onOpen(() => {
if (socket !== socketTask) return;
console.log("WebSocket 连接成功");
isConnecting = false;
startHeartbeat(onMessage);
});
socketTask.onMessage((res) => {
if (socket !== socketTask) return;
const { data, event } = JSON.parse(res.data); const { data, event } = JSON.parse(res.data);
if (event === "pong") return; if (event === "pong") return;
if (data.type) { if (data.type) {
console.log("收到消息:", getMessageTypeName(data.type), data.data); console.log(
"收到 WebSocket 消息",
getMessageTypeName(data.type),
data.data
);
if (onMessage) onMessage({ ...(data.data || {}), type: data.type }); if (onMessage) onMessage({ ...(data.data || {}), type: data.type });
return; return;
} }
if (onMessage && data.updates) onMessage(data.updates); if (onMessage && data.updates) onMessage(data.updates);
const msg = data.updates[0]; const msg = data.updates[0];
if (msg) { if (msg) {
console.log("收到消息:", getMessageTypeName(msg.constructor), msg); console.log(
"收到 WebSocket 更新",
getMessageTypeName(msg.constructor),
msg
);
if (msg.constructor === MESSAGETYPES.RankUpdate) { if (msg.constructor === MESSAGETYPES.RankUpdate) {
uni.setStorageSync("latestRank", msg.lvl); uni.setStorageSync("latestRank", msg.lvl);
} else if (msg.constructor === MESSAGETYPES.LvlUpdate) { } else if (msg.constructor === MESSAGETYPES.LvlUpdate) {
@@ -68,84 +101,109 @@ function createWebSocket(token, onMessage) {
} }
}); });
// 错误处理 socketTask.onError((err) => {
uni.onSocketError((err) => { if (socket !== socketTask) return;
console.error("WebSocket 错误", err); console.error("WebSocket 错误", err);
reconnect(onMessage);
}); });
uni.onSocketClose((result) => { socketTask.onClose(async (result) => {
if (socket !== socketTask) return;
console.log("WebSocket 已关闭", result); console.log("WebSocket 已关闭", result);
stopHeartbeat(); stopHeartbeat();
reconnect(onMessage); socket = null;
isConnecting = false;
if (manualClose || kickedOut) return;
await handleUnexpectedClose(onMessage);
}); });
} }
/** async function handleUnexpectedClose(onMessage) {
* 重连机制 if (checkingSession || manualClose || kickedOut) return;
*/
function reconnect(onMessage) {
reconnectTimer && clearTimeout(reconnectTimer);
closeWebSocket(); // 确保关闭旧连接
const token = uni.getStorageSync( const token = uni.getStorageSync(
`${uni.getAccountInfoSync().miniProgram.envVersion}_token` `${uni.getAccountInfoSync().miniProgram.envVersion}_token`
); );
if (!token) return; if (!token) return;
checkingSession = true;
try {
await getUserGameState();
if (!manualClose && !kickedOut) reconnect(onMessage);
} catch (err) {
if (err?.type === "AUTH_INVALID") {
kickedOut = true;
manualClose = true;
reconnectTimer && clearTimeout(reconnectTimer);
uni.$emit("session-kicked-out");
return;
}
if (!manualClose && !kickedOut) reconnect(onMessage);
} finally {
checkingSession = false;
}
}
function reconnect(onMessage) {
reconnectTimer && clearTimeout(reconnectTimer);
const token = uni.getStorageSync(
`${uni.getAccountInfoSync().miniProgram.envVersion}_token`
);
if (!token || manualClose || kickedOut || socket || isConnecting) return;
reconnectTimer = setTimeout(() => { reconnectTimer = setTimeout(() => {
console.log("reconnecting..."); if (manualClose || kickedOut || socket || isConnecting) return;
console.log("WebSocket 正在重连...");
createWebSocket(token, onMessage); createWebSocket(token, onMessage);
}, 1000); }, 1000);
} }
function closeWebSocket() { function closeWebSocket(isManual = true) {
if (socket) { manualClose = isManual;
reconnectTimer && clearTimeout(reconnectTimer); reconnectTimer && clearTimeout(reconnectTimer);
stopHeartbeat(); stopHeartbeat();
isConnecting = false;
if (socket) {
const currentSocket = socket;
socket = null;
try { try {
socket.close(); currentSocket.close();
} catch (err) { } catch (err) {
console.error("关闭WebSocket连接失败", err); console.error("关闭 WebSocket 失败", err);
} }
socket = null; // 清除socket引用
} }
} }
function sendHeartbeat(onMessage) { function sendHeartbeat(onMessage) {
uni.sendSocketMessage({ if (!socket) return;
const currentSocket = socket;
currentSocket.send({
data: JSON.stringify({ event: "ping", data: {} }), data: JSON.stringify({ event: "ping", data: {} }),
success: () => { success: () => {},
// console.log("发送心跳成功");
},
fail: (err) => { fail: (err) => {
console.error("发送心跳失败", err); if (socket !== currentSocket) return;
console.error("心跳发送失败", err);
stopHeartbeat(); stopHeartbeat();
closeWebSocket(); // 关闭失效的连接 closeWebSocket(false);
reconnect(onMessage); // 触发重连 reconnect(onMessage);
}, },
}); });
} }
/**
* 启动心跳
*/
function startHeartbeat(onMessage) { function startHeartbeat(onMessage) {
stopHeartbeat(); // 防止重复启动 stopHeartbeat();
heartbeatInterval = setInterval(() => { heartbeatInterval = setInterval(() => {
if (socket && socket.readyState === 1) { if (socket) {
// 检查连接状态
sendHeartbeat(onMessage); sendHeartbeat(onMessage);
} }
}, 10000); }, 10000);
} }
/**
* 停止心跳
*/
function stopHeartbeat() { function stopHeartbeat() {
if (heartbeatInterval) { if (heartbeatInterval) {
clearInterval(heartbeatInterval); clearInterval(heartbeatInterval);