mirror of
https://github.com/genxium/DelayNoMore
synced 2025-01-14 23:11:25 +00:00
281 lines
11 KiB
JavaScript
281 lines
11 KiB
JavaScript
const RingBuffer = require('./RingBuffer');
|
|
|
|
window.UPSYNC_MSG_ACT_HB_PING = 1;
|
|
window.UPSYNC_MSG_ACT_PLAYER_CMD = 2;
|
|
window.UPSYNC_MSG_ACT_PLAYER_COLLIDER_ACK = 3;
|
|
|
|
window.DOWNSYNC_MSG_ACT_PLAYER_ADDED_AND_ACKED = -98;
|
|
window.DOWNSYNC_MSG_ACT_PLAYER_READDED_AND_ACKED = -97;
|
|
window.DOWNSYNC_MSG_ACT_BATTLE_READY_TO_START = -1;
|
|
window.DOWNSYNC_MSG_ACT_BATTLE_START = 0;
|
|
window.DOWNSYNC_MSG_ACT_HB_REQ = 1;
|
|
window.DOWNSYNC_MSG_ACT_INPUT_BATCH = 2;
|
|
window.DOWNSYNC_MSG_ACT_BATTLE_STOPPED = 3;
|
|
window.DOWNSYNC_MSG_ACT_FORCED_RESYNC = 4;
|
|
|
|
|
|
window.sendSafely = function(msgStr) {
|
|
/**
|
|
* - "If the data can't be sent (for example, because it needs to be buffered but the buffer is full), the socket is closed automatically."
|
|
*
|
|
* from https://developer.mozilla.org/en-US/docs/Web/API/WebSocket/send.
|
|
*/
|
|
if (null == window.clientSession || window.clientSession.readyState != WebSocket.OPEN) return false;
|
|
window.clientSession.send(msgStr);
|
|
}
|
|
|
|
window.sendUint8AsBase64Safely = function(msgUint8Arr) {
|
|
if (null == window.clientSession || window.clientSession.readyState != WebSocket.OPEN) return false;
|
|
window.clientSession.send(_uint8ToBase64(msgUint8Arr));
|
|
}
|
|
|
|
window.closeWSConnection = function() {
|
|
if (null == window.clientSession || window.clientSession.readyState != WebSocket.OPEN) return;
|
|
console.log(`Closing "window.clientSession" from the client-side.`);
|
|
window.clientSession.close();
|
|
}
|
|
|
|
window.getBoundRoomIdFromPersistentStorage = function() {
|
|
const boundRoomIdExpiresAt = parseInt(cc.sys.localStorage.getItem("boundRoomIdExpiresAt"));
|
|
if (!boundRoomIdExpiresAt || Date.now() >= boundRoomIdExpiresAt) {
|
|
window.clearBoundRoomIdInBothVolatileAndPersistentStorage();
|
|
return null;
|
|
}
|
|
return cc.sys.localStorage.getItem("boundRoomId");
|
|
};
|
|
|
|
window.clearBoundRoomIdInBothVolatileAndPersistentStorage = function() {
|
|
window.boundRoomId = null;
|
|
cc.sys.localStorage.removeItem("boundRoomId");
|
|
cc.sys.localStorage.removeItem("boundRoomIdExpiresAt");
|
|
};
|
|
|
|
window.clearSelfPlayer = function() {
|
|
cc.sys.localStorage.removeItem("selfPlayer");
|
|
};
|
|
|
|
window.boundRoomId = getBoundRoomIdFromPersistentStorage();
|
|
window.handleHbRequirements = function(resp) {
|
|
if (constants.RET_CODE.OK != resp.ret) return;
|
|
if (null == window.boundRoomId) {
|
|
window.boundRoomId = resp.bciFrame.boundRoomId;
|
|
cc.sys.localStorage.setItem('boundRoomId', window.boundRoomId);
|
|
cc.sys.localStorage.setItem('boundRoomIdExpiresAt', Date.now() + 10 * 60 * 1000); // Temporarily hardcoded, for `boundRoomId` only.
|
|
}
|
|
|
|
if (window.handleBattleColliderInfo) {
|
|
window.handleBattleColliderInfo(resp.bciFrame);
|
|
}
|
|
};
|
|
|
|
function _uint8ToBase64(uint8Arr) {
|
|
return window.btoa(uint8Arr);
|
|
}
|
|
|
|
function _base64ToUint8Array(base64) {
|
|
var origBytes = null;
|
|
if (null != window.atob) {
|
|
var origBinaryStr = window.atob(base64);
|
|
var origLen = origBinaryStr.length;
|
|
origBytes = new Uint8Array(origLen);
|
|
for (var i = 0; i < origLen; i++) {
|
|
origBytes[i] = origBinaryStr.charCodeAt(i);
|
|
}
|
|
return origBytes;
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
function _base64ToArrayBuffer(base64) {
|
|
return _base64ToUint8Array(base64).buffer;
|
|
}
|
|
|
|
window.getExpectedRoomIdSync = function() {
|
|
const qDict = window.getQueryParamDict();
|
|
if (qDict) {
|
|
return qDict["expectedRoomId"];
|
|
} else {
|
|
if (window.history && window.history.state) {
|
|
return window.history.state.expectedRoomId;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
window.unsetClientSessionCloseOrErrorFlag = function() {
|
|
cc.sys.localStorage.removeItem("ClientSessionCloseOrErrorFlag");
|
|
return;
|
|
}
|
|
|
|
window.setClientSessionCloseOrErrorFlag = function() {
|
|
const oldVal = cc.sys.localStorage.getItem("ClientSessionCloseOrErrorFlag");
|
|
if (true == oldVal) return false;
|
|
cc.sys.localStorage.setItem("ClientSessionCloseOrErrorFlag", true);
|
|
return true;
|
|
}
|
|
|
|
window.initPersistentSessionClient = function(onopenCb, expectedRoomId) {
|
|
if (window.clientSession && window.clientSession.readyState == WebSocket.OPEN) {
|
|
if (null != onopenCb) {
|
|
onopenCb();
|
|
}
|
|
return;
|
|
}
|
|
|
|
const intAuthToken = cc.sys.localStorage.getItem("selfPlayer") ? JSON.parse(cc.sys.localStorage.getItem('selfPlayer')).intAuthToken : "";
|
|
|
|
let urlToConnect = backendAddress.PROTOCOL.replace('http', 'ws') + '://' + backendAddress.HOST + ":" + backendAddress.PORT + backendAddress.WS_PATH_PREFIX + "?intAuthToken=" + intAuthToken;
|
|
|
|
if (null != expectedRoomId) {
|
|
console.log("initPersistentSessionClient with expectedRoomId == " + expectedRoomId);
|
|
urlToConnect = urlToConnect + "&expectedRoomId=" + expectedRoomId;
|
|
} else {
|
|
window.boundRoomId = getBoundRoomIdFromPersistentStorage();
|
|
if (null != window.boundRoomId) {
|
|
console.log("initPersistentSessionClient with boundRoomId == " + boundRoomId);
|
|
urlToConnect = urlToConnect + "&boundRoomId=" + window.boundRoomId;
|
|
}
|
|
}
|
|
|
|
const currentHistoryState = window.history && window.history.state ? window.history.state : {};
|
|
|
|
const clientSession = new WebSocket(urlToConnect);
|
|
clientSession.binaryType = 'arraybuffer'; // Make 'event.data' of 'onmessage' an "ArrayBuffer" instead of a "Blob"
|
|
|
|
clientSession.onopen = function(event) {
|
|
console.log("The WS clientSession is opened.");
|
|
window.clientSession = clientSession;
|
|
if (null == onopenCb) return;
|
|
onopenCb();
|
|
};
|
|
|
|
clientSession.onmessage = function(event) {
|
|
if (null == event || null == event.data) {
|
|
return;
|
|
}
|
|
try {
|
|
const resp = window.WsResp.decode(new Uint8Array(event.data));
|
|
switch (resp.act) {
|
|
case window.DOWNSYNC_MSG_ACT_HB_REQ:
|
|
window.handleHbRequirements(resp); // 获取boundRoomId并存储到localStorage
|
|
break;
|
|
case window.DOWNSYNC_MSG_ACT_PLAYER_ADDED_AND_ACKED:
|
|
mapIns.onPlayerAdded(resp.rdf);
|
|
break;
|
|
case window.DOWNSYNC_MSG_ACT_PLAYER_READDED_AND_ACKED:
|
|
// Deliberately left blank for now
|
|
mapIns.hideFindingPlayersGUI();
|
|
break;
|
|
case window.DOWNSYNC_MSG_ACT_BATTLE_READY_TO_START:
|
|
mapIns.onBattleReadyToStart(resp.rdf.playerMetas);
|
|
break;
|
|
case window.DOWNSYNC_MSG_ACT_BATTLE_START:
|
|
mapIns.onRoomDownsyncFrame(resp.rdf);
|
|
break;
|
|
case window.DOWNSYNC_MSG_ACT_BATTLE_STOPPED:
|
|
mapIns.onBattleStopped();
|
|
break;
|
|
case window.DOWNSYNC_MSG_ACT_INPUT_BATCH:
|
|
mapIns.onInputFrameDownsyncBatch(resp.inputFrameDownsyncBatch);
|
|
break;
|
|
case window.DOWNSYNC_MSG_ACT_FORCED_RESYNC:
|
|
if (null == resp.inputFrameDownsyncBatch || 0 >= resp.inputFrameDownsyncBatch.length) {
|
|
console.error("Got empty inputFrameDownsyncBatch upon resync@localRenderFrameId=", mapIns.renderFrameId, ", @lastAllConfirmedRenderFrameId=", mapIns.lastAllConfirmedRenderFrameId, "@lastAllConfirmedInputFrameId=", mapIns.lastAllConfirmedInputFrameId, ", @localRecentInputCache=", mapIns._stringifyRecentInputCache(false), ", the incoming resp=\n", JSON.stringify(resp, null, 2));
|
|
return;
|
|
}
|
|
// Unless upon ws session lost and reconnected, it's maintained true that "inputFrameDownsyncBatch[0].inputFrameId == frontend.lastAllConfirmedInputFrameId+1", and in this case we should try to keep frontend moving only by "frontend.recentInputCache" to avoid jiggling of synced positions
|
|
const inputFrameIdConsecutive = (resp.inputFrameDownsyncBatch[0].inputFrameId == mapIns.lastAllConfirmedInputFrameId + 1);
|
|
const renderFrameIdConsecutive = (resp.rdf.id <= mapIns.renderFrameId + mapIns.renderFrameIdLagTolerance);
|
|
if (inputFrameIdConsecutive && renderFrameIdConsecutive) {
|
|
console.log("Got consecutive resync@localRenderFrameId=", mapIns.renderFrameId, ", @lastAllConfirmedRenderFrameId=", mapIns.lastAllConfirmedRenderFrameId, "@lastAllConfirmedInputFrameId=", mapIns.lastAllConfirmedInputFrameId, ", @localRecentInputCache=", mapIns._stringifyRecentInputCache(false), ", the incoming resp=\n", JSON.stringify(resp));
|
|
mapIns.onInputFrameDownsyncBatch(resp.inputFrameDownsyncBatch);
|
|
} else {
|
|
console.warn("Got forced resync@localRenderFrameId=", mapIns.renderFrameId, ", @lastAllConfirmedRenderFrameId=", mapIns.lastAllConfirmedRenderFrameId, "@lastAllConfirmedInputFrameId=", mapIns.lastAllConfirmedInputFrameId, ", @localRecentInputCache=", mapIns._stringifyRecentInputCache(false), ", the incoming resp=\n", JSON.stringify(resp, null, 2));
|
|
// The following order of execution is important
|
|
const dumpRenderCacheRet = mapIns.onRoomDownsyncFrame(resp.rdf);
|
|
mapIns.onInputFrameDownsyncBatch(resp.inputFrameDownsyncBatch, dumpRenderCacheRet);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
} catch (e) {
|
|
console.error("Unexpected error when parsing data of:", event.data, e);
|
|
}
|
|
};
|
|
|
|
clientSession.onerror = function(event) {
|
|
if (!window.setClientSessionCloseOrErrorFlag()) {
|
|
return;
|
|
}
|
|
console.error("Error caught on the WS clientSession: ", event);
|
|
if (window.clientSessionPingInterval) {
|
|
clearInterval(window.clientSessionPingInterval);
|
|
}
|
|
if (window.handleClientSessionCloseOrError) {
|
|
window.handleClientSessionCloseOrError();
|
|
}
|
|
window.unsetClientSessionCloseOrErrorFlag();
|
|
};
|
|
|
|
clientSession.onclose = function(event) {
|
|
if (!window.setClientSessionCloseOrErrorFlag()) {
|
|
return;
|
|
}
|
|
console.warn("The WS clientSession is closed: ", event);
|
|
if (window.clientSessionPingInterval) {
|
|
clearInterval(window.clientSessionPingInterval);
|
|
}
|
|
if (false == event.wasClean) {
|
|
// Chrome doesn't allow the use of "CustomCloseCode"s (yet) and will callback with a "WebsocketStdCloseCode 1006" and "false == event.wasClean" here. See https://tools.ietf.org/html/rfc6455#section-7.4 for more information.
|
|
if (window.handleClientSessionCloseOrError) {
|
|
window.handleClientSessionCloseOrError();
|
|
}
|
|
} else {
|
|
switch (event.code) {
|
|
case constants.RET_CODE.PLAYER_NOT_FOUND:
|
|
case constants.RET_CODE.PLAYER_CHEATING:
|
|
window.clearBoundRoomIdInBothVolatileAndPersistentStorage();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (window.handleClientSessionCloseOrError) {
|
|
window.handleClientSessionCloseOrError();
|
|
}
|
|
}
|
|
window.unsetClientSessionCloseOrErrorFlag();
|
|
};
|
|
};
|
|
|
|
window.clearLocalStorageAndBackToLoginScene = function(shouldRetainBoundRoomIdInBothVolatileAndPersistentStorage) {
|
|
console.warn("+++++++ Calling `clearLocalStorageAndBackToLoginScene`");
|
|
|
|
if (window.mapIns && window.mapIns.musicEffectManagerScriptIns) {
|
|
window.mapIns.musicEffectManagerScriptIns.stopAllMusic();
|
|
}
|
|
/**
|
|
* Here I deliberately removed the callback in the "common `handleClientSessionCloseOrError` callback"
|
|
* within which another invocation to `clearLocalStorageAndBackToLoginScene` will be made.
|
|
*
|
|
* It'll be re-assigned to the common one upon reentrance of `Map.onLoad`.
|
|
*
|
|
* -- YFLu 2019-04-06
|
|
*/
|
|
window.handleClientSessionCloseOrError = () => {
|
|
console.warn("+++++++ Special handleClientSessionCloseOrError() assigned within `clearLocalStorageAndBackToLoginScene`");
|
|
// TBD.
|
|
window.handleClientSessionCloseOrError = null; // To ensure that it's called at most once.
|
|
};
|
|
window.closeWSConnection();
|
|
window.clearSelfPlayer();
|
|
if (true != shouldRetainBoundRoomIdInBothVolatileAndPersistentStorage) {
|
|
window.clearBoundRoomIdInBothVolatileAndPersistentStorage();
|
|
}
|
|
cc.director.loadScene('login');
|
|
};
|
|
|