Xbox CLoud Gaming优化整合

脚本免费!谨防上当受骗!整合和修改现有脚本,优化项详见脚本说明。【若你有好的想法或者BUG可以进xbox云游戏QQ交流1群531602832,2群313340764,3群826510890,4群82737876反馈】

  1. // ==UserScript==
  2. // @name Xbox CLoud Gaming优化整合
  3. // @name:zh-CN Xbox CLoud Gaming优化整合
  4. // @namespace http://tampermonkey.net/xbox/nft
  5. // @version 3.10.3.3
  6. // @author 奈非天
  7. // @license MIT
  8. // @match https://www.xbox.com/*/*play*
  9. // @run-at document-start
  10. // @grant none
  11. // @require https://lf26-cdn-tos.bytecdntp.com/cdn/expire-1-M/jquery/3.4.1/jquery.min.js
  12. // @description:zh-cn 脚本免费!谨防上当受骗!整合和修改现有脚本,优化项详见脚本说明。【若你有好的想法或者BUG可以进xbox云游戏QQ交流1群531602832,2群313340764,3群826510890,4群82737876反馈】
  13. // @description 脚本免费!谨防上当受骗!整合和修改现有脚本,优化项详见脚本说明。【若你有好的想法或者BUG可以进xbox云游戏QQ交流1群531602832,2群313340764,3群826510890,4群82737876反馈】
  14.  
  15. // ==/UserScript==
  16. (function () {
  17. 'use strict';
  18. // Your code here...
  19.  
  20. //★★★★★★★★★★★★★★★★★★★★Reference Project License Agreement Begin 参考项目许可协议开始★★★★★★★★★★★★★★★★★★★★//
  21.  
  22. /* better-xcloud MIT License
  23.  
  24. Copyright (c) 2023 redphx
  25.  
  26. Permission is hereby granted, free of charge, to any person obtaining a copy
  27. of this software and associated documentation files (the "Software"), to deal
  28. in the Software without restriction, including without limitation the rights
  29. to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  30. copies of the Software, and to permit persons to whom the Software is
  31. furnished to do so, subject to the following conditions:
  32.  
  33. The above copyright notice and this permission notice shall be included in all
  34. copies or substantial portions of the Software.
  35.  
  36. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  37. IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  38. FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  39. AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  40. LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  41. OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  42. SOFTWARE.
  43. */
  44.  
  45. //★★★★★★★★★★★★★★★★★★★★Reference Project License Agreement End 参考项目许可协议结束★★★★★★★★★★★★★★★★★★★★//
  46.  
  47.  
  48. let nftxboxversion = 'v3.10.3.3';
  49.  
  50. let naifeitian = {
  51. isType(obj) {
  52. return Object.prototype.toString.call(obj).replace(/^\[object (.+)\]$/, '$1').toLowerCase();
  53. },
  54. getValue(key) {
  55. try {
  56. return JSON.parse(localStorage.getItem(key));
  57. } catch (e) {
  58. return localStorage.getItem(key);
  59. }
  60. },
  61.  
  62. setValue(key, value) {
  63. if (this.isType(value) === 'object' || this.isType(value) === 'array' || this.isType(value) === 'boolean') {
  64. return localStorage.setItem(key, JSON.stringify(value));
  65. }
  66. return localStorage.setItem(key, value);
  67. },
  68. isValidIP(ip) {
  69. let reg = /^(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])\.(\d{1,2}|1\d\d|2[0-4]\d|25[0-5])$/
  70. return reg.test(ip);
  71. },
  72. isNumber(val) {
  73. return !isNaN(parseFloat(val)) && isFinite(val);
  74. },
  75. toElement(key, onChange) {
  76. const CE = createElement;
  77. const setting = key;
  78. const currentValue = key['default'] == undefined ? key : key['default'];
  79.  
  80. let $control;
  81. if (setting['options'] != undefined) {
  82.  
  83. $control = CE('select', { id: 'xcloud_setting_' + key['name'] });
  84.  
  85. for (let value in setting.options) {
  86. const label = setting.options[value];
  87.  
  88. const $option = CE('option', { value: value }, label);
  89. $control.appendChild($option);
  90. }
  91.  
  92. $control.value = currentValue;
  93. $control.addEventListener('change', e => {
  94. key['default'] = e.target.value;
  95.  
  96. this.setValue(key['name'], key);
  97. onChange && onChange(e);
  98. });
  99.  
  100. } else if (typeof setting.default === 'number') {
  101. $control = CE('input', { 'type': 'number', 'min': setting.min, 'max': setting.max });
  102.  
  103. $control.value = currentValue;
  104. $control.addEventListener('change', e => {
  105. let value = Math.max(setting.min, Math.min(setting.max, parseInt(e.target.value)));
  106. e.target.value = value;
  107.  
  108. key['default'] = e.target.value
  109. this.setValue(key['name'], key);
  110. onChange && onChange(e);
  111. });
  112. } else {
  113. if (key.fps == undefined) {
  114. $control = CE('input', { 'type': 'checkbox' });
  115. $control.checked = currentValue;
  116.  
  117. $control.addEventListener('change', e => {
  118. key['default'] = e.target.checked;
  119. NFTconfig[key['name'].slice(0, -2)]['default'] = e.target.checked;
  120. this.setValue(key['name'], key);
  121. if (key['name'] == 'STATS_SLIDE_OPENGM' && e.target.checked) {
  122. if (this.getValue('STATS_SHOW_WHEN_PLAYINGGM')['default']) {
  123. $('#xcloud_setting_STATS_SHOW_WHEN_PLAYINGGM').click();
  124. }
  125. } else if (key['name'] == 'STATS_SHOW_WHEN_PLAYINGGM' && e.target.checked) {
  126. if (this.getValue('STATS_SLIDE_OPENGM')['default']) {
  127. $('#xcloud_setting_STATS_SLIDE_OPENGM').click();
  128. }
  129. }
  130. onChange && onChange(e);
  131. });
  132. } else {
  133.  
  134. let stats_info_sortedEntries = Object.entries(NFTconfig['stats_info']).sort((a, b) => a[1][1] - b[1][1]);
  135. //流统计信息
  136. $control = CE('div', { 'class': 'stats-container' });
  137. stats_info_sortedEntries.forEach(entry => {
  138.  
  139. //entry[1][0] 是否选中
  140. //entry[1][1] 顺序
  141. //entry[1][2] 名字
  142. let divElement = document.createElement('div');
  143. divElement.className = `drag-handle ${entry[1][0] === true ? 'stats-selected' : 'stats-delete'}`;
  144. divElement.draggable = "true";
  145. divElement.dataset.name = entry[0];
  146. divElement.dataset.index = entry[1][1];
  147. divElement.textContent = entry[1][2];
  148.  
  149. let dragIndicator = document.createElement('div');
  150. dragIndicator.className = "drag-indicator";
  151. divElement.appendChild(dragIndicator);
  152.  
  153. $control.appendChild(divElement);
  154.  
  155.  
  156. });
  157. let placeholder = document.createElement('div');
  158. placeholder.className = "placeholder drag-handle";
  159. $control.appendChild(placeholder);
  160. }
  161. }
  162.  
  163. $control.id = `xcloud_setting_${key.name}`;
  164. return $control;
  165. },
  166. isSafari() {
  167. let userAgent = userAgentOriginal.toLowerCase();
  168. if (userAgent.indexOf('safari') !== -1 && userAgent.indexOf('chrome') === -1) {
  169. return true;
  170. } else {
  171. return false;
  172. }
  173. },
  174. getGM(defaultValue, n) {
  175. let newval = this.getValue(n) == null ? defaultValue : this.getValue(n);
  176. if (newval?.options != undefined) {
  177. newval.options = defaultValue.options;
  178. }
  179. naifeitian.setValue(n, newval);
  180. return newval;
  181. },
  182. showSetting() {
  183. $('#settingsBackgroud').css('display', '');
  184. $('body').css('overflow', 'hidden');
  185. },
  186. hideSetting() {
  187. $('#settingsBackgroud').css('display', 'none');
  188. $('body').css('overflow', 'visible');
  189. },
  190. patchFunctionBind() {
  191. const nativeBind = Function.prototype.bind;
  192. Function.prototype.bind = function () {
  193. let valid = false;
  194. if (this.name.length <= 2 && arguments.length === 2 && arguments[0] === null) {
  195. if (arguments[1] === 0 || (typeof arguments[1] === 'function')) {
  196. valid = true;
  197. }
  198. }
  199.  
  200. if (!valid) {
  201. return nativeBind.apply(this, arguments);
  202. }
  203.  
  204. if (typeof arguments[1] === 'function') {
  205. console.log('还原 Function.prototype.bind()');
  206. Function.prototype.bind = nativeBind;
  207. }
  208.  
  209. const orgFunc = this;
  210. const newFunc = (a, item) => {
  211. if (NFTconfig['PATCH_ORDERS'].length === 0) {
  212. orgFunc(a, item);
  213. return;
  214. }
  215.  
  216. naifeitian.patch(item);
  217. orgFunc(a, item);
  218. }
  219.  
  220. return nativeBind.apply(newFunc, arguments);
  221. };
  222. },
  223. patch(item) {
  224. // console.log('patch', '-----');
  225. let patchName;
  226. let appliedPatches;
  227.  
  228. for (let id in item[1]) {
  229. if (NFTconfig['PATCH_ORDERS'].length <= 0) {
  230. return;
  231. }
  232.  
  233. appliedPatches = [];
  234. const func = item[1][id];
  235. let funcStr = func.toString();
  236.  
  237. for (let groupIndex = 0; groupIndex < NFTconfig['PATCH_ORDERS'].length; groupIndex++) {
  238. const group = NFTconfig['PATCH_ORDERS'][groupIndex];
  239. let modified = false;
  240.  
  241. for (let patchIndex = 0; patchIndex < group.length; patchIndex++) {
  242. const patchName = group[patchIndex];
  243. if (appliedPatches.indexOf(patchName) > -1) {
  244. continue;
  245. }
  246.  
  247. const patchedFuncStr = naifeitian.handle_remote_patch(patchName, funcStr);
  248. if (!patchedFuncStr) {
  249. // Only stop if the first patch is failed
  250. if (patchIndex === 0) {
  251. break;
  252. } else {
  253. continue;
  254. }
  255. }
  256.  
  257. modified = true;
  258. funcStr = patchedFuncStr;
  259.  
  260. console.log(`应用 "${patchName}" 修补`);
  261. appliedPatches.push(patchName);
  262.  
  263. // Remove patch from group
  264. group.splice(patchIndex, 1);
  265. patchIndex--;
  266. }
  267.  
  268. // Apply patched functions
  269. if (modified) {
  270. item[1][id] = eval(funcStr);
  271. }
  272.  
  273. // Remove empty group
  274. if (!group.length) {
  275. NFTconfig['PATCH_ORDERS'].splice(groupIndex, 1);
  276. groupIndex--;
  277. }
  278. }
  279. }
  280. },
  281. handle_remote_patch(name, funcStr) {
  282. //根据不同的字符串执行不同的方法
  283. if (name == 'remotePlayConnectMode') {
  284. const text = 'connectMode:"cloud-connect"';
  285. if (!funcStr.includes(text)) {
  286. return false;
  287. }
  288.  
  289. return funcStr.replace(text, `connectMode:window.BX_REMOTE_PLAY_CONFIG?"xhome-connect":"cloud-connect",remotePlayServerId:(window.BX_REMOTE_PLAY_CONFIG&&window.BX_REMOTE_PLAY_CONFIG.serverId)||''`);
  290.  
  291. } else if (name == 'remotePlayDirectConnectUrl') {
  292. const index = funcStr.indexOf('/direct-connect');
  293. if (index === -1) {
  294. return false;
  295. }
  296. return funcStr.replace(funcStr.substring(index - 9, index + 15), 'https://www.xbox.com/play');
  297. } else if (name == 'remotePlayKeepAlive') {
  298. if (!funcStr.includes('onServerDisconnectMessage(e){')) {
  299. return false;
  300. }
  301.  
  302. funcStr = funcStr.replace('onServerDisconnectMessage(e){', `onServerDisconnectMessage (e) {
  303. const msg = JSON.parse(e);
  304. if (msg.reason === 'WarningForBeingIdle') {
  305. try {
  306. this.sendKeepAlive();
  307. return;
  308. } catch (ex) {}
  309. }
  310. `);
  311. return funcStr;
  312.  
  313. } else if (name == 'EnableStreamGate') {
  314. const index = funcStr.indexOf(',EnableStreamGate:');
  315. if (index === -1) {
  316. return false;
  317. }
  318.  
  319. // Find the next "},"
  320. const endIndex = funcStr.indexOf('},', index);
  321.  
  322. const newCode = `
  323. EnableStreamGate: false,
  324. PwaPrompt: false,
  325. `;
  326. funcStr = funcStr.substring(0, endIndex) + ',' + newCode + funcStr.substring(endIndex);
  327. return funcStr;
  328. } else if (name == 'remotePlayGuideWorkaround') {
  329. const text = 'nexusButtonHandler:this.featureGates.EnableClientGuideInStream';
  330. if (!funcStr.includes(text)) {
  331. return false;
  332. }
  333.  
  334. return funcStr.replace(text, `nexusButtonHandler: !window.BX_REMOTE_PLAY_CONFIG && this.featureGates.EnableClientGuideInStream`);
  335.  
  336. } else if (name == 'patchStreamHud') {
  337. const text = 'let{onCollapse';
  338. if (!funcStr.includes(text)) {
  339. return false;
  340. }
  341.  
  342. // 恢复悬浮窗 "..." 按钮
  343. funcStr = funcStr.replace(text, 'e.guideUI = null;' + text);
  344.  
  345. return funcStr;
  346.  
  347. } else if (name == "loadingEndingChunks") {
  348. // Add patches that are only needed when start playing
  349. const text = 'Symbol("ChatSocketPlugin")';
  350. if (!funcStr.includes(text)) {
  351. return false;
  352. }
  353.  
  354. NFTconfig['PATCH_ORDERS'] = NFTconfig['PATCH_ORDERS'].concat(NFTconfig['PLAYING_PATCH_ORDERS']);
  355.  
  356. return funcStr;
  357. }
  358. },
  359. isDivTopOrBottomOutOfBounds(divElement) {
  360. const $div = $(divElement);
  361. $div.css("height","")
  362.  
  363. // 获取div的边界信息
  364. const divRect = $div[0].getBoundingClientRect();
  365. const divTop = divRect.top;
  366. const divBottom = divRect.bottom;
  367.  
  368. const viewportHeight = window.innerHeight || document.documentElement.clientHeight;
  369.  
  370. return (
  371. divBottom > viewportHeight ||
  372. divTop < 0
  373. );
  374. }
  375.  
  376. }
  377. //★★ 1=开 0=关 ★★//
  378. let default_language_list = { '智能简繁': 'Auto', '简体': 'zh-CN', '繁体': 'zh-TW' }
  379. let NFTconfig =
  380. {
  381. enableRemotePlay: 0,
  382. PATCH_ORDERS: [],
  383. PLAYING_PATCH_ORDERS: [],
  384. no_need_VPN_play: 1,
  385. regionBlock: {
  386. blockIp: '美服',
  387. options: {
  388. '韩服': '168.126.63.1',
  389. '美服': '4.2.2.2',
  390. '日服': '210.131.113.123'
  391. }
  392. },
  393. chooseLanguage: 1,
  394. IfErrUsedefaultGameLanguage: 'zh-CN',
  395. high_bitrate: 1,
  396. disableCheckNetwork: 1,
  397. IPv6: 0,
  398. autoFullScreen: 0,
  399. blockXcloudServer: 0,
  400. blockXcloudServerList: ['AustraliaEast', 'AustraliaSouthEast', 'BrazilSouth', 'EastUS', 'EastUS2', 'JapanEast', 'KoreaCentral', 'NorthCentralUs', 'SouthCentralUS', 'UKSouth', 'WestEurope', 'WestUS', 'WestUS2'],
  401. defaultXcloudServer: 'KoreaCentral',
  402. video_stretch: {
  403. default: 'none',
  404. options: {
  405. none: '无',
  406. fill: '填充',
  407. setting: '微调'
  408. },
  409. name: 'video_stretchGM'
  410. },
  411. rtcCodecPreferences: {
  412. default: '自动',
  413. options: [
  414. '默认',
  415. '自动'
  416. ]
  417. },
  418. video_stretch_x_y: {
  419. x: 0,
  420. y: 0,
  421. name: 'video_stretch_x_yGM'
  422. },
  423.  
  424. noPopSetting: 0,
  425. disableTouchControls: 0,
  426. autoOpenOC: 1,
  427. autoShowTouch: true,
  428. STATS_SHOW_WHEN_PLAYING: {
  429. default: false,
  430. name: 'STATS_SHOW_WHEN_PLAYINGGM'
  431. },
  432.  
  433. STATS_POSITION: {
  434. default: 'top-left',
  435. options: {
  436. 'top-left': '上左',
  437. 'top-center': '上中',
  438. 'top-right': '上右'
  439. },
  440.  
  441. name: 'STATS_POSITIONGM'
  442. },
  443.  
  444. STATS_TRANSPARENT: {
  445. default: false,
  446. name: 'STATS_TRANSPARENTGM'
  447. },
  448.  
  449. STATS_OPACITY: {
  450. default: 80,
  451. min: 10,
  452. max: 100,
  453. name: 'STATS_OPACITYGM'
  454. },
  455.  
  456. STATS_TEXT_SIZE: {
  457. default: '0.9rem',
  458. options: {
  459. '0.9rem': '小',
  460. '1.0rem': '中',
  461. '1.1rem': '大'
  462. },
  463.  
  464. name: 'STATS_TEXT_SIZEGM'
  465. },
  466.  
  467. STATS_CONDITIONAL_FORMATTING: {
  468. default: false,
  469. name: 'STATS_CONDITIONAL_FORMATTINGGM'
  470. },
  471. STATS_SLIDE_OPEN: {
  472. default: false,
  473. name: 'STATS_SLIDE_OPENGM'
  474. },
  475. VIDEO_CLARITY: {
  476. default: 0,
  477. min: 0,
  478. max: 3,
  479. name: 'VIDEO_CLARITYGM'
  480. },
  481.  
  482. VIDEO_CONTRAST: {
  483. default: 100,
  484. min: 0,
  485. max: 150,
  486. name: 'VIDEO_CONTRASTGM'
  487. },
  488.  
  489. VIDEO_SATURATION: {
  490. default: 100,
  491. min: 0,
  492. max: 150,
  493. name: 'VIDEO_SATURATIONGM'
  494. },
  495.  
  496. VIDEO_BRIGHTNESS: {
  497. default: 100,
  498. min: 0,
  499. max: 150,
  500. name: 'VIDEO_BRIGHTNESSGM'
  501. },
  502. antiKick: 0,
  503. useCustomfakeIp: 0,
  504. customfakeIp: '',
  505. xcloud_game_language: default_language_list['简体'],
  506. REMOTE_PLAY_RESOLUTION: {
  507. 'default': '1080p',
  508. 'options': {
  509. '1080p': '1080p',
  510. '720p': '720p',
  511. },
  512. 'name': 'REMOTE_PLAY_RESOLUTIONGM'
  513. },
  514. REMOTE_SERVER_LIST: ['eau', 'seau', 'brs', 'eus', 'eus2', 'ejp', 'ckr', 'mxc', 'ncus', 'scus', 'uks', 'weu', 'wus', 'wus2'],
  515.  
  516. stats_info: {
  517. fps: [true, 1, "帧率"],
  518. rtt: [true, 2, "延迟"],
  519. dt: [true, 3, "解码"],
  520. br: [true, 4, "码率"],
  521. pl: [true, 5, "丢包"],
  522. fl: [true, 6, "丢帧"],
  523. }
  524. }
  525.  
  526. const integratekeys = Object.keys(NFTconfig);
  527.  
  528. integratekeys.forEach(key => {
  529. NFTconfig[key] = naifeitian.getGM(NFTconfig[key], key + 'GM');
  530. });
  531.  
  532. NFTconfig['PATCH_ORDERS'] = [
  533.  
  534. NFTconfig['enableRemotePlay'] == 1 && ['remotePlayKeepAlive'],
  535. NFTconfig['enableRemotePlay'] == 1 && ['remotePlayDirectConnectUrl'],
  536.  
  537. ];
  538. NFTconfig['PATCH_ORDERS'] = [
  539.  
  540. NFTconfig['enableRemotePlay'] == 1 && ['remotePlayConnectMode'],
  541. NFTconfig['enableRemotePlay'] == 1 && ['remotePlayGuideWorkaround'],
  542.  
  543. ['patchStreamHud'],
  544. ['EnableStreamGate']
  545. ]
  546.  
  547.  
  548. let regionsMenuItemList = [];
  549. let languageMenuItemList = [];
  550. let crturl = "";
  551. let canShowOC = null;
  552.  
  553. let letmeOb = true;
  554. let checkIpsuc = false;
  555.  
  556. let STREAM_WEBRTC;
  557. const ICON_VIDEO_SETTINGS = '<path d="M16 9.144A6.89 6.89 0 0 0 9.144 16 6.89 6.89 0 0 0 16 22.856 6.89 6.89 0 0 0 22.856 16 6.9 6.9 0 0 0 16 9.144zm0 11.427c-2.507 0-4.571-2.064-4.571-4.571s2.064-4.571 4.571-4.571 4.571 2.064 4.571 4.571-2.064 4.571-4.571 4.571zm15.704-7.541c-.065-.326-.267-.607-.556-.771l-4.26-2.428-.017-4.802c-.001-.335-.15-.652-.405-.868-1.546-1.307-3.325-2.309-5.245-2.953-.306-.103-.641-.073-.923.085L16 3.694l-4.302-2.407c-.282-.158-.618-.189-.924-.086a16.02 16.02 0 0 0-5.239 2.964 1.14 1.14 0 0 0-.403.867L5.109 9.84.848 12.268a1.14 1.14 0 0 0-.555.771 15.22 15.22 0 0 0 0 5.936c.064.326.267.607.555.771l4.261 2.428.017 4.802c.001.335.149.652.403.868 1.546 1.307 3.326 2.309 5.245 2.953.306.103.641.073.923-.085L16 28.306l4.302 2.407a1.13 1.13 0 0 0 .558.143 1.18 1.18 0 0 0 .367-.059c1.917-.648 3.695-1.652 5.239-2.962.255-.216.402-.532.405-.866l.021-4.807 4.261-2.428a1.14 1.14 0 0 0 .555-.771 15.21 15.21 0 0 0-.003-5.931zm-2.143 4.987l-4.082 2.321a1.15 1.15 0 0 0-.429.429l-.258.438a1.13 1.13 0 0 0-.174.601l-.022 4.606a13.71 13.71 0 0 1-3.623 2.043l-4.117-2.295a1.15 1.15 0 0 0-.559-.143h-.546c-.205-.005-.407.045-.586.143l-4.119 2.3a13.74 13.74 0 0 1-3.634-2.033l-.016-4.599a1.14 1.14 0 0 0-.174-.603l-.257-.437c-.102-.182-.249-.333-.429-.437l-4.085-2.328a12.92 12.92 0 0 1 0-4.036l4.074-2.325a1.15 1.15 0 0 0 .429-.429l.258-.438a1.14 1.14 0 0 0 .175-.601l.021-4.606a13.7 13.7 0 0 1 3.625-2.043l4.11 2.295a1.14 1.14 0 0 0 .585.143h.52c.205.005.407-.045.586-.143l4.119-2.3a13.74 13.74 0 0 1 3.634 2.033l.016 4.599a1.14 1.14 0 0 0 .174.603l.257.437c.102.182.249.333.429.438l4.085 2.327a12.88 12.88 0 0 1 .007 4.041h.007z" fill-rule="nonzero"/>';
  558.  
  559. //视频调整
  560. const ICON_HD_VIDEO_SETTINGS = '<g transform="matrix(.142357 0 0 .142357 -2.22021 -2.22164)" fill="none" stroke="#fff" stroke-width="16"><circle cx="128" cy="128" r="40"/><path d="M130.05 206.11h-4L94 224c-12.477-4.197-24.049-10.711-34.11-19.2l-.12-36c-.71-1.12-1.38-2.25-2-3.41L25.9 147.24a99.16 99.16 0 0 1 0-38.46l31.84-18.1c.65-1.15 1.32-2.29 2-3.41l.16-36C69.951 42.757 81.521 36.218 94 32l32 17.89h4L162 32c12.477 4.197 24.049 10.711 34.11 19.2l.12 36c.71 1.12 1.38 2.25 2 3.41l31.85 18.14a99.16 99.16 0 0 1 0 38.46l-31.84 18.1c-.65 1.15-1.32 2.29-2 3.41l-.16 36A104.59 104.59 0 0 1 162 224l-31.95-17.89z"/></g>';
  561.  
  562. //流监控
  563. const ICON_HD_STREAM_STATS = '<g transform="scale(2)" class="ICON_HD_STREAM_STATS_OFF" style="display:block"><path d="M13.359 11.238C15.06 9.72 16 8 16 8s-3-5.5-8-5.5a7 7 0 0 0-2.79.588l.77.771A6 6 0 0 1 8 3.5c2.12 0 3.879 1.168 5.168 2.457A13 13 0 0 1 14.828 8q-.086.13-.195.288c-.335.48-.83 1.12-1.465 1.755q-.247.248-.517.486z"></path><path d="M11.297 9.176a3.5 3.5 0 0 0-4.474-4.474l.823.823a2.5 2.5 0 0 1 2.829 2.829zm-2.943 1.299.822.822a3.5 3.5 0 0 1-4.474-4.474l.823.823a2.5 2.5 0 0 0 2.829 2.829"></path><path d="M3.35 5.47q-.27.24-.518.487A13 13 0 0 0 1.172 8l.195.288c.335.48.83 1.12 1.465 1.755C4.121 11.332 5.881 12.5 8 12.5c.716 0 1.39-.133 2.02-.36l.77.772A7 7 0 0 1 8 13.5C3 13.5 0 8 0 8s.939-1.721 2.641-3.238l.708.709zm10.296 8.884-12-12 .708-.708 12 12z"></path></g><g transform="scale(2)" class="ICON_HD_STREAM_STATS_ON" style="display:none"><path d="M16 8s-3-5.5-8-5.5S0 8 0 8s3 5.5 8 5.5S16 8 16 8M1.173 8a13 13 0 0 1 1.66-2.043C4.12 4.668 5.88 3.5 8 3.5s3.879 1.168 5.168 2.457A13 13 0 0 1 14.828 8q-.086.13-.195.288c-.335.48-.83 1.12-1.465 1.755C11.879 11.332 10.119 12.5 8 12.5s-3.879-1.168-5.168-2.457A13 13 0 0 1 1.172 8z"/><path d="M8 5.5a2.5 2.5 0 1 0 0 5 2.5 2.5 0 0 0 0-5M4.5 8a3.5 3.5 0 1 1 7 0 3.5 3.5 0 0 1-7 0"/></g>';
  564.  
  565. // Quickly create a tree of elements without having to use innerHTML
  566. function createElement(elmName, props = {}) {
  567. let $elm;
  568. const hasNs = 'xmlns' in props;
  569.  
  570. if (hasNs) {
  571. $elm = document.createElementNS(props.xmlns, elmName);
  572. } else {
  573. $elm = document.createElement(elmName);
  574. }
  575.  
  576. for (let key in props) {
  577. if (key === 'xmlns') {
  578. continue;
  579. }
  580.  
  581. if (!props.hasOwnProperty(key) || $elm.hasOwnProperty(key)) {
  582. continue;
  583. }
  584.  
  585. if (hasNs) {
  586. $elm.setAttributeNS(null, key, props[key]);
  587. } else {
  588. $elm.setAttribute(key, props[key]);
  589. }
  590. }
  591.  
  592. for (let i = 2, size = arguments.length; i < size; i++) {
  593. const arg = arguments[i];
  594. const argType = typeof arg;
  595.  
  596. if (argType === 'string' || argType === 'number') {
  597. $elm.textContent = arg;
  598. } else if (arg) {
  599. $elm.appendChild(arg);
  600. }
  601. }
  602.  
  603. return $elm;
  604. }
  605.  
  606. function setMachineFullScreen() {
  607. try {
  608. let element = document.documentElement;
  609. if (element.requestFullscreen) {
  610. element.requestFullscreen();
  611. } else if (element.mozRequestFullScreen) {
  612. element.mozRequestFullScreen();
  613. } else if (element.msRequestFullscreen) {
  614. element.msRequestFullscreen();
  615. } else if (element.webkitRequestFullscreen) {
  616. element.webkitRequestFullScreen();
  617. }
  618. screen?.orientation?.lock("landscape");
  619. } catch (e) {
  620. }
  621. }
  622.  
  623. function exitMachineFullscreen() {
  624. try {
  625. screen?.orientation?.unlock();
  626. if (document.exitFullScreen) {
  627. document.exitFullScreen();
  628. } else if (document.mozCancelFullScreen) {
  629. document.mozCancelFullScreen();
  630. } else if (document.webkitExitFullscreen) {
  631. document.webkitExitFullscreen();
  632. } else if (element.msExitFullscreen) {
  633. element.msExitFullscreen();
  634. }
  635. } catch (e) {
  636. }
  637. }
  638. function exitGame() {
  639.  
  640. canShowOC = null;
  641. setTimeout(RemotePlay.detect, 10);
  642. document.documentElement.style.overflowY = "";
  643. StreamStats.hideSettingsUi();
  644. letmeOb = true;
  645. StreamStats.stop();
  646. bindmslogoevent();
  647. $('.better-xcloud-quick-settings-bar').css("display", "none");
  648. if (NFTconfig['autoFullScreen'] == 1) {
  649. exitMachineFullscreen();
  650. }
  651. if (NFTconfig['noPopSetting'] == 0) {
  652. $('#popSetting').css('display', 'block');
  653. }
  654. }
  655. function inGame() {
  656. if (!IS_REMOTE_PLAYING) {
  657. let path = window.location.pathname;
  658. history.pushState({}, null, window.location.pathname.substr(0, window.location.pathname.indexOf("/launch/")));
  659. history.pushState({}, null, path);
  660. }
  661. document.documentElement.style.overflowY = "hidden";
  662.  
  663. document.body.style.top = '0px';
  664. if (NFTconfig['autoFullScreen'] == 1) {
  665. setMachineFullScreen();
  666. }
  667. if (NFTconfig['noPopSetting'] == 0) {
  668. $('#popSetting').css('display', 'none');
  669. }
  670. }
  671. class StreamBadges {
  672. static get BADGE_PLAYTIME() { return '游玩时间'; };
  673. static get BADGE_BATTERY() { return '电量'; };
  674. static get BADGE_IN() { return '下载'; };
  675. static get BADGE_OUT() { return '上传'; };
  676.  
  677. static get BADGE_SERVER() { return '服务器'; };
  678. static get BADGE_VIDEO() { return '编解码器'; };
  679. static get BADGE_AUDIO() { return '音频'; };
  680.  
  681. static get BADGE_BREAK() { return 'break'; };
  682.  
  683. static ipv6 = false;
  684. static resolution = null;
  685. static video = null;
  686. static audio = null;
  687. static fps = 0;
  688. static region = '';
  689.  
  690. static startBatteryLevel = 100;
  691. static startTimestamp = 0;
  692.  
  693. static #cachedDoms = {};
  694.  
  695. static #interval;
  696. static get #REFRESH_INTERVAL() { return 3000; };
  697.  
  698. static #renderBadge(name, value, color) {
  699. const CE = createElement;
  700.  
  701. if (name === StreamBadges.BADGE_BREAK) {
  702. return CE('div', { 'style': 'display: block' });
  703. }
  704.  
  705. let $badge;
  706. if (StreamBadges.#cachedDoms[name]) {
  707. $badge = StreamBadges.#cachedDoms[name];
  708. $badge.lastElementChild.textContent = value;
  709. return $badge;
  710. }
  711.  
  712. $badge = CE('div', { 'class': 'better-xcloud-badge' },
  713. CE('span', { 'class': 'better-xcloud-badge-name' }, name),
  714. CE('span', { 'class': 'better-xcloud-badge-value', 'style': `background-color: ${color}` }, value));
  715.  
  716. if (name === StreamBadges.BADGE_BATTERY) {
  717. $badge.classList.add('better-xcloud-badge-battery');
  718. }
  719.  
  720. StreamBadges.#cachedDoms[name] = $badge;
  721. return $badge;
  722. }
  723.  
  724. static async #updateBadges(forceUpdate) {
  725. if (!forceUpdate && !document.querySelector('.better-xcloud-badges')) {
  726. StreamBadges.#stop();
  727. return;
  728. }
  729.  
  730. // 游玩时间
  731. let now = +new Date;
  732. const diffSeconds = Math.ceil((now - StreamBadges.startTimestamp) / 1000);
  733. const playtime = StreamBadges.#secondsToHm(diffSeconds);
  734.  
  735. // 电量
  736. let batteryLevel = '100%';
  737. let batteryLevelInt = 100;
  738. let isCharging = false;
  739. if (navigator.getBattery) {
  740. try {
  741. const bm = await navigator.getBattery();
  742. isCharging = bm.charging;
  743. batteryLevelInt = Math.round(bm.level * 100);
  744. batteryLevel = `${batteryLevelInt}%`;
  745.  
  746. if (batteryLevelInt != StreamBadges.startBatteryLevel) {
  747. const diffLevel = Math.round(batteryLevelInt - StreamBadges.startBatteryLevel);
  748. const sign = diffLevel > 0 ? '+' : '';
  749. batteryLevel += ` (${sign}${diffLevel}%)`;
  750. }
  751. } catch (e) { }
  752. }
  753.  
  754. const stats = await STREAM_WEBRTC.getStats();
  755. let totalIn = 0;
  756. let totalOut = 0;
  757. stats.forEach(stat => {
  758. if (stat.type === 'candidate-pair' && stat.state == 'succeeded') {
  759. totalIn += stat.bytesReceived;
  760. totalOut += stat.bytesSent;
  761. }
  762. });
  763.  
  764. const badges = {
  765. [StreamBadges.BADGE_IN]: totalIn ? StreamBadges.#humanFileSize(totalIn) : null,
  766. [StreamBadges.BADGE_OUT]: totalOut ? StreamBadges.#humanFileSize(totalOut) : null,
  767. [StreamBadges.BADGE_PLAYTIME]: playtime,
  768. [StreamBadges.BADGE_BATTERY]: batteryLevel,
  769. };
  770.  
  771. for (let name in badges) {
  772. const value = badges[name];
  773. if (value === null) {
  774. continue;
  775. }
  776.  
  777. const $elm = StreamBadges.#cachedDoms[name];
  778. $elm && ($elm.lastElementChild.textContent = value);
  779.  
  780. if (name === StreamBadges.BADGE_BATTERY) {
  781. // Show charging status
  782. $elm.setAttribute('data-charging', isCharging);
  783.  
  784. if (StreamBadges.startBatteryLevel === 100 && batteryLevelInt === 100) {
  785. $elm.style.display = 'none';
  786. } else {
  787. $elm.style = '';
  788. }
  789. }
  790. }
  791. }
  792.  
  793. static #stop() {
  794. StreamBadges.#interval && clearInterval(StreamBadges.#interval);
  795. StreamBadges.#interval = null;
  796. }
  797.  
  798. static #secondsToHm(seconds) {
  799. const h = Math.floor(seconds / 3600);
  800. const m = Math.floor(seconds % 3600 / 60) + 1;
  801.  
  802. const hDisplay = h > 0 ? `${h}小时` : '';
  803. const mDisplay = m > 0 ? `${m}分钟` : '';
  804. return hDisplay + mDisplay;
  805. }
  806.  
  807. // https://stackoverflow.com/a/20732091
  808. static #humanFileSize(size) {
  809. let i = size == 0 ? 0 : Math.floor(Math.log(size) / Math.log(1024));
  810. return (size / Math.pow(1024, i)).toFixed(2) * 1 + ' ' + ['B', 'kB', 'MB', 'GB', 'TB'][i];
  811. }
  812.  
  813. static async render() {
  814. // Video
  815. let video = '';
  816. if (StreamBadges.resolution) {
  817. video = `${StreamBadges.resolution.height}p`;
  818. }
  819.  
  820. if (StreamBadges.video) {
  821. video && (video += '/');
  822. video += StreamBadges.video.codec;
  823. if (StreamBadges.video.profile) {
  824. let profile = StreamBadges.video.profile;
  825. profile = profile.startsWith('4d') ? '高' : (profile.startsWith('42e') ? '中' : '低');
  826. video += ` (${profile})`;
  827. }
  828. }
  829.  
  830. // 音频
  831. let audio;
  832. if (StreamBadges.audio) {
  833. audio = StreamBadges.audio.codec;
  834. const bitrate = StreamBadges.audio.bitrate / 1000;
  835. audio += ` (${bitrate} kHz)`;
  836. }
  837.  
  838. // 电量
  839. let batteryLevel = '';
  840. if (navigator.getBattery) {
  841. batteryLevel = '100%';
  842. }
  843.  
  844. // Server + Region
  845. let server = StreamBadges.region;
  846. server += '@' + (StreamBadges.ipv6 ? 'IPv6' : 'IPv4');
  847.  
  848. const BADGES = [
  849. [StreamBadges.BADGE_PLAYTIME, '1m', '#ff004d'],
  850. [StreamBadges.BADGE_BATTERY, batteryLevel, '#00b543'],
  851. [StreamBadges.BADGE_IN, StreamBadges.#humanFileSize(0), '#29adff'],
  852. [StreamBadges.BADGE_OUT, StreamBadges.#humanFileSize(0), '#ff77a8'],
  853. [StreamBadges.BADGE_BREAK],
  854. [StreamBadges.BADGE_SERVER, server, '#ff6c24'],
  855. video ? [StreamBadges.BADGE_VIDEO, video, '#742f29'] : null,
  856. audio ? [StreamBadges.BADGE_AUDIO, audio, '#5f574f'] : null,
  857. ];
  858.  
  859. const $wrapper = createElement('div', { 'class': 'better-xcloud-badges' });
  860. BADGES.forEach(item => item && $wrapper.appendChild(StreamBadges.#renderBadge(...item)));
  861.  
  862. await StreamBadges.#updateBadges(true);
  863. StreamBadges.#stop();
  864. StreamBadges.#interval = setInterval(StreamBadges.#updateBadges, StreamBadges.#REFRESH_INTERVAL);
  865.  
  866. return $wrapper;
  867. }
  868. }
  869. class StreamStats {
  870. static #interval;
  871. static #updateInterval = 1000;
  872.  
  873. static #$container;
  874. static #$fps;
  875. static #$rtt;
  876. static #$dt;
  877. static #$pl;
  878. static #$fl;
  879. static #$br;
  880.  
  881. static #$settings;
  882.  
  883. static #lastStat;
  884.  
  885. static status() {
  886. return StreamStats.#interval != null;
  887. }
  888.  
  889. static start() {
  890. clearInterval(StreamStats.#interval);
  891.  
  892. StreamStats.#$container.classList.remove('better-xcloud-gone');
  893. StreamStats.#interval = setInterval(StreamStats.update, StreamStats.#updateInterval);
  894. $('#xcloud_setting_STATS_BUTTON').text("关闭监控");
  895. $('.ICON_HD_STREAM_STATS_ON').css("display", 'block');
  896. $('.ICON_HD_STREAM_STATS_OFF').css("display", 'none');
  897.  
  898. }
  899.  
  900.  
  901. static stop() {
  902. clearInterval(StreamStats.#interval);
  903.  
  904. StreamStats.#$container.classList.add('better-xcloud-gone');
  905. StreamStats.#interval = null;
  906. StreamStats.#lastStat = null;
  907. $('#xcloud_setting_STATS_BUTTON').text("启动监控");
  908. $('.ICON_HD_STREAM_STATS_ON').css("display", 'none');
  909. $('.ICON_HD_STREAM_STATS_OFF').css("display", 'block');
  910. }
  911.  
  912. static toggle() {
  913. StreamStats.#isHidden() ? StreamStats.start() : StreamStats.stop();
  914. screenClicktohide();
  915. if (naifeitian.isDivTopOrBottomOutOfBounds(".better-xcloud-stats-settings")) {
  916. $(".better-xcloud-stats-settings").css("height", "90%");
  917. } else {
  918. $(".better-xcloud-stats-settings").css("height", "");
  919. }
  920. }
  921.  
  922. static #isHidden = () => StreamStats.#$container.classList.contains('better-xcloud-gone');
  923.  
  924. static update() {
  925. if (StreamStats.#isHidden() || !STREAM_WEBRTC) {
  926. StreamStats.stop();
  927. return;
  928. }
  929.  
  930. try {
  931.  
  932. STREAM_WEBRTC.getStats().then(stats => {
  933. stats.forEach(stat => {
  934. let grade = '';
  935. if (stat.type === 'inbound-rtp' && stat.kind === 'video') {
  936. // FPS
  937. $(".stats_fps span").text(stat.framesPerSecond || 0);
  938.  
  939. // Packets Lost
  940. const packetsLost = stat.packetsLost;
  941. if (packetsLost != undefined) {
  942. const packetsReceived = stat.packetsReceived;
  943. const packetsLostPercentage = (packetsLost * 100 / ((packetsLost + packetsReceived) || 1)).toFixed(2);
  944. $(".stats_pl span").text(`${packetsLost} (${packetsLostPercentage}%)`);
  945. } else {
  946. $(".stats_pl span").text(`-1 (-1%)`);
  947. }
  948.  
  949. // Frames Dropped
  950. const framesDropped = stat.framesDropped;
  951. if (framesDropped != undefined) {
  952. const framesReceived = stat.framesReceived;
  953. const framesDroppedPercentage = (framesDropped * 100 / ((framesDropped + framesReceived) || 1)).toFixed(2);
  954. $(".stats_fl span").text(`${framesDropped} (${framesDroppedPercentage}%)`);
  955. } else {
  956. $(".stats_fl span").text(`-1 (-1%)`);
  957. }
  958. if (StreamStats.#lastStat) {
  959. const lastStat = StreamStats.#lastStat;
  960. // Bitrate
  961. const timeDiff = stat.timestamp - lastStat.timestamp;
  962. const bitrate = 8 * (stat.bytesReceived - lastStat.bytesReceived) / timeDiff / 1000;
  963. $(".stats_br span").text(`${bitrate.toFixed(2)} Mbps`);
  964.  
  965. // Decode time
  966. const totalDecodeTimeDiff = stat.totalDecodeTime - lastStat.totalDecodeTime;
  967. const framesDecodedDiff = stat.framesDecoded - lastStat.framesDecoded;
  968. const currentDecodeTime = totalDecodeTimeDiff / framesDecodedDiff * 1000;
  969. $(".stats_dt span").text(`${currentDecodeTime.toFixed(2)}ms`);
  970.  
  971. if (NFTconfig['STATS_CONDITIONAL_FORMATTING']['default']) {
  972. grade = (currentDecodeTime > 12) ? 'bad' : (currentDecodeTime > 9) ? 'ok' : (currentDecodeTime > 6) ? 'good' : '';
  973. }
  974. $(".stats_dt span").attr('data-grade', grade);
  975. }
  976.  
  977. StreamStats.#lastStat = stat;
  978. } else if (stat.type === 'candidate-pair' && stat.state === 'succeeded') {
  979. // Round Trip Time
  980. const roundTripTime = typeof stat.currentRoundTripTime !== 'undefined' ? stat.currentRoundTripTime * 1000 : '???';
  981. $(".stats_rtt span").text(`${roundTripTime}ms`);
  982.  
  983. if (NFTconfig['STATS_CONDITIONAL_FORMATTING']['default']) {
  984. grade = (roundTripTime > 100) ? 'bad' : (roundTripTime > 75) ? 'ok' : (roundTripTime > 40) ? 'good' : '';
  985. }
  986. $(".stats_rtt span").attr('data-grade', grade);
  987. }
  988. });
  989. });
  990. } catch (e) { }
  991. }
  992.  
  993. static #refreshStyles() {
  994. const PREF_POSITION = NFTconfig['STATS_POSITION']['default'];
  995. const PREF_TRANSPARENT = NFTconfig['STATS_TRANSPARENT']['default'];
  996. const PREF_OPACITY = NFTconfig['STATS_OPACITY']['default'];
  997. const PREF_TEXT_SIZE = NFTconfig['STATS_TEXT_SIZE']['default'];
  998.  
  999. StreamStats.#$container.setAttribute('data-position', PREF_POSITION);
  1000. StreamStats.#$container.setAttribute('data-transparent', PREF_TRANSPARENT);
  1001. StreamStats.#$container.style.opacity = PREF_OPACITY + '%';
  1002. StreamStats.#$container.style.fontSize = PREF_TEXT_SIZE;
  1003. }
  1004.  
  1005. static hideSettingsUi() {
  1006. StreamStats.#$settings.style.display = 'none';
  1007. }
  1008.  
  1009. static toggleSettingsUi() {
  1010. const display = StreamStats.#$settings.style.display;
  1011. StreamStats.#$settings.style.display = display === 'block' ? 'none' : 'block';
  1012. screenClicktohide();
  1013. if (naifeitian.isDivTopOrBottomOutOfBounds(".better-xcloud-stats-settings")) {
  1014. $(".better-xcloud-stats-settings").css("height", "90%");
  1015. } else {
  1016. $(".better-xcloud-stats-settings").css("height", "");
  1017. }
  1018. }
  1019.  
  1020. static render() {
  1021. if (StreamStats.#$container) {
  1022. return;
  1023. }
  1024.  
  1025. const CE = createElement;
  1026. StreamStats.#$container = CE('div', { 'class': 'better-xcloud-stats-bar better-xcloud-gone' },
  1027. CE('div', { 'class': 'stats_fps' }, CE('label', {}, '帧率'),
  1028. StreamStats.#$fps = CE('span', {}, 0)),
  1029. CE('div', { 'class': 'stats_rtt' }, CE('label', {}, '延迟'),
  1030. StreamStats.#$rtt = CE('span', {}, '0ms')),
  1031. CE('div', { 'class': 'stats_dt' }, CE('label', {}, '解码'),
  1032. StreamStats.#$dt = CE('span', {}, '0ms')),
  1033. CE('div', { 'class': 'stats_br' }, CE('label', {}, '码率'),
  1034. StreamStats.#$br = CE('span', {}, '0ms')),
  1035. CE('div', { 'class': 'stats_pl' }, CE('label', {}, '丢包'),
  1036. StreamStats.#$pl = CE('span', {}, '0ms')),
  1037. CE('div', { 'class': 'stats_fl' }, CE('label', {}, '丢帧'),
  1038. StreamStats.#$fl = CE('span', {}, '0ms')));
  1039.  
  1040. let clicked_count = 0;
  1041. StreamStats.#$container.addEventListener('ontouchstart' in document ? 'touchstart' : 'mousedown', function (e) {
  1042. clicked_count++;
  1043. setTimeout(function () {
  1044. clicked_count = 0;
  1045. }, 500);
  1046.  
  1047. if (clicked_count > 1) {
  1048. //双击
  1049. StreamStats.toggleSettingsUi();
  1050. clicked_count = 0;
  1051. }
  1052. }, false);
  1053.  
  1054. document.documentElement.appendChild(StreamStats.#$container);
  1055.  
  1056. const refreshFunc = e => {
  1057. StreamStats.#refreshStyles()
  1058. };
  1059. const $position = naifeitian.toElement(NFTconfig['STATS_POSITION'], refreshFunc);
  1060.  
  1061. let $open_button;
  1062. const $showStartup = naifeitian.toElement(NFTconfig['STATS_SHOW_WHEN_PLAYING'], refreshFunc);
  1063. const $transparent = naifeitian.toElement(NFTconfig['STATS_TRANSPARENT'], refreshFunc);
  1064. const $formatting = naifeitian.toElement(NFTconfig['STATS_CONDITIONAL_FORMATTING'], refreshFunc);
  1065. const $opacity = naifeitian.toElement(NFTconfig['STATS_OPACITY'], refreshFunc);
  1066. const $textSize = naifeitian.toElement(NFTconfig['STATS_TEXT_SIZE'], refreshFunc);
  1067. const $slideopen = naifeitian.toElement(NFTconfig['STATS_SLIDE_OPEN'], refreshFunc);
  1068. const $stats_info = naifeitian.toElement(NFTconfig['stats_info'], refreshFunc);
  1069. StreamStats.#$settings = CE('div', { 'class': 'better-xcloud-stats-settings' },
  1070. CE('b', {}, '流监控设置'),
  1071. CE('div', {},
  1072. CE('label', { 'for': `xcloud_setting_NFTconfig['STATS_SHOW_WHEN_PLAYING']` }, '自启动'),
  1073. $showStartup
  1074. ),
  1075. CE('div', {},
  1076. CE('label', {}, '位置'),
  1077. $position
  1078. ),
  1079. CE('div', {},
  1080. CE('label', {}, '统计信息'),
  1081. $stats_info
  1082. ),
  1083. CE('div', {},
  1084. CE('label', {}, '字体大小'),
  1085. $textSize
  1086. ),
  1087. CE('div', {},
  1088. CE('label', { 'for': `xcloud_setting_STATS_OPACITY` }, '透明度 (10-100%)'),
  1089. $opacity
  1090. ),
  1091. CE('div', {},
  1092. CE('label', { 'for': `xcloud_setting_STATS_TRANSPARENT` }, '背景透明'),
  1093. $transparent
  1094. ),
  1095. CE('div', {},
  1096. CE('label', { 'for': `xcloud_setting_STATS_CONDITIONAL_FORMATTING` }, '数值颜色'),
  1097. $formatting
  1098. ),
  1099. CE('div', {},
  1100. CE('label', { 'for': `xcloud_setting_STATS_SLIDE_OPEN` }, '仅悬浮窗展开时打开'),
  1101. $slideopen
  1102. ),
  1103. $open_button = CE('button', { 'id': 'xcloud_setting_STATS_BUTTON' }, '启动监控'));
  1104.  
  1105. $open_button.addEventListener('click', () => {
  1106.  
  1107. if (StreamStats.status()) {
  1108. //需关闭
  1109. StreamStats.stop();
  1110.  
  1111. } else {
  1112. //需启动
  1113. StreamStats.start();
  1114.  
  1115. }
  1116.  
  1117. });
  1118. document.documentElement.appendChild(StreamStats.#$settings);
  1119.  
  1120. let stats_info_sortedEntries = Object.entries(NFTconfig['stats_info']).sort((a, b) => a[1][1] - b[1][1]);
  1121.  
  1122. let infos = [];
  1123.  
  1124. stats_info_sortedEntries.forEach(entry => {
  1125.  
  1126. //entry[1][0] 是否选中
  1127. //entry[1][1] 顺序
  1128. //entry[1][2] 名字
  1129. let tempDom = $(".stats_" + entry[0]).clone();
  1130.  
  1131. if (entry[1][0]) {
  1132. tempDom.css("display", "block");
  1133. } else {
  1134. tempDom.css("display", "none");
  1135. }
  1136. tempDom.css("border-right", "2px solid #fff");
  1137. infos.push(tempDom);
  1138. $(".stats_" + entry[0]).remove();
  1139.  
  1140. });
  1141. infos.forEach(function (item, index, array) {
  1142. $(".better-xcloud-stats-bar").append(item);
  1143. });
  1144. $('.better-xcloud-stats-bar > *')
  1145. .filter(function () {
  1146. return $(this).css('display') === 'block';
  1147. })
  1148. .last().css("border-right", '0px');
  1149.  
  1150.  
  1151. //流统计信息事件
  1152.  
  1153. let draggedItemIndex = -1;
  1154. let draggedItem;
  1155. let draggedItemClone; // 声明一个变量来存储被拖拽元素的副本
  1156. let isClick = false;
  1157. let isTouchstart = false;
  1158. let touchTimeout;
  1159. // 当拖拽开始时,添加拖拽中的样式,并设置拖拽数据
  1160. $('.drag-handle').on('dragstart', function (event) {
  1161. draggedItem = $(this);
  1162. draggedItemIndex = draggedItem.index();
  1163.  
  1164. });
  1165.  
  1166. // 当拖拽对象在可放置区域上方移动时触发
  1167. $('.drag-handle').on('dragover', function (event) {
  1168. event.preventDefault();
  1169. $('.drag-handle').removeClass('drag-over');
  1170.  
  1171. $(this).addClass('drag-over');
  1172.  
  1173. var placeholder = $('.placeholder');
  1174. var target = $(this);
  1175.  
  1176. if (event.originalEvent.clientY < target.offset().top + target.outerHeight() / 2) {
  1177. placeholder.insertBefore(target);
  1178. } else {
  1179. placeholder.insertAfter(target);
  1180. }
  1181. if (!placeholder.prev().hasClass("dragging") && !placeholder.next().hasClass("dragging")) {
  1182. placeholder.show();
  1183. }
  1184.  
  1185. });
  1186.  
  1187. // 当拖拽对象离开可放置区域时触发
  1188. $('.drag-handle').on('dragleave', function (event) {
  1189. $(this).removeClass('drag-over');
  1190.  
  1191. });
  1192.  
  1193. // 当拖拽对象被松开触发
  1194. $('.drag-handle').on('dragend', function (event) {
  1195.  
  1196. var droppedIndex = $('.placeholder').index();
  1197.  
  1198. var draggedItem = $('.drag-handle').eq(draggedItemIndex);
  1199. draggedItem.insertBefore($('.placeholder'));
  1200.  
  1201. draggedItem.removeClass('dragging').hide().fadeIn();
  1202.  
  1203.  
  1204.  
  1205. $('.drag-handle').removeClass('drag-over');
  1206. $('.drag-handle').removeClass('dragging');
  1207.  
  1208. $('.placeholder').hide();
  1209. draggedItemClone?.remove(); // 移除被拖拽元素的副本
  1210.  
  1211. let drag_index = 0;
  1212. let infos = [];
  1213. $('.drag-handle').each(function (index, d) {
  1214. let name = $(d).attr("data-name");
  1215. if (name != undefined) {
  1216. drag_index = drag_index + 1;
  1217. NFTconfig['stats_info'][name][1] = drag_index;
  1218. $(this).attr("data-index", drag_index);
  1219. let tempDom = $(".stats_" + name).clone();
  1220.  
  1221. if (!NFTconfig['stats_info'][name][0]) {
  1222. //隐藏
  1223. tempDom.css("display", "none");
  1224. } else {
  1225. tempDom.css("display", "block");
  1226. }
  1227. tempDom.css("border-right", "2px solid #fff");
  1228. infos.push(tempDom);
  1229. $(".stats_" + name).remove();
  1230. }
  1231. });
  1232. naifeitian.setValue("stats_infoGM", NFTconfig['stats_info']);
  1233.  
  1234. infos.forEach(function (item, index, array) {
  1235. $(".better-xcloud-stats-bar").append(item);
  1236. });
  1237. $('.better-xcloud-stats-bar > *')
  1238. .filter(function () {
  1239. return $(this).css('display') === 'block';
  1240. })
  1241. .last().css("border-right", '0px');
  1242.  
  1243.  
  1244. });
  1245.  
  1246.  
  1247. // 添加触摸事件
  1248. $('.drag-handle').on('touchstart', function (event) {
  1249. isTouchstart = true;
  1250. draggedItem = $(this);
  1251. draggedItemIndex = $(this).index();
  1252.  
  1253. touchTimeout = setTimeout(function () {
  1254. isClick = false;
  1255. if ($(".dragged-copy-item").length == 0) {
  1256. // 创建被拖拽元素的副本
  1257. draggedItemClone = draggedItem.clone().addClass('stats-container dragged-copy-item');
  1258. $('.better-xcloud-stats-settings').after(draggedItemClone);
  1259. } else {
  1260. draggedItemClone = $(".dragged-copy-item");
  1261. }
  1262.  
  1263. }, 200);
  1264. isClick = true;
  1265. });
  1266.  
  1267. $('.stats-container').on('touchmove', function (event) {
  1268. event.preventDefault();
  1269. if (isClick) { return; }
  1270. draggedItem.addClass("dragging");
  1271.  
  1272. $(".dragged-copy-item").css("display", "block");
  1273.  
  1274. const touch = event.originalEvent.touches[0] || event.originalEvent.changedTouches[0];
  1275.  
  1276. var previousElement = $(".better-xcloud-stats-settings");
  1277.  
  1278. // 计算右边界位置
  1279. var rightEdgeOfPrevious = previousElement.offset().left + previousElement.outerWidth() - 100;
  1280.  
  1281. // 更新被拖拽元素的副本位置
  1282. draggedItemClone.css({
  1283. top: touch.clientY - draggedItemClone.outerHeight() / 2 + 'px',
  1284. left: rightEdgeOfPrevious + 'px'
  1285. });
  1286.  
  1287. $('.drag-handle').each(function () {
  1288. if ($(this).hasClass("dragged-copy-item")) { return }
  1289. const itemOffset = $(this).offset().top;
  1290. const itemHeight = $(this).outerHeight();
  1291. if (draggedItem.is($(this))) return true;
  1292. let placeholder = $('.placeholder');
  1293. if (touch.clientY > itemOffset && touch.clientY < itemOffset + itemHeight) {
  1294. $(this).addClass('drag-over');
  1295.  
  1296.  
  1297. if (touch.clientY < itemOffset + itemHeight / 4) {
  1298. placeholder.insertBefore($(this));
  1299. } else {
  1300. placeholder.insertAfter($(this));
  1301. }
  1302. if (!placeholder.prev().hasClass("dragging") && !placeholder.next().hasClass("dragging")) {
  1303. placeholder.css("display", "block");
  1304. }
  1305.  
  1306. return false;
  1307. } else {
  1308. $(this).removeClass("drag-over");
  1309.  
  1310. }
  1311. });
  1312. });
  1313.  
  1314. $('.stats-container').on('touchend', function (event) {
  1315. isTouchstart = false;
  1316. if (isClick) { return; }
  1317. var droppedIndex = $('.placeholder').index();
  1318.  
  1319. var draggedItem = $('.drag-handle').eq(draggedItemIndex);
  1320. draggedItem.insertBefore($('.placeholder'));
  1321.  
  1322. draggedItem.removeClass('dragging').hide().fadeIn();
  1323.  
  1324. $('.drag-handle').removeClass('drag-over');
  1325. $('.drag-handle').removeClass('dragging');
  1326.  
  1327. $('.placeholder').hide();
  1328. draggedItemClone.remove();
  1329.  
  1330. let drag_index = 0;
  1331. let infos = [];
  1332. $('.drag-handle').each(function (index, d) {
  1333. let name = $(d).attr("data-name");
  1334. if (name != undefined) {
  1335. drag_index = drag_index + 1;
  1336. NFTconfig['stats_info'][name][1] = drag_index;
  1337. $(this).attr("data-index", drag_index);
  1338. let tempDom = $(".stats_" + name).clone();
  1339. if (!NFTconfig['stats_info'][name][0]) {
  1340. //隐藏
  1341. tempDom.css("display", "none");
  1342. } else {
  1343. tempDom.css("display", "block");
  1344. }
  1345. tempDom.css("border-right", "2px solid #fff");
  1346. infos.push(tempDom);
  1347. $(".stats_" + name).remove();
  1348. }
  1349. });
  1350. naifeitian.setValue("stats_infoGM", NFTconfig['stats_info']);
  1351.  
  1352. infos.forEach(function (item, index, array) {
  1353. $(".better-xcloud-stats-bar").append(item);
  1354. });
  1355. $('.better-xcloud-stats-bar > *')
  1356. .filter(function () {
  1357. return $(this).css('display') === 'block';
  1358. })
  1359. .last().css("border-right", '0px');
  1360. });
  1361.  
  1362. // 点击事件,选中/取消选中按钮
  1363. $('.drag-handle').on('click', function (event) {
  1364. if (!isClick) {
  1365. if (isTouchstart) {
  1366. return;
  1367. }
  1368. }
  1369. if ($(this).hasClass('stats-selected')) {
  1370. //取消
  1371. $(this).removeClass('stats-selected');
  1372. $(this).addClass('stats-delete');
  1373. NFTconfig['stats_info'][$(this).attr("data-name")][0] = false;
  1374.  
  1375. } else {
  1376. //启用
  1377. $(this).addClass('stats-selected');
  1378. $(this).removeClass('stats-delete');
  1379. NFTconfig['stats_info'][$(this).attr("data-name")][0] = true;
  1380. }
  1381. naifeitian.setValue("stats_infoGM", NFTconfig['stats_info']);
  1382.  
  1383. let stats_info_sortedEntries = Object.entries(NFTconfig['stats_info']).sort((a, b) => a[1][1] - b[1][1]);
  1384.  
  1385. stats_info_sortedEntries.forEach(entry => {
  1386. //entry[1][0] 是否选中
  1387. //entry[1][1] 顺序
  1388. //entry[1][2] 名字
  1389. if (entry[1][0]) {
  1390. $(".stats_" + entry[0]).css("display", "block");
  1391. } else {
  1392. $(".stats_" + entry[0]).css("display", "none");
  1393. }
  1394. $(".stats_" + entry[0]).css("border-right", "2px solid #fff");
  1395. })
  1396. $('.better-xcloud-stats-bar > *')
  1397. .filter(function () {
  1398. return $(this).css('display') === 'block';
  1399. })
  1400. .last().css("border-right", '0px');
  1401.  
  1402. });
  1403.  
  1404. StreamStats.#refreshStyles();
  1405. }
  1406. }
  1407. function numberPicker(key, suffix = '', disabled = false, range = true) {
  1408.  
  1409. const setting = key.name;
  1410. let value = key.default;
  1411. let $text, $decBtn, $incBtn, $range;
  1412.  
  1413. const MIN = key.min;
  1414. const MAX = key.max;
  1415.  
  1416. const CE = createElement;
  1417. const $wrapper = CE('div', {},
  1418. $decBtn = CE('button', { 'data-type': 'dec' }, '-'),
  1419. $text = CE('span', {}, value + suffix),
  1420. $incBtn = CE('button', { 'data-type': 'inc' }, '+'),
  1421. );
  1422. if (range) {
  1423. $range = CE('input', { 'type': 'range', 'style': "width:100px", 'min': 0, 'max': 150, 'value': value });
  1424. $range.addEventListener('input', e => {
  1425. value = parseInt(e.target.value);
  1426.  
  1427. $text.textContent = value + "%";
  1428.  
  1429. key['default'] = value;
  1430.  
  1431. naifeitian.setValue(key['name'], key);
  1432.  
  1433. updateVideoPlayerCss();
  1434.  
  1435.  
  1436. });
  1437. $wrapper.appendChild($range);
  1438.  
  1439. }
  1440. if (disabled) {
  1441. $incBtn.disabled = true;
  1442. $incBtn.classList.add('better-xcloud-hidden');
  1443.  
  1444. $decBtn.disabled = true;
  1445. $decBtn.classList.add('better-xcloud-hidden');
  1446. return $wrapper;
  1447. }
  1448.  
  1449. let interval;
  1450. let isHolding = false;
  1451.  
  1452. const onClick = e => {
  1453. if (isHolding) {
  1454. e.preventDefault();
  1455. isHolding = false;
  1456.  
  1457. return;
  1458. }
  1459.  
  1460. const btnType = e.target.getAttribute('data-type');
  1461. if (btnType === 'dec') {
  1462. value = (value <= MIN) ? MIN : value - 1;
  1463.  
  1464. } else {
  1465. value = (value >= MAX) ? MAX : value + 1;
  1466. }
  1467. $($range).val(value);
  1468. $text.textContent = value + suffix;
  1469.  
  1470. key['default'] = value;
  1471.  
  1472. naifeitian.setValue(key['name'], key);
  1473.  
  1474. updateVideoPlayerCss();
  1475.  
  1476. isHolding = false;
  1477. }
  1478.  
  1479. const onMouseDown = e => {
  1480. isHolding = true;
  1481.  
  1482. const args = arguments;
  1483. interval = setInterval(() => {
  1484. const event = new Event('click');
  1485. event.arguments = args;
  1486.  
  1487. e.target.dispatchEvent(event);
  1488. }, 200);
  1489. };
  1490.  
  1491. const onMouseUp = e => {
  1492. clearInterval(interval);
  1493. isHolding = false;
  1494. };
  1495.  
  1496. $decBtn.addEventListener('click', onClick);
  1497. $decBtn.addEventListener('mousedown', onMouseDown);
  1498. $decBtn.addEventListener('mouseup', onMouseUp);
  1499. $decBtn.addEventListener('touchstart', onMouseDown);
  1500. $decBtn.addEventListener('touchend', onMouseUp);
  1501.  
  1502. $incBtn.addEventListener('click', onClick);
  1503. $incBtn.addEventListener('mousedown', onMouseDown);
  1504. $incBtn.addEventListener('mouseup', onMouseUp);
  1505. $incBtn.addEventListener('touchstart', onMouseDown);
  1506. $incBtn.addEventListener('touchend', onMouseUp);
  1507.  
  1508. return $wrapper;
  1509. }
  1510.  
  1511. function setupVideoSettingsBar() {
  1512. const CE = createElement;
  1513. let $stretchInp;
  1514. const refreshFunc = e => {
  1515. updateVideoPlayerCss();
  1516. };
  1517. const $stretch = naifeitian.toElement(NFTconfig['video_stretch'], refreshFunc);
  1518. const $wrapper = CE('div', { 'class': 'better-xcloud-quick-settings-bar' },
  1519. CE('div', {},
  1520. CE('label', { 'for': 'better-xcloud-quick-setting-stretch' }, '去黑边'),
  1521. $stretch),
  1522. CE('div', {},
  1523. CE('label', {}, '清晰'),
  1524. numberPicker(NFTconfig['VIDEO_CLARITY'], '', naifeitian.isSafari(), false)),
  1525. CE('div', {},
  1526. CE('label', {}, '饱和'),
  1527. numberPicker(NFTconfig['VIDEO_SATURATION'], '%')),
  1528. CE('div', {},
  1529. CE('label', {}, '对比'),
  1530. numberPicker(NFTconfig['VIDEO_CONTRAST'], '%')),
  1531. CE('div', {},
  1532. CE('label', {}, '亮度'),
  1533. numberPicker(NFTconfig['VIDEO_BRIGHTNESS'], '%'))
  1534. );
  1535.  
  1536.  
  1537. $stretch.addEventListener('change', e => {
  1538. if (e.target.value == 'setting') {
  1539. $('#video_stretch_x_y').css('display', 'block');
  1540. } else {
  1541. $('#video_stretch_x_y').css('display', 'none');
  1542. }
  1543. NFTconfig['video_stretch'].default = e.target.value;
  1544. naifeitian.setValue('video_stretchGM', NFTconfig['video_stretch']);
  1545. updateVideoPlayerCss();
  1546. });
  1547.  
  1548. document.documentElement.appendChild($wrapper);
  1549. if ($stretch.id == 'xcloud_setting_video_stretchGM') {
  1550. let dom = $('#xcloud_setting_video_stretchGM');
  1551. dom.after(`<div id="video_stretch_x_y" style="display: ${NFTconfig['video_stretch'].default == 'setting' ? 'block' : 'none'}">
  1552. <lable>左右
  1553. <input type=\'text\'class="video_stretch_x_y_Listener" id="video_stretch_x" style="width:35px" value="${NFTconfig['video_stretch_x_y']['x']}"/>
  1554. </lable><br/>
  1555. <lable>上下
  1556. <input type=\'text\'class="video_stretch_x_y_Listener" id="video_stretch_y" style="width:35px" value="${NFTconfig['video_stretch_x_y']['y']}"/>
  1557. </lable>
  1558. </div>`);
  1559.  
  1560. $(document).on('blur', '.video_stretch_x_y_Listener', function () {
  1561. let newval = $(this).val();
  1562. if (naifeitian.isNumber($(this).val())) {
  1563. if ($(this).attr('id') == 'video_stretch_x') {
  1564. NFTconfig['video_stretch_x_y']['x'] = newval;
  1565. naifeitian.setValue('video_stretch_x_yGM', NFTconfig['video_stretch_x_y']);
  1566. } else {
  1567. NFTconfig['video_stretch_x_y']['y'] = newval;
  1568. naifeitian.setValue('video_stretch_x_yGM', NFTconfig['video_stretch_x_y']);
  1569. }
  1570. } else {
  1571. $(this).val("0");
  1572. NFTconfig['video_stretch_x_y']['x'] = 0;
  1573. NFTconfig['video_stretch_x_y']['y'] = 0;
  1574. naifeitian.setValue('video_stretch_x_yGM', NFTconfig['video_stretch_x_y']);
  1575. }
  1576. updateVideoPlayerCss();
  1577. });
  1578. }
  1579. }
  1580. function cloneStreamHudButton($orgButton, class_, label, svg_icon) {
  1581. const $container = $orgButton.cloneNode(true);
  1582. if (class_ != "") {
  1583. $($container).addClass(class_);
  1584. }
  1585.  
  1586.  
  1587. const $button = $container.querySelector('button');
  1588. $button.setAttribute('title', label);
  1589.  
  1590. const $svg = $button.querySelector('svg');
  1591. $svg.innerHTML = svg_icon;
  1592.  
  1593. const attrs = {
  1594. 'fill-rule': 'evenodd',
  1595. 'stroke-linecap': 'round',
  1596. 'stroke-linejoin': 'round',
  1597. 'stroke-width': 2,
  1598. 'viewBox': '0 0 32 32'
  1599. };
  1600.  
  1601. for (const attr in attrs) {
  1602. $svg.setAttribute(attr, attrs[attr]);
  1603. }
  1604.  
  1605. return $container;
  1606. }
  1607. function cloneStreamMenuButton($orgButton, label, svg_icon) {
  1608. const $button = $orgButton.cloneNode(true);
  1609. $button.setAttribute('aria-label', label);
  1610. $button.querySelector('div[class*=label]').textContent = label;
  1611.  
  1612. const $svg = $button.querySelector('svg');
  1613. $svg.innerHTML = svg_icon;
  1614. $svg.setAttribute('viewBox', '0 0 32 32');
  1615.  
  1616. return $button;
  1617. }
  1618.  
  1619. function HookProperty(object, property, value) {
  1620. Object.defineProperty(object, property, {
  1621. value: value
  1622. });
  1623. }
  1624.  
  1625. let userAgentOriginal = window.navigator.userAgent;
  1626. try {
  1627. HookProperty(window.navigator, "userAgent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/999.0.0.0 Safari/537.36 Edg/999.0.0.0");
  1628. HookProperty(window.navigator, "maxTouchPoints", 10);
  1629. if (NFTconfig['disableCheckNetwork'] == 1) {
  1630. Object.defineProperty(window.navigator, 'connection', {
  1631. get: () => undefined,
  1632. });
  1633. }
  1634. HookProperty(window.navigator, "standalone", true);
  1635.  
  1636. } catch (e) { }
  1637. let consoleIp;
  1638. let consolePort;
  1639. const patchIceCandidates = function (...arg) {
  1640. // ICE server candidates
  1641. const request = arg[0];
  1642. const url = (typeof request === 'string') ? request : request.url;
  1643.  
  1644. if (url && url.endsWith('/ice') && url.includes('/sessions/') && request.method === 'GET') {
  1645. const promise = originFetch(...arg);
  1646.  
  1647. return promise.then(response => {
  1648. return response.clone().text().then(text => {
  1649. if (!text.length) {
  1650. return response;
  1651. }
  1652.  
  1653. const options = {
  1654. preferIpv6Server: NFTconfig['IPv6'] == 1,
  1655. consoleIp: consoleIp,
  1656. };
  1657.  
  1658. const obj = JSON.parse(text);
  1659. let exchangeResponse = JSON.parse(obj.exchangeResponse);
  1660. exchangeResponse = updateIceCandidates(exchangeResponse, options)
  1661. obj.exchangeResponse = JSON.stringify(exchangeResponse);
  1662.  
  1663. response.json = () => Promise.resolve(obj);
  1664. response.text = () => Promise.resolve(JSON.stringify(obj));
  1665.  
  1666. return response;
  1667. });
  1668. });
  1669. }
  1670.  
  1671. return null;
  1672. }
  1673. const originFetch = window.fetch;
  1674. window.fetch = async (...arg) => {
  1675. let arg0 = arg[0];
  1676. let url = "";
  1677. let isRequest = false;
  1678. switch (typeof arg0) {
  1679. case "object":
  1680. url = arg0.url;
  1681. isRequest = true;
  1682. break;
  1683. case "string":
  1684. url = arg0;
  1685. break;
  1686. default:
  1687. break;
  1688. }
  1689.  
  1690. // 串流
  1691. if (IS_REMOTE_PLAYING && url.includes('/sessions/home') || url.includes('inputconfigs')) {
  1692.  
  1693. canShowOC = true;
  1694. const clone = arg0.clone();
  1695.  
  1696. const headers = {};
  1697. for (const pair of clone.headers.entries()) {
  1698. headers[pair[0]] = pair[1];
  1699. }
  1700. headers['authorization'] = `Bearer ${RemotePlay.XHOME_TOKEN}`;
  1701.  
  1702. const deviceInfo = RemotePlay.BASE_DEVICE_INFO;
  1703. if (NFTconfig['REMOTE_PLAY_RESOLUTION']['default'] == '720p') {
  1704. deviceInfo.dev.os.name = 'android';
  1705. }
  1706.  
  1707. headers['x-ms-device-info'] = JSON.stringify(deviceInfo);
  1708.  
  1709. const opts = {
  1710. method: clone.method,
  1711. headers: headers,
  1712. };
  1713.  
  1714. if (clone.method === 'POST') {
  1715. opts.body = await clone.text();
  1716. }
  1717.  
  1718. const index = arg0.url.indexOf('.xboxlive.com');
  1719. let newUrl = `https://${REMOTE_PLAY_SERVER}.core.gssv-play-prodxhome` + arg0.url.substring(index);
  1720.  
  1721. arg0 = new Request(newUrl, opts);
  1722.  
  1723. arg[0] = arg0;
  1724. url = (typeof request === 'string') ? arg0 : arg0.url;
  1725. // Get console IP
  1726. if (url.includes('/configuration')) {
  1727. const promise = originFetch(...arg);
  1728.  
  1729. return promise.then(response => {
  1730. return response.clone().json().then(obj => {
  1731. console.log(obj);
  1732. consoleIp = obj.serverDetails.ipAddress;
  1733. consolePort = obj.serverDetails.port;
  1734.  
  1735. response.json = () => Promise.resolve(obj);
  1736.  
  1737. return response;
  1738. });
  1739. });
  1740. }
  1741. if (url.includes('/sessions/home/play')) {
  1742. inGame();
  1743. document.title = document.title.replace('Fortnite', '串流');
  1744. }
  1745.  
  1746. return patchIceCandidates(...arg) || originFetch(...arg);
  1747.  
  1748. }
  1749. if (IS_REMOTE_PLAYING && url.includes('/login/user')) {
  1750. try {
  1751. const clone = arg0.clone();
  1752.  
  1753. const obj = await clone.json();
  1754. obj.offeringId = 'xhome';
  1755.  
  1756. arg0 = new Request('https://xhome.core.gssv-play-prod.xboxlive.com/v2/login/user', {
  1757. method: 'POST',
  1758. body: JSON.stringify(obj),
  1759. headers: {
  1760. 'Content-Type': 'application/json',
  1761. },
  1762. });
  1763.  
  1764. arg[0] = arg0;
  1765. } catch (e) {
  1766. alert(e);
  1767. console.log(e);
  1768. }
  1769.  
  1770. return originFetch(...arg);
  1771. }
  1772. // ICE server candidates
  1773. const patchedIpv6 = patchIceCandidates(...arg);
  1774. if (patchedIpv6) {
  1775. return patchedIpv6;
  1776. }
  1777. if (!url.includes('xhome.') && url.indexOf('/v2/login/user') > -1) {//xgpuweb.gssv-play-prod.xboxlive.com
  1778. checkIpsuc = true;
  1779.  
  1780.  
  1781. let json = await arg0.json();
  1782. let body = JSON.stringify(json);
  1783. // 处理免代理逻辑
  1784. if (NFTconfig['no_need_VPN_play'] == 1) {
  1785.  
  1786. if (NFTconfig['useCustomfakeIp'] == 1 && naifeitian.isValidIP(NFTconfig['customfakeIp'])) {
  1787. arg0.headers.set('x-forwarded-for', NFTconfig['customfakeIp']);
  1788. } else {
  1789. arg0.headers.set('x-forwarded-for', NFTconfig['regionBlock']['options'][NFTconfig['regionBlock']['blockIp']]);
  1790. }
  1791. }
  1792.  
  1793. arg[0] = new Request(url, {
  1794. method: arg0.method,
  1795. headers: arg0.headers,
  1796. body: body,
  1797. });
  1798.  
  1799. const promise = originFetch(...arg);
  1800. if (NFTconfig['useCustomfakeIp'] == 1 && naifeitian.isValidIP(NFTconfig['customfakeIp'])) {
  1801. console.log('免代理成功,已设置为自定义IP【' + NFTconfig['customfakeIp'] + "】");
  1802. } else {
  1803. console.log('免代理成功,已设置为【' + NFTconfig['regionBlock']['blockIp'] + "】");
  1804. }
  1805. RemotePlay.preload();
  1806. return promise.then(response => {
  1807. return response.clone().json().then(json => {
  1808. //获取服务器列表
  1809. let newServerList = [];
  1810. let currentAutoServer;
  1811. let REMOTE_SERVER_LIST = [];
  1812. json["offeringSettings"]["regions"].forEach((region) => {
  1813. REMOTE_SERVER_LIST.push(region.networkTestHostname.split(".")[0]);
  1814. newServerList.push(region["name"]);
  1815. if (region["isDefault"] === true) {
  1816. currentAutoServer = region["name"];
  1817. }
  1818. });
  1819. NFTconfig['REMOTE_SERVER_LIST'] = REMOTE_SERVER_LIST;
  1820. naifeitian.setValue("REMOTE_SERVER_LISTGM", REMOTE_SERVER_LIST);
  1821. naifeitian.setValue("blockXcloudServerListGM", newServerList);
  1822. NFTconfig['blockXcloudServerList'] = newServerList;
  1823. if (NFTconfig['blockXcloudServerList'].indexOf(NFTconfig['defaultXcloudServer']) == -1) {
  1824. naifeitian.setValue("defaultXcloudServerGM", "");
  1825. NFTconfig['defaultXcloudServer'] = "";
  1826. NFTconfig['blockXcloudServer'] = 0;
  1827. naifeitian.setValue("blockXcloudServerGM", 0);
  1828. }
  1829. if (NFTconfig['blockXcloudServer'] == 1) {
  1830. console.log('修改服务器开始');
  1831. json["offeringSettings"]["allowRegionSelection"] = true;
  1832. let selectedServer = NFTconfig['defaultXcloudServer'];
  1833. if (selectedServer !== "Auto" && newServerList.includes(selectedServer)) {
  1834. json["offeringSettings"]["regions"].forEach((region) => {
  1835. if (region["name"] === selectedServer) {
  1836. region["isDefault"] = true;
  1837. } else {
  1838. region["isDefault"] = false;
  1839. }
  1840. });
  1841. }
  1842. console.log('修改服务器结束');
  1843. }
  1844. try {
  1845. json["offeringSettings"]["regions"].forEach((region) => {
  1846. if (region.isDefault) {
  1847. StreamBadges.region = region.name;
  1848. throw new Error();
  1849. }
  1850. });
  1851.  
  1852. } catch (e) { }
  1853.  
  1854. response.json = () => Promise.resolve(json);
  1855. return response;
  1856. });
  1857. });
  1858. } else if (url.indexOf('/cloud/play') > -1) {
  1859.  
  1860. inGame();
  1861. const clone = arg0.clone();
  1862. const body = await clone.json();
  1863. if (NFTconfig['chooseLanguage'] == 1) {
  1864. let selectedLanguage = NFTconfig['xcloud_game_language'];
  1865. if (selectedLanguage == 'Auto') {
  1866. let parts = window.location.pathname.split('/');
  1867. let pid = parts[parts.length - 1];
  1868. try {
  1869. let res = await fetch(
  1870. "https://catalog.gamepass.com/products?market=US&language=en-US&hydration=PCInline", {
  1871. "headers": {
  1872. "content-type": "application/json;charset=UTF-8",
  1873. },
  1874. "body": "{\"Products\":[\"" + pid + "\"]}",
  1875. "method": "POST",
  1876. "mode": "cors",
  1877. "credentials": "omit"
  1878. });
  1879. let jsonObj = await res.json();
  1880. let languageSupport = jsonObj["Products"][pid]["LanguageSupport"]
  1881. for (let language of Object.keys(default_language_list)) {
  1882. if (default_language_list[language] in languageSupport) {
  1883. selectedLanguage = default_language_list[language];
  1884. break;
  1885. }
  1886. }
  1887. if (selectedLanguage == 'Auto') {
  1888. //防止接口没有返回支持语言
  1889. selectedLanguage = NFTconfig['IfErrUsedefaultGameLanguage'];
  1890. console.log("使用次选语言");
  1891. }
  1892.  
  1893. } catch (e) {
  1894. }
  1895. }
  1896. console.log('语言已设置:【' + selectedLanguage + '】');
  1897. body.settings.locale = selectedLanguage;
  1898. }
  1899. body.settings.osName = NFTconfig['high_bitrate'] == 1 ? 'windows' : 'android';
  1900.  
  1901. const newRequest = new Request(arg0, {
  1902. body: JSON.stringify(body),
  1903. });
  1904.  
  1905. arg[0] = newRequest;
  1906. return originFetch(...arg);
  1907.  
  1908.  
  1909. } else if (url.endsWith('/configuration') && url.includes('/sessions/cloud/') && NFTconfig['autoOpenOC'] == 1 && NFTconfig['disableTouchControls'] == 0) {
  1910. // Enable CustomTouchOverlay
  1911.  
  1912. return new Promise((resolve, reject) => {
  1913. originFetch(...arg).then(res => {
  1914. res.json().then(json => {
  1915. // console.error(json);
  1916. let inputOverrides = JSON.parse(json.clientStreamingConfigOverrides || '{}') || {};
  1917. inputOverrides.inputConfiguration = {
  1918. enableTouchInput: true,
  1919. maxTouchPoints: 10,
  1920. enableVibration: true
  1921. };
  1922. json.clientStreamingConfigOverrides = JSON.stringify(inputOverrides);
  1923. let cdom = $('#BabylonCanvasContainer-main').children();
  1924. if (cdom.length > 0) {
  1925. canShowOC = false;
  1926. } else {
  1927. canShowOC = true;
  1928. }
  1929. let body = JSON.stringify(json);
  1930. let newRes = new Response(body, {
  1931. status: res.status,
  1932. statusText: res.statusText,
  1933. headers: res.headers
  1934. })
  1935. resolve(newRes);
  1936.  
  1937. console.log('修改触摸成功')
  1938. }).catch(err => {
  1939. reject(err);
  1940. });
  1941. }).catch(err => {
  1942. reject(err);
  1943. });
  1944. });
  1945. } else {
  1946. return originFetch(...arg);
  1947. }
  1948. }
  1949. function updateIceCandidates(candidates, options) {
  1950. const pattern = new RegExp(/a=candidate:(?<foundation>\d+) (?<component>\d+) UDP (?<priority>\d+) (?<ip>[^\s]+) (?<the_rest>.*)/);
  1951.  
  1952. const lst = [];
  1953. for (let item of candidates) {
  1954. if (item.candidate == 'a=end-of-candidates') {
  1955. continue;
  1956. }
  1957.  
  1958. const groups = pattern.exec(item.candidate).groups;
  1959. lst.push(groups);
  1960. }
  1961.  
  1962. if (options.preferIpv6Server) {
  1963. lst.sort((a, b) => (!a.ip.includes(':') && b.ip.includes(':')) ? 1 : -1);
  1964. }
  1965.  
  1966. const newCandidates = [];
  1967. let foundation = 1;
  1968. lst.forEach(item => {
  1969. item.foundation = foundation;
  1970. item.priority = (foundation == 1) ? 10000 : 1;
  1971.  
  1972. newCandidates.push({
  1973. 'candidate': `a=candidate:${item.foundation} 1 UDP ${item.priority} ${item.ip} ${item.the_rest}`,
  1974. 'messageType': 'iceCandidate',
  1975. 'sdpMLineIndex': '0',
  1976. 'sdpMid': '0',
  1977. });
  1978.  
  1979. ++foundation;
  1980. });
  1981.  
  1982. if (options.consoleIp) {
  1983. newCandidates.push({
  1984. 'candidate': `a=candidate:${newCandidates.length + 1} 1 UDP 1 ${options.consoleIp} 9002 typ host`,
  1985. 'messageType': 'iceCandidate',
  1986. 'sdpMLineIndex': '0',
  1987. 'sdpMid': '0',
  1988. });
  1989. }
  1990.  
  1991. newCandidates.push({
  1992. 'candidate': 'a=end-of-candidates',
  1993. 'messageType': 'iceCandidate',
  1994. 'sdpMLineIndex': '0',
  1995. 'sdpMid': '0',
  1996. });
  1997.  
  1998. return newCandidates;
  1999. }
  2000. function checkCodec() {
  2001.  
  2002. let rtcCodecPreferences = naifeitian.getValue('rtcCodecPreferencesGM');
  2003. let codecs = RTCRtpReceiver.getCapabilities('video').codecs;
  2004. let codesOptions = ['默认', '自动'];
  2005. const codecProfileMap = { "高": "4d", "中": "42e", "低": "420" };
  2006. codecs.forEach((codec, index) => {
  2007. if (codec.mimeType === 'video/H264') {
  2008. for (let key in codecProfileMap) {
  2009. if (codec.sdpFmtpLine.includes(codecProfileMap[key])) {
  2010. codesOptions.push(codec.mimeType.substring(6) + key);
  2011. break;
  2012. }
  2013. }
  2014. } else {
  2015. codesOptions.push(codec.mimeType.substring(6));
  2016. }
  2017. });
  2018.  
  2019. codesOptions = [...new Set(codesOptions)];
  2020.  
  2021. let sortOrder = ['默认', '自动', 'AV1', 'VP9', 'H265', 'VP8', 'H264高', 'H264中', 'H264低', 'flexfec-03', 'ulpfec', 'rtx', 'red'];
  2022. const customSort = (a, b) => {
  2023. const indexOfA = sortOrder.indexOf(a);
  2024. const indexOfB = sortOrder.indexOf(b);
  2025.  
  2026. if (indexOfA === -1) {
  2027. return 1;
  2028. }
  2029. if (indexOfB === -1) {
  2030. return -1;
  2031. }
  2032. return indexOfA - indexOfB;
  2033. };
  2034. codesOptions.sort(customSort);
  2035. rtcCodecPreferences['options'] = codesOptions;
  2036.  
  2037. if (!rtcCodecPreferences['options'].includes(rtcCodecPreferences['default'])) {
  2038. rtcCodecPreferences['default'] = "默认";
  2039. }
  2040. NFTconfig['rtcCodecPreferences'] = rtcCodecPreferences;
  2041. naifeitian.setValue('rtcCodecPreferencesGM', rtcCodecPreferences);
  2042. }
  2043. checkCodec();
  2044.  
  2045.  
  2046.  
  2047. if (NFTconfig['autoOpenOC'] == 1 && NFTconfig['disableTouchControls'] == 0 && NFTconfig['autoShowTouch']) {
  2048. window.RTCPeerConnection.prototype.originalCreateDataChannelGTC = window.RTCPeerConnection.prototype.createDataChannel;
  2049. window.RTCPeerConnection.prototype.createDataChannel = function (...params) {
  2050.  
  2051. const dc = this.originalCreateDataChannelGTC.apply(this, arguments);
  2052. if (dc.label == "message") {
  2053. dc.addEventListener('open', e => {
  2054. setTimeout(() => {
  2055. if (canShowOC) {
  2056. dc.dispatchEvent(new MessageEvent('message', {
  2057. data: '{"content":"{\\"layoutId\\":\\"\\"}","target":"/streaming/touchcontrols/showlayoutv2","type":"Message"}'
  2058. }));
  2059. }
  2060. }, 1000);
  2061. });
  2062. dc.addEventListener("message", function (de) {
  2063. if (typeof (de.data) != "string") { return; }
  2064. let msgdata = JSON.parse(de.data);
  2065. if (msgdata.target == "/streaming/touchcontrols/showtitledefault") {
  2066. let chkItv = setInterval(() => {
  2067. if (canShowOC == null) { return; }
  2068. clearInterval(chkItv);
  2069. if (canShowOC) {
  2070. dc.dispatchEvent(new MessageEvent('message', {
  2071. data: '{"content":"{\\"layoutId\\":\\"\\"}","target":"/streaming/touchcontrols/showlayoutv2","type":"Message"}'
  2072. }));
  2073. }
  2074.  
  2075. }, 1000);
  2076. }
  2077. });
  2078. }
  2079. return dc;
  2080.  
  2081. }
  2082.  
  2083. }
  2084.  
  2085. // 配置对象,定义每个设置项的信息
  2086. const settingsConfig = [
  2087. {
  2088. label: '选择语言:',
  2089. type: 'radio',
  2090. name: 'chooseLanguage',
  2091. display: 'block',
  2092. options: [
  2093. { value: 1, text: '开', id: 'chooseLanguageOn' },
  2094. { value: 0, text: '关', id: 'chooseLanguageOff' }
  2095. ],
  2096. checkedValue: NFTconfig['chooseLanguage'],
  2097. needHr: false
  2098. },
  2099. {
  2100. label: '首选语言:',
  2101. type: 'radio',
  2102. name: 'selectLanguage',
  2103. display: NFTconfig['chooseLanguage'] === 1 ? 'block' : 'none',
  2104. options: Object.keys(default_language_list).map(languageChinese => {
  2105. return {
  2106. value: default_language_list[languageChinese],
  2107. text: languageChinese,
  2108. id: default_language_list[languageChinese]
  2109. };
  2110. }),
  2111. checkedValue: NFTconfig['xcloud_game_language'],
  2112. needHr: false
  2113.  
  2114. },
  2115. {
  2116. label: '次选语言:',
  2117. type: 'radio',
  2118. name: 'IfErrUsedefaultGameLanguage',
  2119. display: NFTconfig['xcloud_game_language'] === 'Auto' ? 'block' : 'none',
  2120. options: Object.keys(default_language_list).map(languageChinese => {
  2121. if (languageChinese == '智能简繁') { return; }
  2122. return {
  2123. value: default_language_list[languageChinese],
  2124. text: languageChinese,
  2125. id: default_language_list[languageChinese] + 'ifErr'
  2126. };
  2127.  
  2128. }),
  2129. checkedValue: NFTconfig['IfErrUsedefaultGameLanguage'],
  2130. needHr: true
  2131. },
  2132. {
  2133. label: '免代理直连:',
  2134. type: 'radio',
  2135. name: 'noNeedVpn',
  2136. display: 'block',
  2137. options: [
  2138. { value: 1, text: '开', id: 'noNeedVpnOn' },
  2139. { value: 0, text: '关', id: 'noNeedVpnOff' },
  2140. ],
  2141. checkedValue: NFTconfig['no_need_VPN_play'],
  2142. needHr: false
  2143. },
  2144. {
  2145. label: '选服:',
  2146. type: 'radio',
  2147. name: 'selectRegion',
  2148. display: NFTconfig['no_need_VPN_play'] === 1 ? 'block' : 'none',
  2149. options: Object.keys(NFTconfig['regionBlock']['options']).map(region => {
  2150. return {
  2151. value: NFTconfig['regionBlock']['options'][region],
  2152. text: region,
  2153. id: NFTconfig['regionBlock']['options'][region]
  2154. };
  2155. }),
  2156. checkedValue: NFTconfig['regionBlock']['options'][NFTconfig['regionBlock']['blockIp']],
  2157. needHr: false
  2158. },
  2159. {
  2160. label: '自定义IP:',
  2161. type: 'radio',
  2162. name: 'customfakeIpInput',
  2163. display: NFTconfig['no_need_VPN_play'] === 1 ? 'block' : 'none',
  2164. value: NFTconfig['useCustomfakeIp'],
  2165. needHr: true,
  2166. moreDom: `<input type="radio" class="selectRegionListener settingsBoxInputRadio" style="outline:none;"
  2167. name='selectRegion' id="customfakeIp" value="customfakeIp" ${NFTconfig['useCustomfakeIp'] == 1 ? 'checked' : ''}>
  2168. <label for="customfakeIp" style="padding-right: 7px;">自定义IP:</label>
  2169. <input type='text' style="display: ` + (NFTconfig['useCustomfakeIp'] == 1 ? 'inline' : 'none')
  2170. + `;outline: none;width: 125px;" id="customfakeIpInput" class="customfakeIpListener" value="${NFTconfig['customfakeIp']}" placeholder="请输入IP"/>`
  2171.  
  2172. },
  2173. {
  2174. label: '分辨率:',
  2175. type: 'radio',
  2176. name: 'highBitrate',
  2177. display: 'block',
  2178. options: [
  2179. { value: 1, text: '1080P', id: 'high_bitrateOn' },
  2180. { value: 0, text: '720P', id: 'high_bitrateOff' }
  2181. ],
  2182. checkedValue: NFTconfig['high_bitrate'],
  2183. needHr: true
  2184. },
  2185. {
  2186. label: '浏览器编解码偏好:',
  2187. showLable: true,
  2188. type: 'dropdown',
  2189. name: 'rtcCodecPreferences',
  2190. display: "block",
  2191. options: NFTconfig['rtcCodecPreferences']['options'],
  2192. selectedValue: NFTconfig['rtcCodecPreferences']['default'],
  2193. needHr: true
  2194. },
  2195. {
  2196. label: '禁止检测网络状况:',
  2197. type: 'radio',
  2198. name: 'disableCheckNetwork',
  2199. display: 'block',
  2200. options: [
  2201. { value: 1, text: '开', id: 'disableCheckNetworkOn' },
  2202. { value: 0, text: '关', id: 'disableCheckNetworkOff' }
  2203. ],
  2204. checkedValue: NFTconfig['disableCheckNetwork'],
  2205. needHr: true
  2206. },
  2207. {
  2208. label: '强制触控:',
  2209. type: 'radio',
  2210. name: 'autoOpenOC',
  2211. display: 'block',
  2212. options: [
  2213. { value: 1, text: '开', id: 'autoOpenOCOn' },
  2214. { value: 0, text: '关', id: 'autoOpenOCOff' }
  2215. ],
  2216. checkedValue: NFTconfig['autoOpenOC'],
  2217. needHr: true,
  2218. moreDom: `<div id="autoShowTouchDom" style="padding-right: 0px;display: ${NFTconfig['autoOpenOC'] == 1 ? 'inline' : 'none'}">
  2219. <input type="checkbox" class="autoShowTouchListener settingsBoxInputRadio" style="outline:none;cursor: pointer;" name='autoShowTouch'
  2220. id="autoShowTouch" ${NFTconfig['autoShowTouch'] == true ? 'checked' : ''}><label for="autoShowTouch" style="cursor: pointer;">自动弹出</label></div>`
  2221. },
  2222. {
  2223. label: '屏蔽触控:',
  2224. type: 'radio',
  2225. name: 'disableTouchControls',
  2226. display: 'block',
  2227. options: [
  2228. { value: 1, text: '开', id: 'disableTouchControlsOn' },
  2229. { value: 0, text: '关', id: 'disableTouchControlsOff' },
  2230. ],
  2231. checkedValue: NFTconfig['disableTouchControls'],
  2232. needHr: true
  2233. },
  2234. {
  2235. label: '自动全屏:',
  2236. type: 'radio',
  2237. name: 'autoFullScreen',
  2238. display: 'block',
  2239. options: [
  2240. { value: 1, text: '开', id: 'autoFullScreenOn' },
  2241. { value: 0, text: '关', id: 'autoFullScreenOff' }
  2242. ],
  2243. checkedValue: NFTconfig['autoFullScreen'],
  2244. needHr: true
  2245. },
  2246. {
  2247. label: '优先IPv6:',
  2248. type: 'radio',
  2249. name: 'IPv6server',
  2250. display: 'block',
  2251. options: [
  2252. { value: 1, text: '开', id: 'IPv6On' },
  2253. { value: 0, text: '关', id: 'IPv6Off' }
  2254. ],
  2255. checkedValue: NFTconfig['IPv6'],
  2256. needHr: true
  2257. }
  2258. ,
  2259. {
  2260. label: '物理服务器:',
  2261. type: 'radio',
  2262. name: 'blockXcloudServer',
  2263. display: 'block',
  2264. options: [
  2265. { value: 1, text: '开', id: 'blockXcloudServerOn' },
  2266. { value: 0, text: '关', id: 'blockXcloudServerOff' }
  2267. ],
  2268. checkedValue: NFTconfig['blockXcloudServer'],
  2269. needHr: false
  2270. },
  2271. {
  2272. label: '选择服务器:',
  2273. type: 'dropdown',
  2274. name: 'defaultXcloudServer',
  2275. display: NFTconfig['blockXcloudServer'] === 1 ? "block" : "none",
  2276. options: NFTconfig['blockXcloudServerList'],
  2277. selectedValue: NFTconfig['defaultXcloudServer'],
  2278. needHr: true
  2279.  
  2280. },
  2281. {
  2282. label: '挂机防踢:',
  2283. type: 'radio',
  2284. name: 'antiKick',
  2285. display: 'block',
  2286. options: [
  2287. { value: 1, text: '开', id: 'antiKickOn' },
  2288. { value: 0, text: '关', id: 'antiKickOff' }
  2289. ],
  2290. checkedValue: NFTconfig['antiKick'],
  2291. needHr: true
  2292.  
  2293. },
  2294. {
  2295. label: '设置悬浮窗:',
  2296. type: 'radio',
  2297. name: 'noPopSetting',
  2298. display: 'block',
  2299. options: [
  2300. { value: 0, text: '显示', id: 'noPopSettingOff' },
  2301. { value: 1, text: '隐藏', id: 'noPopSettingOn' }
  2302. ],
  2303. checkedValue: NFTconfig['noPopSetting'],
  2304. needHr: true
  2305. },
  2306. {
  2307. label: '开启串流功能:',
  2308. type: 'radio',
  2309. name: 'enableRemotePlay',
  2310. display: 'block',
  2311. options: [
  2312. { value: 1, text: '开', id: 'enableRemotePlayOn' },
  2313. { value: 0, text: '关', id: 'enableRemotePlayOff' }
  2314. ],
  2315. checkedValue: NFTconfig['enableRemotePlay'],
  2316. needHr: true
  2317. }
  2318.  
  2319.  
  2320. ];
  2321.  
  2322. // 函数用于生成单个设置项的HTML
  2323. function generateSettingElement(setting) {
  2324. let settingHTML = `<lable style="display:${setting.display};white-space: nowrap;margin-bottom:0.375rem;" class="${setting.name + 'Dom'}">`;
  2325. if (setting.type === 'radio') {
  2326. if (setting.options != undefined) {
  2327. settingHTML += `<label style="display:block;text-align:left;"><div style="display: inline;">${setting.label}</div>`;
  2328. setting.options.forEach(option => {
  2329. if (option == null) { return; }
  2330.  
  2331. settingHTML += `
  2332. <label style="cursor: pointer;"><input type="radio" class="${setting.name + 'Listener'} settingsBoxInputRadio" style="outline:none;" name="${setting.name}"
  2333. id="${option.id}" value="${option.value}" ${option.value === setting.checkedValue ? 'checked' : ''}>${option.text}</label>
  2334. `;
  2335. });
  2336. }
  2337. if (setting.moreDom != undefined) {
  2338. settingHTML += setting.moreDom;
  2339. }
  2340. settingHTML += '</label>';
  2341. } else if (setting.type === 'text') {
  2342. settingHTML += `<label style="display: block;text-align:left;"><div style="display: inline;">${setting.label}</div>`;
  2343. settingHTML += `
  2344. <input type="text" style="display: inline;outline: none;width: 125px;" id="${setting.name}" class="${setting.name}Listener" value="${setting.value}" placeholder="请输入${setting.label}"/>
  2345. `;
  2346. settingHTML += `</label>`;
  2347. } else if (setting.type === 'dropdown') {
  2348. if (setting.showLable == true) {
  2349. settingHTML += `<label style="display: block;text-align:left;${setting.css}"><div style="display: inline;">${setting.label}</div>`;
  2350. }
  2351. if (setting.options.length == undefined) {
  2352. setting.options = Object.keys(setting.options);
  2353. }
  2354. settingHTML += `
  2355. <select style="outline: none;margin-bottom:5px;" class="${setting.name + 'Listener'}">
  2356. ${setting.options.map(option => `<option value="${option}" ${option === setting.selectedValue ? 'selected' : ''}>${option}</option>`).join('')}
  2357. </select>
  2358. `;
  2359.  
  2360. if (setting.moreDom != undefined) {
  2361. settingHTML += setting.moreDom;
  2362. }
  2363. }
  2364.  
  2365. settingHTML += `</lable>`;
  2366.  
  2367. if (setting.needHr) {
  2368. settingHTML += `<hr style="background-color: black;width:95%" />`
  2369. }
  2370. return settingHTML;
  2371. }
  2372. function generateSettingsPage() {
  2373. let settingsHTML = `
  2374. <div style="padding: 10px;color: black;display:none;" class="settingsBackgroud" id="settingsBackgroud">
  2375. <div class="settingsBox"><span class="blink-text" onclick="window.location.href='https://gf.qytechs.cn/zh-CN/scripts/455741';">更新咯~</span>
  2376. `;
  2377. settingsConfig.forEach(setting => {
  2378. settingsHTML += generateSettingElement(setting);
  2379. });
  2380.  
  2381. settingsHTML += `
  2382. <button class="closeSetting1 closeSetting2" style="outline: none;">关闭</button>
  2383. <div style="text-align: right;margin-top: 8px;font-size: 16px;">
  2384. <label>捐赠:</label>
  2385. <a style="margin-right:15px;outline: none;color: #107c10;text-decoration: underline;" href="https://gf.qytechs.cn/rails/active_storage/blobs/redirect/eyJfcmFpbHMiOnsibWVzc2FnZSI6IkJBaHBBMVNFQVE9PSIsImV4cCI6bnVsbCwicHVyIjoiYmxvYl9pZCJ9fQ==--14c278e3f72d104cff50bf130d4039229fc25a6b/wx.png?locale=zh-CN">微信</a>
  2386. <a style="outline: none;color: #107c10;text-decoration: underline;" href="https://gf.qytechs.cn/rails/active_storage/blobs/redirect/eyJfcmFpbHMiOnsibWVzc2FnZSI6IkJBaHBBMU9FQVE9PSIsImV4cCI6bnVsbCwicHVyIjoiYmxvYl9pZCJ9fQ==--5fc08aaa8407cc6099654d65455b7966bf2c60ee/alipay.png?locale=zh-CN">支付宝</a>
  2387. </div>
  2388. </div>
  2389. </div>
  2390. `;
  2391.  
  2392. return settingsHTML;
  2393. }
  2394. let needrefresh = 0;
  2395. function initSettingBox() {
  2396. $('body').append(generateSettingsPage());
  2397.  
  2398. //确定
  2399. $(document).on('click', '.closeSetting1', function () {
  2400.  
  2401. naifeitian.hideSetting();
  2402. if (needrefresh == 1) {
  2403. history.go(0);
  2404. }
  2405. });
  2406.  
  2407. //开启串流
  2408. $(document).on('click', '.enableRemotePlayListener', function () {
  2409. needrefresh = 1;
  2410. naifeitian.setValue('enableRemotePlayGM', $(this).val());
  2411. $('.closeSetting1').text('确定');
  2412. });
  2413. //设置悬浮窗
  2414. $(document).on('click', '.noPopSettingListener', function () {
  2415. naifeitian.setValue('noPopSettingGM', $(this).val());
  2416. needrefresh = 1;
  2417. $('.closeSetting1').text('确定');
  2418. });
  2419. //挂机防踢
  2420. $(document).on('click', '.antiKickListener', function () {
  2421. needrefresh = 1;
  2422. naifeitian.setValue('antiKickGM', $(this).val());
  2423. $('.closeSetting1').text('确定');
  2424. });
  2425. //ipv6
  2426. $(document).on('click', '.IPv6serverListener', function () {
  2427. naifeitian.setValue('IPv6GM', $(this).val());
  2428. needrefresh = 1;
  2429. $('.closeSetting1').text('确定');
  2430. });
  2431. //选择服务器change
  2432. $(document).on('change', '.defaultXcloudServerListener', function () {
  2433. naifeitian.setValue('defaultXcloudServerGM', $(this).val());
  2434. needrefresh = 1;
  2435. $('.closeSetting1').text('确定');
  2436. });
  2437. //物理服务器
  2438. $(document).on('click', '.blockXcloudServerListener', function () {
  2439. if ($(this).val() == 0) {
  2440. $('.defaultXcloudServerDom').css('display', 'none');
  2441. } else {
  2442. $('.defaultXcloudServerDom').css('display', 'block');
  2443. }
  2444. naifeitian.setValue('blockXcloudServerGM', $(this).val());
  2445. needrefresh = 1;
  2446. $('.closeSetting1').text('确定');
  2447. });
  2448.  
  2449. //自动全屏
  2450. $(document).on('click', '.autoFullScreenListener', function () {
  2451. naifeitian.setValue('autoFullScreenGM', $(this).val());
  2452. needrefresh = 1;
  2453. $('.closeSetting1').text('确定');
  2454. });
  2455. //屏蔽触控
  2456. $(document).on('click', '.disableTouchControlsListener', function () {
  2457. if ($(this).val() == 1) {
  2458. if (!confirm("确定要屏蔽触控吗?")) {
  2459. $('#disableTouchControlsOff').click();
  2460. return;
  2461. }
  2462. $('#autoOpenOCOff').click();
  2463. }
  2464.  
  2465. needrefresh = 1;
  2466. naifeitian.setValue('disableTouchControlsGM', $(this).val());
  2467. $('.closeSetting1').text('确定');
  2468. });
  2469.  
  2470. //自动弹出
  2471. $(document).on('change', '.autoShowTouchListener', function () {
  2472. let newVal = $(this).attr('checked') == 'checked';
  2473. if (newVal) {
  2474. $(this).removeAttr('checked');
  2475. } else {
  2476. $(this).attr('checked');
  2477. }
  2478. naifeitian.setValue('autoShowTouchGM', !newVal);
  2479. needrefresh = 1;
  2480. $('.closeSetting1').text('确定');
  2481. });
  2482. //强制触控
  2483. $(document).on('click', '.autoOpenOCListener', function () {
  2484.  
  2485. if ($(this).val() == 0) {
  2486. $('#autoShowTouchDom').css('display', 'none');
  2487. } else {
  2488. $('#autoShowTouchDom').css('display', 'inline');
  2489. $('#disableTouchControlsOff').click();
  2490. }
  2491.  
  2492. naifeitian.setValue('autoOpenOCGM', $(this).val());
  2493. needrefresh = 1;
  2494. $('.closeSetting1').text('确定');
  2495. });
  2496.  
  2497. //禁止检测网络
  2498. $(document).on('click', '.disableCheckNetworkListener', function () {
  2499. naifeitian.setValue('disableCheckNetworkGM', $(this).val());
  2500. needrefresh = 1;
  2501. $('.closeSetting1').text('确定');
  2502. });
  2503. //浏览器编解码偏好
  2504. $(document).on('change', '.rtcCodecPreferencesListener', function () {
  2505. NFTconfig['rtcCodecPreferences']['default'] = $(this).val();
  2506. naifeitian.setValue('rtcCodecPreferencesGM', NFTconfig['rtcCodecPreferences']);
  2507. needrefresh = 1;
  2508. $('.closeSetting1').text('确定');
  2509. });
  2510. //分辨率
  2511. $(document).on('click', '.highBitrateListener', function () {
  2512. naifeitian.setValue('high_bitrateGM', $(this).val());
  2513. needrefresh = 1;
  2514. $('.closeSetting1').text('确定');
  2515. });
  2516.  
  2517.  
  2518. //自定义ip输入框
  2519. $(document).on('blur', '.customfakeIpListener', function () {
  2520. if (naifeitian.isValidIP($(this).val())) {
  2521. naifeitian.setValue('customfakeIpGM', $(this).val());
  2522. } else {
  2523. $(this).val("");
  2524. naifeitian.setValue('customfakeIpGM', '');
  2525. alert('IP格式错误!');
  2526. return;
  2527. }
  2528. needrefresh = 1;
  2529. $('.closeSetting1').text('确定');
  2530. });
  2531. //选服
  2532. $(document).on('click', '.selectRegionListener', function () {
  2533. if ($(this).val() == 'customfakeIp') {
  2534. naifeitian.setValue('useCustomfakeIpGM', 1);
  2535. $('#customfakeIpInput').css('display', 'inline');
  2536. } else {
  2537. NFTconfig['regionBlock']['blockIp'] = Object.keys(NFTconfig['regionBlock']['options']).find(key => NFTconfig['regionBlock']['options'][key] === $(this).val());
  2538. naifeitian.setValue('regionBlockGM', NFTconfig['regionBlock']);
  2539. naifeitian.setValue('useCustomfakeIpGM', 0);
  2540. $('#customfakeIpInput').css('display', 'none');
  2541. }
  2542. needrefresh = 1;
  2543. $('.closeSetting1').text('确定');
  2544. });
  2545.  
  2546. //免代理直连
  2547. $(document).on('click', '.noNeedVpnListener', function () {
  2548. if ($(this).val() == 0) {
  2549. $('.selectRegionDom').css('display', 'none');;
  2550. $('.customfakeIpInputDom').css('display', 'none');
  2551. } else {
  2552. $('.selectRegionDom').css('display', 'block');
  2553. $('.customfakeIpInputDom').css('display', 'block');
  2554. }
  2555. naifeitian.setValue('no_need_VPN_playGM', $(this).val());
  2556. needrefresh = 1;
  2557. $('.closeSetting1').text('确定');
  2558. });
  2559.  
  2560. //智能简繁错误
  2561. $(document).on('click', '.IfErrUsedefaultGameLanguageListener', function () {
  2562. naifeitian.setValue('IfErrUsedefaultGameLanguageGM', $(this).val());
  2563. needrefresh = 1;
  2564. $('.closeSetting1').text('确定');
  2565. });
  2566. //语言
  2567. $(document).on('click', '.selectLanguageListener', function () {
  2568. if ($(this).val() != 'Auto') {
  2569. $('.IfErrUsedefaultGameLanguageDom').css('display', 'none');
  2570. } else {
  2571. $('.IfErrUsedefaultGameLanguageDom').css('display', 'block');
  2572. }
  2573. naifeitian.setValue('xcloud_game_languageGM', $(this).val());
  2574. needrefresh = 1;
  2575. $('.closeSetting1').text('确定');
  2576. });
  2577.  
  2578. //选择语言
  2579. $(document).on('click', '.chooseLanguageListener', function () {
  2580. if ($(this).val() == 0) {
  2581. $('.selectLanguageDom').css('display', 'none');
  2582. $('.IfErrUsedefaultGameLanguageDom').css('display', 'none');
  2583. } else {
  2584. $('.selectLanguageDom').css('display', 'block');
  2585.  
  2586. if (naifeitian.getValue('xcloud_game_languageGM') == 'Auto') {
  2587. $('.IfErrUsedefaultGameLanguageDom').css('display', 'block');
  2588. }
  2589. }
  2590. naifeitian.setValue('chooseLanguageGM', $(this).val());
  2591. needrefresh = 1;
  2592. $('.closeSetting1').text('确定');
  2593. });
  2594. }
  2595. //手势显隐触控
  2596. function initSlideHide() {
  2597. var gestureArea = $("<div></div>");
  2598. gestureArea.attr("id", "touchControllerEventArea");
  2599. $(document.documentElement).append(gestureArea);
  2600.  
  2601. gestureArea = $("#touchControllerEventArea");
  2602. let startX, startY, endX, endY;
  2603. let threshold = 60; // 手势滑动的阈值
  2604. gestureArea.on("touchstart", function (e) {
  2605. startX = e.originalEvent.touches[0].clientX;
  2606. startY = e.originalEvent.touches[0].clientY;
  2607. });
  2608. gestureArea.on("touchmove", function (e) {
  2609. endX = e.originalEvent.touches[0].clientX;
  2610. endY = e.originalEvent.touches[0].clientY;
  2611. });
  2612. gestureArea.on("touchend", function (e) {
  2613. if (startX !== undefined && startY !== undefined && endX !== undefined && endY !== undefined) {
  2614. const deltaX = endX - startX;
  2615. const deltaY = endY - startY;
  2616. if (Math.abs(deltaX) > Math.abs(deltaY) && Math.abs(deltaX) > threshold) {
  2617. if (deltaX < 0) {
  2618. // 左滑
  2619. $('#BabylonCanvasContainer-main').css('display', 'none');
  2620. $('#MultiTouchSurface').css('display', 'none');
  2621. e.preventDefault();
  2622. } else {
  2623. // 右滑
  2624. $('#BabylonCanvasContainer-main').css('display', 'block');
  2625. $('#MultiTouchSurface').css('display', 'block');
  2626. e.preventDefault();
  2627. }
  2628. }
  2629. }
  2630. });
  2631.  
  2632. }
  2633.  
  2634. async function checkUPD() {
  2635. try {
  2636. const response = await fetch("https://gf.qytechs.cn/zh-CN/scripts/455741-xbox-cloud-gaming%E4%BC%98%E5%8C%96%E6%95%B4%E5%90%88/versions");
  2637. const data = await response.text();
  2638.  
  2639. let historyVersion = $(data).find('.history_versions')[0];
  2640. let hli = $($(historyVersion).find('li .version-number > a')[0]);
  2641. let version = hli.text();
  2642. if (nftxboxversion != version) {
  2643. $('head').append('<style>.blink-text{ display:block!important }</style>');
  2644. }
  2645. } catch (error) {
  2646. //console.error('Fetch error:', error);
  2647. }
  2648. }
  2649.  
  2650. $(document).ready(function () {
  2651. //检测Thank you for your interest
  2652. let checkTYFYIInterval = setInterval(() => {
  2653. if (checkIpsuc) {
  2654. clearInterval(checkTYFYIInterval);
  2655. return;
  2656. }
  2657. var title = $("[class*='UnsupportedMarketPage-module__title']");
  2658. if (title.length > 0) {
  2659. console.log("脚本检测到免代理没有成功,自动刷新中");
  2660. title.text("脚本检测到免代理没有成功,自动刷新中");
  2661. history.go(0);
  2662. clearInterval(checkTYFYIInterval);
  2663. }
  2664.  
  2665. }, 5000);
  2666.  
  2667. setTimeout(function () {
  2668.  
  2669. if (NFTconfig['noPopSetting'] == 0) {
  2670. $('body').append(`<div id="popSetting" style="display:block">⚙️ 设置</div>`);
  2671. $(document).on('click', '#popSetting', function () {
  2672. naifeitian.showSetting();
  2673. });
  2674. }
  2675. checkUPD();
  2676. initSettingBox();
  2677. updateVideoPlayerCss();
  2678. StreamStats.render();
  2679. setupVideoSettingsBar();
  2680. initSlideHide();
  2681. }, 2000);
  2682.  
  2683. });
  2684.  
  2685. let timer;
  2686. let mousehidding = false;
  2687. $(document).mousemove(function () {
  2688. if (mousehidding) {
  2689. mousehidding = false;
  2690. return;
  2691. }
  2692. if (timer) {
  2693. clearTimeout(timer);
  2694. timer = 0;
  2695. }
  2696. $('html').css({
  2697. cursor: ''
  2698. });
  2699. timer = setTimeout(function () {
  2700. mousehidding = true;
  2701. $('html').css({
  2702. cursor: 'none'
  2703. });
  2704. }, 2000);
  2705. });
  2706.  
  2707.  
  2708. let _pushState = window.history.pushState;
  2709. window.history.pushState = function () {
  2710.  
  2711. setTimeout(RemotePlay.detect, 10);
  2712. if (NFTconfig['noPopSetting'] == 0) {
  2713. if (arguments[2].substring(arguments[2].length, arguments[2].length - 5) == '/play') {
  2714. exitGame();
  2715. } else {
  2716. $('#popSetting').css('display', 'none');
  2717. }
  2718. }
  2719. if (arguments[2].indexOf("/play/games/") > -1) {
  2720. let timeout = 0;
  2721. let checkPlayInterval = setInterval(() => {
  2722. var playButtons = $("[class*='PlayButton-module__playButton'][disabled]");
  2723. if (playButtons.length > 0) {
  2724. playButtons.text("该游戏无法在" + [NFTconfig['regionBlock']['blockIp']] + "游玩,请使用脚本切换其他服尝试");
  2725. clearInterval(checkPlayInterval);
  2726.  
  2727. }
  2728. if (timeout == 10) {
  2729. clearInterval(checkPlayInterval);
  2730. }
  2731. timeout++;
  2732.  
  2733. }, 1333);
  2734. }
  2735.  
  2736. if ((!arguments[2].includes("/play/launch/") && !arguments[2].includes('/remote-play'))) {
  2737. exitGame();
  2738. }
  2739. if(arguments[2].includes("/play/launch/")){
  2740. $($("div[class^='StreamGateDialog-module__scrollable']")[0]).css("display","none");
  2741. }
  2742.  
  2743. if (arguments[2].includes("/https")) {
  2744. if (window.location.href.includes("/remote-play")) {
  2745. exitGame();
  2746. }
  2747. setTimeout(() => { history.go(-1) }, 10);
  2748. return;
  2749. }
  2750. if (arguments[2].includes("/dev-tools/direct-connect")) {
  2751. exitGame();
  2752. setTimeout(() => { history.go(-1) }, 10);
  2753. return
  2754. }
  2755.  
  2756. return _pushState.apply(this, arguments);
  2757. }
  2758.  
  2759. function getVideoPlayerFilterStyle() {
  2760. const filters = [];
  2761.  
  2762. const clarity = NFTconfig['VIDEO_CLARITY']['default'];
  2763. if (clarity != 0) {
  2764. const level = 7 - (clarity - 1); // 5,6,7
  2765. const matrix = `0 -1 0 -1 ${level} -1 0 -1 0`;
  2766. document.getElementById('better-xcloud-filter-clarity-matrix').setAttributeNS(null, 'kernelMatrix', matrix);
  2767.  
  2768. filters.push(`url(#better-xcloud-filter-clarity)`);
  2769. }
  2770.  
  2771. const saturation = NFTconfig['VIDEO_SATURATION']['default'];
  2772. if (saturation != 100) {
  2773. filters.push(`saturate(${saturation}%)`);
  2774. }
  2775.  
  2776. const contrast = NFTconfig['VIDEO_CONTRAST']['default'];
  2777. if (contrast != 100) {
  2778. filters.push(`contrast(${contrast}%)`);
  2779. }
  2780.  
  2781. const brightness = NFTconfig['VIDEO_BRIGHTNESS']['default'];
  2782. if (brightness != 100) {
  2783. filters.push(`brightness(${brightness}%)`);
  2784. }
  2785.  
  2786. return filters.join(' ');
  2787. }
  2788.  
  2789.  
  2790. function updateVideoPlayerCss() {
  2791. let $elm = document.getElementById('better-xcloud-video-css');
  2792. if (!$elm) {
  2793. const CE = createElement;
  2794.  
  2795. $elm = CE('style', { id: 'better-xcloud-video-css' });
  2796. document.documentElement.appendChild($elm);
  2797.  
  2798. // Setup SVG filters
  2799. const $svg = CE('svg', {
  2800. 'id': 'better-xcloud-video-filters',
  2801. 'xmlns': 'http://www.w3.org/2000/svg',
  2802. 'class': 'better-xcloud-gone',
  2803. }, CE('defs', { 'xmlns': 'http://www.w3.org/2000/svg' },
  2804. CE('filter', { 'id': 'better-xcloud-filter-clarity', 'xmlns': 'http://www.w3.org/2000/svg' },
  2805. CE('feConvolveMatrix', { 'id': 'better-xcloud-filter-clarity-matrix', 'order': '3', 'xmlns': 'http://www.w3.org/2000/svg' }))
  2806. )
  2807. );
  2808. document.documentElement.appendChild($svg);
  2809. }
  2810.  
  2811. let filters = getVideoPlayerFilterStyle();
  2812. let css = '';
  2813. if (filters) {
  2814. css += `filter: ${filters} !important;`;
  2815. }
  2816.  
  2817. if (NFTconfig['video_stretch'].default == 'fill') {
  2818. css += 'object-fit: fill !important;';
  2819. }
  2820.  
  2821. if (NFTconfig['video_stretch'].default == 'setting') {
  2822. css += `transform: scaleX(` + (NFTconfig['video_stretch_x_y'].x * 1 + 1) + `) scaleY(` + (NFTconfig['video_stretch_x_y'].y * 1 + 1) + `) !important;`;
  2823. }
  2824.  
  2825. if (css) {
  2826. css = `#game-stream video {${css}}`;
  2827. }
  2828.  
  2829. $elm.textContent = css;
  2830. }
  2831. function screenClicktohide() {
  2832. const $screen = document.querySelector('#PageContent section[class*=PureScreens]');
  2833. const $quickBar = document.querySelector('.better-xcloud-quick-settings-bar');
  2834. const $parent = $screen.parentElement;
  2835. const hideQuickBarFunc = e => {
  2836. e.stopPropagation();
  2837. if (e.target != $parent && e.target.id !== 'MultiTouchSurface' && !e.target.querySelector('#BabylonCanvasContainer-main')) {
  2838. return;
  2839. }
  2840.  
  2841. // Hide Quick settings bar
  2842. $quickBar.style.display = 'none';
  2843. $('.better-xcloud-stats-settings').css("display", "none");
  2844.  
  2845. $parent.removeEventListener('click', hideQuickBarFunc);
  2846. $parent.removeEventListener('touchstart', hideQuickBarFunc);
  2847.  
  2848. if (e.target.id === 'MultiTouchSurface') {
  2849. e.target.removeEventListener('touchstart', hideQuickBarFunc);
  2850. }
  2851. }
  2852. $parent.addEventListener('click', hideQuickBarFunc);
  2853. $parent.addEventListener('touchstart', hideQuickBarFunc);
  2854.  
  2855. }
  2856. //插入按钮
  2857. function injectVideoSettingsButton() {
  2858. const $screen = document.querySelector('#PageContent section[class*=PureScreens]');
  2859. if (!$screen) {
  2860. return;
  2861. }
  2862.  
  2863. if ($screen.xObserving) {
  2864. return;
  2865. }
  2866.  
  2867. $screen.xObserving = true;
  2868. const $parent = $screen.parentElement;
  2869. const $quickBar = document.querySelector('.better-xcloud-quick-settings-bar');
  2870. screenClicktohide();
  2871. let $btnStreamStats;
  2872. let $btnVideoSettings_HD;
  2873. const observer = new MutationObserver(mutationList => {
  2874. mutationList.forEach(item => {
  2875. if (item.type !== 'childList') {
  2876. return;
  2877. }
  2878.  
  2879. item.addedNodes.forEach(async node => {
  2880.  
  2881. if (IS_REMOTE_PLAYING) {
  2882. try {
  2883. let btn = $(node).find('button[class*=PopupScreen-module__button][data-auto-focus=false]');
  2884. if ($(btn).length > 0) {
  2885. $(btn).click();
  2886. throw new Error("巴啦啦能量-呼尼拉-魔仙变身!");
  2887. }
  2888. } catch (e) { }
  2889. }
  2890.  
  2891. if (!node.className || !node.className.startsWith('StreamMenu')) {
  2892. return;
  2893. }
  2894.  
  2895. const $orgButton = node.querySelector('div > div > button');
  2896. if (!$orgButton) {
  2897. return;
  2898. }
  2899.  
  2900. // 创建视频调整
  2901. const $btnVideoSettings = cloneStreamMenuButton($orgButton, '视频调整', ICON_VIDEO_SETTINGS);
  2902. $btnVideoSettings.addEventListener('click', e => {
  2903. e.preventDefault();
  2904. e.stopPropagation();
  2905.  
  2906. // Show Quick settings bar
  2907. $quickBar.style.display = 'flex';
  2908.  
  2909. $parent.addEventListener('click', screenClicktohide());
  2910. $parent.addEventListener('touchstart', screenClicktohide());
  2911.  
  2912. const $touchSurface = document.getElementById('MultiTouchSurface');
  2913. $touchSurface && $touchSurface.style.display != 'none' && $touchSurface.addEventListener('touchstart', screenClicktohide());
  2914. });
  2915. // Add button at the beginning
  2916. $orgButton.parentElement.insertBefore($btnVideoSettings, $orgButton.parentElement.firstChild);
  2917.  
  2918. // Hide Quick bar when closing HUD
  2919. const $btnCloseHud = document.querySelector('button[class*=StreamMenu-module__backButton]');
  2920. $btnCloseHud.addEventListener('click', e => {
  2921. $quickBar.style.display = 'none';
  2922. });
  2923.  
  2924. // 创建流监控
  2925. const $btnStreamStats = cloneStreamMenuButton($orgButton, '流监控', ICON_HD_STREAM_STATS);
  2926. $btnStreamStats.addEventListener('click', e => {
  2927. e.preventDefault();
  2928. e.stopPropagation();
  2929.  
  2930. // Toggle Stream Stats
  2931. StreamStats.toggle();
  2932. });
  2933.  
  2934. // Insert after Video Settings button
  2935. $orgButton.parentElement.insertBefore($btnStreamStats, $btnVideoSettings);
  2936.  
  2937. //menu图标样式
  2938. if (StreamStats.status()) {
  2939. $('.ICON_HD_STREAM_STATS_ON').css("display", 'block');
  2940. $('.ICON_HD_STREAM_STATS_OFF').css("display", 'none');
  2941. } else {
  2942. $('.ICON_HD_STREAM_STATS_ON').css("display", 'none');
  2943. $('.ICON_HD_STREAM_STATS_OFF').css("display", 'block');
  2944. }
  2945. //桥
  2946. const $menu = document.querySelector('div[class*=StreamMenu-module__menuContainer] > div[class*=Menu-module]');
  2947. //const streamBadgesElement = await StreamBadges.render();
  2948. $menu.appendChild(await StreamBadges.render());
  2949.  
  2950. //$menu.insertAdjacentElement('afterend', streamBadgesElement);
  2951.  
  2952. });
  2953. });
  2954.  
  2955. mutationList.forEach(item => {
  2956. if (item.type !== 'childList') {
  2957. return;
  2958. }
  2959.  
  2960. item.removedNodes.forEach($node => {
  2961. if (!$node || !$node.className || !$node.className.startsWith) {
  2962. return;
  2963. }
  2964.  
  2965. });
  2966.  
  2967. item.addedNodes.forEach(async $node => {
  2968. if (!$node || !$node.className) {
  2969. return;
  2970. }
  2971.  
  2972. if ($node.className.startsWith('Overlay-module_') || $node.className.startsWith('InProgressScreen')) {
  2973. $node = $node.querySelector('#StreamHud');
  2974. }
  2975.  
  2976. if (!$node || ($node.id || '') !== 'StreamHud') {
  2977. return;
  2978. }
  2979.  
  2980. $(document).on('transitionend', '#StreamHud', function () {
  2981. if ($('#StreamHud').css('left') == '0px') {
  2982. $('.hd-stream-setting').removeClass("hd-stream-setting-hide");
  2983. } else {
  2984. $('.hd-stream-setting').addClass("hd-stream-setting-hide");
  2985. }
  2986. if (!NFTconfig['STATS_SLIDE_OPEN']['default']) { return; }
  2987. if ($('#StreamHud').css('left') == '0px') {
  2988. if (!StreamStats.status()) {
  2989. StreamStats.start();
  2990. }
  2991. } else {
  2992. StreamStats.stop();
  2993. }
  2994. });
  2995.  
  2996. // Get the second last button
  2997. const $orgButton = $node.querySelector('div[class^=HUDButton]');
  2998. if (!$orgButton) {
  2999. return;
  3000. }
  3001.  
  3002. // 流监控设置
  3003. if (!$btnStreamStats) {
  3004. $btnStreamStats = cloneStreamHudButton($orgButton, "hd-stream-setting", "流监控设置", ICON_HD_STREAM_STATS);
  3005. $btnStreamStats.addEventListener('click', e => {
  3006. e.preventDefault();
  3007. StreamStats.toggleSettingsUi();
  3008. });
  3009. }
  3010.  
  3011. // 视频调整
  3012. if (!$btnVideoSettings_HD) {
  3013. $btnVideoSettings_HD = cloneStreamHudButton($orgButton, "hd-stream-setting", '视频调整', ICON_HD_VIDEO_SETTINGS);
  3014. $btnVideoSettings_HD.addEventListener('click', e => {
  3015. e.preventDefault();
  3016. e.stopPropagation();
  3017.  
  3018. // Show Quick settings bar
  3019. $quickBar.style.display = 'flex';
  3020.  
  3021. $parent.addEventListener('click', screenClicktohide());
  3022. $parent.addEventListener('touchstart', screenClicktohide());
  3023.  
  3024. const $touchSurface = document.getElementById('MultiTouchSurface');
  3025. $touchSurface && $touchSurface.style.display != 'none' && $touchSurface.addEventListener('touchstart', screenClicktohide());
  3026. });
  3027. }
  3028. // Insert buttons after Stream Settings button
  3029. $orgButton.parentElement.insertBefore($btnStreamStats, $orgButton.parentElement.lastElementChild);
  3030. $orgButton.parentElement.insertBefore($btnVideoSettings_HD, $btnStreamStats);
  3031. if ($('#StreamHud').css('left') == '0px' && NFTconfig['STATS_SLIDE_OPEN']['default']) {
  3032. StreamStats.start();
  3033. }
  3034. if (StreamStats.status()) {
  3035. $('.ICON_HD_STREAM_STATS_ON').css("display", 'block');
  3036. $('.ICON_HD_STREAM_STATS_OFF').css("display", 'none');
  3037. } else {
  3038. $('.ICON_HD_STREAM_STATS_ON').css("display", 'none');
  3039. $('.ICON_HD_STREAM_STATS_OFF').css("display", 'block');
  3040. }
  3041. // Move the Dots button to the beginning
  3042. const $dotsButton = $orgButton.parentElement.lastElementChild;
  3043. $dotsButton.parentElement.insertBefore($dotsButton, $dotsButton.parentElement.firstElementChild);
  3044.  
  3045. });
  3046. });
  3047. });
  3048. observer.observe($screen, { subtree: true, childList: true });
  3049. }
  3050.  
  3051.  
  3052.  
  3053. function patchVideoApi() {
  3054.  
  3055. // Show video player when it's ready
  3056. let showFunc;
  3057. showFunc = function () {
  3058.  
  3059. this.removeEventListener('playing', showFunc);
  3060.  
  3061. if (!this.videoWidth) {
  3062. return;
  3063. }
  3064.  
  3065. onStreamStarted(this);
  3066. STREAM_WEBRTC?.getStats().then(stats => {
  3067.  
  3068. if (NFTconfig['STATS_SHOW_WHEN_PLAYING']['default']) {
  3069. StreamStats.start();
  3070. }
  3071. });
  3072. }
  3073. HTMLMediaElement.prototype.orgPlay = HTMLMediaElement.prototype.play;
  3074. HTMLMediaElement.prototype.play = function () {
  3075. if (letmeOb && NFTconfig['antiKick'] == 1) {
  3076. const divElement = $('div[data-testid="ui-container"]')[0];
  3077. const observer = new MutationObserver(function (mutations) {
  3078. try {
  3079. mutations.forEach(function (mutation) {
  3080. if (mutation.type === 'childList') {
  3081. mutation.addedNodes.forEach(function (addedNode) {
  3082. let btn = $(addedNode).find('button[data-auto-focus="true"]');
  3083. if ($(btn).length > 0 && btn.parent().children().length == 1) {
  3084. $(btn).click();
  3085. throw new Error("巴啦啦能量-呼尼拉-魔仙变身!");
  3086. }
  3087. });
  3088. }
  3089. });
  3090. } catch (e) { }
  3091. });
  3092.  
  3093. setTimeout(() => {
  3094. observer.observe(divElement, { childList: true, subtree: true });
  3095. console.log('antiKick已部署');
  3096. }, 1000 * 20);
  3097. letmeOb = false;
  3098. }
  3099. if (this.className && this.className.startsWith('XboxSplashVideo')) {
  3100. this.volume = 0;
  3101. this.style.display = 'none';
  3102. this.dispatchEvent(new Event('ended'));
  3103.  
  3104. return {
  3105. catch: () => { },
  3106. };
  3107. return nativePlay.apply(this);
  3108. }
  3109.  
  3110. this.addEventListener('playing', showFunc);
  3111. injectVideoSettingsButton();
  3112. return this.orgPlay.apply(this);
  3113. };
  3114. }
  3115.  
  3116. function onStreamStarted($video) {
  3117.  
  3118. StreamBadges.resolution = { width: $video.videoWidth, height: $video.videoHeight };
  3119. StreamBadges.startTimestamp = +new Date;
  3120.  
  3121. // Get battery level
  3122. try {
  3123. navigator.getBattery && navigator.getBattery().then(bm => {
  3124. StreamBadges.startBatteryLevel = Math.round(bm.level * 100);
  3125. });
  3126.  
  3127.  
  3128. STREAM_WEBRTC.getStats().then(stats => {
  3129. const allVideoCodecs = {};
  3130. let videoCodecId;
  3131.  
  3132. const allAudioCodecs = {};
  3133. let audioCodecId;
  3134.  
  3135. const allCandidates = {};
  3136. let candidateId;
  3137.  
  3138. stats.forEach(stat => {
  3139. if (stat.type == 'codec') {
  3140. const mimeType = stat.mimeType.split('/');
  3141. if (mimeType[0] === 'video') {
  3142. // Store all video stats
  3143. allVideoCodecs[stat.id] = stat;
  3144. } else if (mimeType[0] === 'audio') {
  3145. // Store all audio stats
  3146. allAudioCodecs[stat.id] = stat;
  3147. }
  3148. } else if (stat.type === 'inbound-rtp' && stat.packetsReceived > 0) {
  3149. // Get the codecId of the video/audio track currently being used
  3150. if (stat.kind === 'video') {
  3151. videoCodecId = stat.codecId;
  3152. } else if (stat.kind === 'audio') {
  3153. audioCodecId = stat.codecId;
  3154. }
  3155. } else if (stat.type === 'candidate-pair' && stat.packetsReceived > 0 && stat.state === 'succeeded') {
  3156. candidateId = stat.remoteCandidateId;
  3157. } else if (stat.type === 'remote-candidate') {
  3158. allCandidates[stat.id] = stat.address;
  3159. }
  3160. });
  3161.  
  3162. // Get video codec from codecId
  3163. if (videoCodecId) {
  3164. const videoStat = allVideoCodecs[videoCodecId];
  3165. const video = {
  3166. codec: videoStat.mimeType.substring(6),
  3167. };
  3168.  
  3169. if (video.codec === 'H264') {
  3170. const match = /profile-level-id=([0-9a-f]{6})/.exec(videoStat.sdpFmtpLine);
  3171. video.profile = match ? match[1] : null;
  3172. }
  3173.  
  3174. StreamBadges.video = video;
  3175. }
  3176.  
  3177. // Get audio codec from codecId
  3178. if (audioCodecId) {
  3179. const audioStat = allAudioCodecs[audioCodecId];
  3180. StreamBadges.audio = {
  3181. codec: audioStat.mimeType.substring(6),
  3182. bitrate: audioStat.clockRate,
  3183. }
  3184. }
  3185. // Get server type
  3186. if (candidateId) {
  3187. //console.log(candidateId, allCandidates);
  3188. StreamBadges.ipv6 = allCandidates[candidateId].includes(':');
  3189. }
  3190.  
  3191. });
  3192. } catch (e) { }
  3193.  
  3194. }
  3195. function moveCodecToIndex(array, currentIndex, targetIndex, element) {
  3196. array.splice(currentIndex, 1);
  3197. array.splice(targetIndex, 0, element);
  3198. }
  3199. function customizeRtcCodecs() {
  3200.  
  3201. const customCodecProfile = NFTconfig['rtcCodecPreferences']['default'];
  3202.  
  3203. if (customCodecProfile === '默认') {
  3204. console.log("customizeRtcCodecs:默认");
  3205. return;
  3206. }
  3207. if (typeof RTCRtpTransceiver === 'undefined' || !('setCodecPreferences' in RTCRtpTransceiver.prototype)) {
  3208. return false;
  3209. }
  3210.  
  3211. let codecProfilePrefix = "";
  3212. let codecProfileLevelId = "";
  3213. let codecMimeType = "";
  3214. const codecProfileMap = { "264": { "高": "4d", "中": "42e", "低": "420" } };
  3215.  
  3216. if (customCodecProfile.includes("264")) {
  3217. const codecLevel = Object.keys(codecProfileMap["264"]).find(level => customCodecProfile.includes(level));
  3218. if (codecLevel) {
  3219. codecProfilePrefix = codecProfileMap["264"][codecLevel];
  3220. codecProfileLevelId = `profile-level-id=${codecProfilePrefix}`;
  3221. }
  3222. } else {
  3223. codecMimeType = "video/" + customCodecProfile;
  3224. }
  3225.  
  3226. RTCRtpTransceiver.prototype.originalSetCodecPreferences = RTCRtpTransceiver.prototype.setCodecPreferences;
  3227. RTCRtpTransceiver.prototype.setCodecPreferences = function (codecs) {
  3228. let customizedCodecs = null;
  3229. if (customCodecProfile === '自动') {
  3230.  
  3231. let a = [];
  3232. let b = [];
  3233. let c = [];
  3234. let d = [];
  3235. codecs.slice().forEach((item) => {
  3236. if (item.mimeType == 'video/H264') {
  3237. if (item.sdpFmtpLine.indexOf('id=4d') > -1) {
  3238. a.push(item);
  3239. } else if (item.sdpFmtpLine.indexOf('id=42e') > -1) {
  3240. b.push(item);
  3241. } else if (item.sdpFmtpLine.indexOf('id=420') > -1) {
  3242. c.push(item);
  3243. } else {
  3244. d.push(item);
  3245. }
  3246. } else {
  3247. d.push(item);
  3248. }
  3249. });
  3250. customizedCodecs = a.concat(b, c, d);
  3251.  
  3252. } else {
  3253. customizedCodecs = codecs.slice();
  3254. let insertionIndex = 0;
  3255.  
  3256. customizedCodecs.forEach((codec, index) => {
  3257. if (codecProfileLevelId !== '' && codec.sdpFmtpLine && codec.sdpFmtpLine.includes(codecProfileLevelId)) {
  3258. moveCodecToIndex(customizedCodecs, index, insertionIndex, codec);
  3259. insertionIndex++;
  3260. } else if (codec.mimeType === codecMimeType) {
  3261. moveCodecToIndex(customizedCodecs, index, insertionIndex, codec);
  3262. insertionIndex++;
  3263. }
  3264. });
  3265. }
  3266.  
  3267. try {
  3268. this.originalSetCodecPreferences.apply(this, [customizedCodecs]);
  3269. console.log("编解码偏好配置成功");
  3270. } catch (error) {
  3271. console.log("无法修改编解码配置,将使用默认设置");
  3272. this.originalSetCodecPreferences.apply(this, [codecs]);
  3273. }
  3274. }
  3275. }
  3276.  
  3277. customizeRtcCodecs();
  3278. patchVideoApi();
  3279.  
  3280. let mslogotimeOut = 0;
  3281. function mslogoClickevent(mslogoInterval, s) {
  3282. let mslogodom = $($('header>div>div>button')[1]);
  3283. if (mslogodom.length > 0) {
  3284. clearInterval(mslogoInterval);
  3285. mslogodom = mslogodom.next();
  3286. if (mslogodom.text() == ("⚙️ 设置" + nftxboxversion)) { return; }
  3287. mslogodom.removeAttr('href');
  3288. mslogodom.css("color", 'white');
  3289. mslogodom.text("⚙️ 设置" + nftxboxversion);
  3290. mslogodom.click(() => {
  3291. naifeitian.showSetting();
  3292. });
  3293. setTimeout(() => { mslogoClickevent(mslogoInterval) }, 5000);
  3294. }
  3295. mslogotimeOut = mslogotimeOut + 1;
  3296. if (mslogotimeOut > 10) {
  3297. mslogotimeOut = 0;
  3298. clearInterval(mslogoInterval);
  3299. }
  3300. }
  3301. let mslogoInterval = setInterval(() => {
  3302. mslogoClickevent(mslogoInterval, 3000);
  3303. }, 1000);
  3304.  
  3305. class Dialog {
  3306. constructor(title, className, $content, onClose) {
  3307. const CE = createElement;
  3308.  
  3309. // Create dialog overlay
  3310. this.$overlay = document.querySelector('.bx-dialog-overlay');
  3311. if (!this.$overlay) {
  3312. this.$overlay = CE('div', { 'class': 'bx-dialog-overlay bx-gone' });
  3313. document.documentElement.appendChild(this.$overlay);
  3314. }
  3315.  
  3316. let $close;
  3317. this.onClose = onClose;
  3318. this.$dialog = CE('div', { 'class': `bx-dialog ${className} bx-gone` },
  3319. CE('b', {}, title),
  3320. CE('div', { 'class': 'bx-dialog-content' }, $content),
  3321. $close = CE('button', {}, "关闭"));
  3322.  
  3323. $close.addEventListener('click', e => {
  3324. this.hide(e);
  3325. });
  3326. document.documentElement.appendChild(this.$dialog);
  3327. }
  3328.  
  3329. show() {
  3330. this.$dialog.classList.remove('bx-gone');
  3331. this.$overlay.classList.remove('bx-gone');
  3332. }
  3333.  
  3334. hide(e) {
  3335. this.$dialog.classList.add('bx-gone');
  3336. this.$overlay.classList.add('bx-gone');
  3337. this.onClose && this.onClose(e);
  3338. }
  3339.  
  3340. toggle() {
  3341. this.$dialog.classList.toggle('bx-gone');
  3342. this.$overlay.classList.toggle('bx-gone');
  3343. }
  3344.  
  3345. preload() {
  3346. this.$dialog.classList.add('bx-gone');
  3347. this.$overlay.classList.add('bx-gone');
  3348. }
  3349. }
  3350. let REMOTE_PLAY_CONFIG;
  3351. let IS_REMOTE_PLAYING;
  3352. let REMOTE_PLAY_SERVER;
  3353.  
  3354. class RemotePlay {
  3355. static XCLOUD_TOKEN;
  3356. static XHOME_TOKEN;
  3357. static #CONSOLES;
  3358.  
  3359. static #STATE_LABELS = {
  3360. 'On': "已开机",
  3361. 'Off': "已关机",
  3362. 'ConnectedStandby': "待机中",
  3363. 'Unknown': "未知",
  3364. };
  3365.  
  3366. static get BASE_DEVICE_INFO() {
  3367. return {
  3368. appInfo: {
  3369. env: {
  3370. clientAppId: window.location.host,
  3371. clientAppType: 'browser',
  3372. clientAppVersion: '21.1.98',
  3373. clientSdkVersion: '8.5.3',
  3374. httpEnvironment: 'prod',
  3375. sdkInstallId: '',
  3376. },
  3377. },
  3378. dev: {
  3379. displayInfo: {
  3380. dimensions: {
  3381. widthInPixels: 1920,
  3382. heightInPixels: 1080,
  3383. },
  3384. pixelDensity: {
  3385. dpiX: 1,
  3386. dpiY: 1,
  3387. },
  3388. },
  3389. hw: {
  3390. make: 'Microsoft',
  3391. model: 'unknown',
  3392. sdktype: 'web',
  3393. },
  3394. os: {
  3395. name: 'windows',
  3396. ver: '22631.2715',
  3397. platform: 'desktop',
  3398. },
  3399. browser: {
  3400. browserName: 'chrome',
  3401. browserVersion: '119.0',
  3402. },
  3403. },
  3404. };
  3405. }
  3406.  
  3407. static #dialog;
  3408. static #$content;
  3409. static #$consoles;
  3410.  
  3411. static #initialize() {
  3412. if (RemotePlay.#$content) {
  3413. return;
  3414. }
  3415. const CE = createElement;
  3416. RemotePlay.#$content = CE('div', {}, "获取控制台列表");
  3417. RemotePlay.#dialog = new Dialog(("串流"), '', RemotePlay.#$content);
  3418.  
  3419. RemotePlay.#getXhomeToken(() => {
  3420. RemotePlay.#getConsolesList(() => {
  3421. console.log(RemotePlay.#CONSOLES);
  3422. RemotePlay.#renderConsoles();
  3423. });
  3424. });
  3425. }
  3426.  
  3427. static #renderConsoles() {
  3428. const CE = createElement;
  3429.  
  3430. const $fragment = document.createDocumentFragment();
  3431.  
  3432. if (!RemotePlay.#CONSOLES || RemotePlay.#CONSOLES.length === 0) {
  3433. $fragment.appendChild(CE('span', {}, "未找到主机"));
  3434. } else {
  3435. const $settingNote = CE('p', {});
  3436.  
  3437. const resolutions = [1080, 720];
  3438. const currentResolution = NFTconfig['REMOTE_PLAY_RESOLUTION']['default'];
  3439. const $resolutionSelect = CE('select', {});
  3440. for (const resolution of resolutions) {
  3441. const value = `${resolution}p`;
  3442.  
  3443. const $option = CE('option', { 'value': value }, value);
  3444. if (currentResolution === value) {
  3445. $option.selected = true;
  3446. }
  3447.  
  3448. $resolutionSelect.appendChild($option);
  3449. }
  3450. $resolutionSelect.addEventListener('change', e => {
  3451. const value = $resolutionSelect.value;
  3452.  
  3453. $settingNote.textContent = value === '1080p' ? '✅ ' + "可串流xbox360游戏" : '❌ ' + "不可串流xbox360游戏";
  3454.  
  3455. NFTconfig['REMOTE_PLAY_RESOLUTION']['default'] = value;
  3456. naifeitian.setValue(NFTconfig['REMOTE_PLAY_RESOLUTION']['name'], NFTconfig['REMOTE_PLAY_RESOLUTION']);
  3457. });
  3458. $resolutionSelect.dispatchEvent(new Event('change'));
  3459.  
  3460. const $qualitySettings = CE('div', { 'class': 'bx-remote-play-settings' },
  3461. CE('div', {},
  3462. CE('label', {}, "目标分辨率", $settingNote),
  3463. $resolutionSelect,
  3464. )
  3465. );
  3466.  
  3467. $fragment.appendChild($qualitySettings);
  3468. }
  3469.  
  3470. for (let con of RemotePlay.#CONSOLES) {
  3471. let $connectButton;
  3472. const $child = CE('div', { 'class': 'bx-remote-play-device-wrapper' },
  3473. CE('div', { 'class': 'bx-remote-play-device-info' },
  3474. CE('div', {},
  3475. CE('span', { 'class': 'bx-remote-play-device-name' }, con.deviceName),
  3476. CE('span', { 'class': 'bx-remote-play-console-type' }, con.consoleType)
  3477. ),
  3478. CE('div', { 'class': 'bx-remote-play-power-state' }, RemotePlay.#STATE_LABELS[con.powerState]),
  3479. ),
  3480. $connectButton = CE('button', { 'class': 'bx-primary-button bx-no-margin' }, "连接"),
  3481. );
  3482.  
  3483. $connectButton.addEventListener('click', e => {
  3484. REMOTE_PLAY_CONFIG = {
  3485. serverId: con.serverId,
  3486. };
  3487. window.BX_REMOTE_PLAY_CONFIG = REMOTE_PLAY_CONFIG;
  3488.  
  3489. const url = window.location.href.substring(0, 31) + '/launch/fortnite/BT5P2X999VH2#remote-play';
  3490.  
  3491. const $pageContent = document.getElementById('PageContent');
  3492. const $anchor = CE('a', { href: url, class: 'bx-hidden', style: 'position:absolute;top:-9990px;left:-9999px' }, '');
  3493. $anchor.addEventListener('click', e => {
  3494. setTimeout(() => {
  3495. $pageContent.removeChild($anchor);
  3496. }, 1000);
  3497. });
  3498.  
  3499. $pageContent.appendChild($anchor);
  3500. $anchor.click();
  3501.  
  3502. RemotePlay.#dialog.hide();
  3503. });
  3504. $fragment.appendChild($child);
  3505. }
  3506.  
  3507. RemotePlay.#$content.parentElement.replaceChild($fragment, RemotePlay.#$content);
  3508. }
  3509.  
  3510. static detect() {
  3511.  
  3512. IS_REMOTE_PLAYING = window.location.pathname.includes('/remote-play') || window.location.hash.startsWith('#remote-play');
  3513. if (IS_REMOTE_PLAYING) {
  3514. window.BX_REMOTE_PLAY_CONFIG = REMOTE_PLAY_CONFIG;
  3515. // 移除 /launch/...
  3516. window.history.replaceState({}, '', 'https://www.xbox.com/' + location.pathname.substring(1, 6) + '/remote-play');
  3517.  
  3518. } else {
  3519. window.BX_REMOTE_PLAY_CONFIG = null;
  3520. }
  3521. }
  3522.  
  3523. static #getXhomeToken(callback) {
  3524. if (RemotePlay.XHOME_TOKEN) {
  3525. callback();
  3526. return;
  3527. }
  3528.  
  3529. let GSSV_TOKEN;
  3530. try {
  3531. const xboxUserInfo = JSON.parse(localStorage.getItem('xboxcom_xbl_user_info'));
  3532. GSSV_TOKEN = xboxUserInfo.tokens['http://gssv.xboxlive.com/'].token;
  3533. } catch (e) {
  3534. for (let i = 0; i < localStorage.length; i++) {
  3535. const key = localStorage.key(i);
  3536.  
  3537. if (key.startsWith('Auth.User.')) {
  3538. const json = JSON.parse(localStorage.getItem(key));
  3539.  
  3540. GSSV_TOKEN = json.tokens.find(token => token.relyingParty.includes('gssv.xboxlive.com'))?.tokenData.token;
  3541.  
  3542. if (GSSV_TOKEN) {
  3543. break;
  3544. }
  3545. }
  3546. }
  3547. }
  3548.  
  3549. fetch('https://xhome.gssv-play-prod.xboxlive.com/v2/login/user', {
  3550. method: 'POST',
  3551. body: JSON.stringify({
  3552. offeringId: 'xhome',
  3553. token: GSSV_TOKEN,
  3554. }),
  3555. headers: {
  3556. 'Content-Type': 'application/json; charset=utf-8',
  3557. },
  3558. }).then(resp => resp.json())
  3559. .then(json => {
  3560. RemotePlay.XHOME_TOKEN = json.gsToken;
  3561. callback();
  3562. });
  3563. }
  3564. //获取xbox列表
  3565. static async #getConsolesList(callback) {
  3566. if (RemotePlay.#CONSOLES) {
  3567. callback();
  3568. return;
  3569. }
  3570.  
  3571. let servers;
  3572. if (!REMOTE_PLAY_SERVER) {
  3573. if (NFTconfig['REMOTE_SERVER_LIST'].length == 0) {
  3574. servers = ['wus2', 'eus', 'uks', 'ejp'];
  3575. } else {
  3576. servers = NFTconfig['REMOTE_SERVER_LIST'];
  3577. }
  3578.  
  3579. } else {
  3580. servers = REMOTE_PLAY_SERVER;
  3581. }
  3582.  
  3583. const options = {
  3584. method: 'GET',
  3585. headers: {
  3586. 'Authorization': `Bearer ${RemotePlay.XHOME_TOKEN}`,
  3587. },
  3588. };
  3589.  
  3590. const controller = new AbortController();
  3591. const signal = controller.signal;
  3592.  
  3593. let foundResponse = false;
  3594.  
  3595. const promises = NFTconfig['REMOTE_SERVER_LIST'].map(async (server) => {
  3596. const url = `https://${server}.core.gssv-play-prodxhome.xboxlive.com/v6/servers/home?mr=50`;
  3597.  
  3598. try {
  3599. const response = await fetch(url, { ...options, signal });
  3600. const json = await response.json();
  3601. if (json.results && !foundResponse) {
  3602. foundResponse = true;
  3603. RemotePlay.#CONSOLES = json.results;
  3604. REMOTE_PLAY_SERVER = server;
  3605.  
  3606. // 取消剩余请求
  3607. controller.abort();
  3608.  
  3609. return 'Found response';
  3610. }
  3611.  
  3612. } catch (error) {
  3613. console.log(`请求至服务器 ${server} 时遇到错误,已忽略:`, error);
  3614. }
  3615. });
  3616.  
  3617.  
  3618. // 等待所有请求完成
  3619. const results=await Promise.allSettled(promises);
  3620.  
  3621.  
  3622. // 检查是否有请求成功
  3623. const successfulResult = results.find(result => result.status === 'fulfilled' && result.value === 'Found response');
  3624.  
  3625. if (successfulResult) {
  3626. callback();
  3627. } else {
  3628. RemotePlay.#CONSOLES = [];
  3629. }
  3630.  
  3631. }
  3632.  
  3633. static preload() {
  3634. RemotePlay.#initialize();
  3635. RemotePlay.#dialog.preload();
  3636. }
  3637.  
  3638.  
  3639. static showDialog() {
  3640. RemotePlay.#initialize();
  3641. RemotePlay.#dialog.show();
  3642. }
  3643. }
  3644. function bindmslogoevent() {
  3645. let divElement = $('#gamepass-root > div > div');
  3646. if (divElement.length == 0) {
  3647. setTimeout(() => {
  3648. bindmslogoevent();
  3649. }, 2333);
  3650. return;
  3651. }
  3652. divElement = divElement.get(0);
  3653. let mslogodom = $(divElement).children('header').find('a[href]');
  3654. if (mslogodom.length == 0) {
  3655. setTimeout(() => {
  3656. bindmslogoevent();
  3657. }, 2333);
  3658. return;
  3659. }
  3660. if (mslogodom.length > 0) { mslogodom = $(mslogodom.get(0)); }
  3661. let linkElement = $("a:contains('⚙️ 设置" + nftxboxversion + "')");
  3662. for (let i = 0; i < linkElement.length; i++) {
  3663. let ele = linkElement.get(i);
  3664. if ($(ele).attr('class').indexOf('button') > -1) {
  3665. return;
  3666. }
  3667. }
  3668. mslogodom.removeAttr('href');
  3669. mslogodom.css("color", 'white');
  3670. mslogodom.text("⚙️ 设置" + nftxboxversion);
  3671. mslogodom.click(() => {
  3672. naifeitian.showSetting();
  3673. });
  3674.  
  3675. if (NFTconfig['enableRemotePlay'] == 1) {
  3676. let remotePlayBtn = $('.bx-remote-play-button');
  3677. if (remotePlayBtn.length > 0) { return; }
  3678. //添加串流按钮
  3679. var targetElement = $("[title*='Account Settings']");
  3680.  
  3681. var newButton = $(`<button class="bx-remote-play-button" title="远程串流"><svg fill="none" stroke="#fff" fill-rule="evenodd" stroke-linecap="round" stroke-linejoin="round" stroke-width="4" viewBox="0 0 32 32"><g transform="matrix(.492308 0 0 .581818 -14.7692 -11.6364)"><clipPath id="A"><path d="M30 20h65v55H30z"></path></clipPath><g clip-path="url(#A)"><g transform="matrix(.395211 0 0 .334409 11.913 7.01124)"><g transform="matrix(.555556 0 0 .555556 57.8889 -20.2417)" fill="none" stroke="#fff" stroke-width="13.88"><path d="M200 140.564c-42.045-33.285-101.955-33.285-144 0M168 165c-23.783-17.3-56.217-17.3-80 0"></path></g><g transform="matrix(-.555556 0 0 -.555556 200.111 262.393)"><g transform="matrix(1 0 0 1 0 11.5642)"><path d="M200 129c-17.342-13.728-37.723-21.795-58.636-24.198C111.574 101.378 80.703 109.444 56 129" fill="none" stroke="#fff" stroke-width="13.88"></path></g><path d="M168 165c-23.783-17.3-56.217-17.3-80 0" fill="none" stroke="#fff" stroke-width="13.88"></path></g><g transform="matrix(.75 0 0 .75 32 32)"><path d="M24 72h208v93.881H24z" fill="none" stroke="#fff" stroke-linejoin="miter" stroke-width="9.485"></path><circle cx="188" cy="128" r="12" stroke-width="10" transform="matrix(.708333 0 0 .708333 71.8333 12.8333)"></circle><path d="M24.358 103.5h110" fill="none" stroke="#fff" stroke-linecap="butt" stroke-width="10.282"></path></g></g></g></g></svg></button>`);
  3682. newButton.on("click", function () {
  3683. RemotePlay.showDialog();
  3684. });
  3685.  
  3686. newButton.insertBefore(targetElement);
  3687. }
  3688. setTimeout(() => { bindmslogoevent() }, 5000);
  3689. }
  3690.  
  3691. bindmslogoevent();
  3692.  
  3693. if (window.location.pathname.includes('/play/')) {
  3694. NFTconfig['PATCH_ORDERS'] = NFTconfig['PATCH_ORDERS'].concat(NFTconfig['PLAYING_PATCH_ORDERS']);
  3695. } else {
  3696. NFTconfig['PATCH_ORDERS'].push(['loadingEndingChunks']);
  3697. }
  3698. naifeitian.patchFunctionBind();
  3699. RemotePlay.detect();
  3700.  
  3701.  
  3702. if (window.location.pathname.toLocaleLowerCase() == '/zh-cn/play') {
  3703. window.location.href = "https://www.xbox.com/en-us/play";
  3704. }
  3705. if (window.location.href.endsWith('consoles/remote-play') || window.location.href.endsWith('/remote-play')) {
  3706. //https://www.xbox.com/en-US/consoles/remote-play
  3707. let jurl = window.location.href.replace('/consoles', '');
  3708. jurl = window.location.href.replace('/remote-play', '/play');
  3709. window.location.href = jurl;
  3710. }
  3711.  
  3712. if (window.location.href.endsWith('play/dev-tools')) {
  3713. window.location.href="/play"
  3714. }
  3715.  
  3716.  
  3717.  
  3718. RTCPeerConnection.prototype.orgAddIceCandidate = RTCPeerConnection.prototype.addIceCandidate;
  3719.  
  3720. RTCPeerConnection.prototype.addIceCandidate = function (...args) {
  3721. STREAM_WEBRTC = this;
  3722. return this.orgAddIceCandidate(...args);
  3723. };
  3724.  
  3725.  
  3726. function addCss() {
  3727.  
  3728. let popCss = `
  3729.  
  3730. #popSetting {
  3731. width: 76px;
  3732. height: 33px;
  3733. background: #fff;
  3734. position: absolute;
  3735. top: 30%;
  3736. cursor: pointer;
  3737. box-sizing: border-box;
  3738. background-size: 100% 100%;
  3739. overflow: hidden;
  3740. font-family: Arial;
  3741. font-size: 18px;
  3742. line-height: 30px;
  3743. font-weight: bold;
  3744. color: #000000bf;
  3745. border: 2px solid;
  3746. border-radius: 10px;
  3747. -webkit-user-select: none;
  3748. -moz-user-select: none;
  3749. -ms-user-select: none;
  3750. user-select: none ;
  3751. }
  3752. .better-xcloud-hidden {
  3753. visibility: hidden !important;
  3754. }
  3755. .hd-stream-setting-hide{
  3756. pointer-events: none !important;
  3757. }
  3758. div[class^=HUDButton-module__hiddenContainer] ~ div:not([class^=HUDButton-module__hiddenContainer]) {
  3759. opacity: 0;
  3760. position: absolute;
  3761. top: -9999px;
  3762. left: -9999px;
  3763. }
  3764. .bx-remote-play-button {
  3765. height: auto;
  3766. margin-right: 8px !important;
  3767. position: relative;
  3768. background-color: transparent;
  3769. border:0px;
  3770. border-radius: 50%;
  3771. }
  3772.  
  3773. .bx-remote-play-button svg {
  3774. width: 28px;
  3775. height: 46px;
  3776. }
  3777. .bx-remote-play-button:hover {
  3778. background-color: #515863;
  3779. }
  3780.  
  3781. .better-xcloud-stats-bar {
  3782. display: block;
  3783. user-select: none;
  3784. position: fixed;
  3785. top: 0;
  3786. background-color: #000;
  3787. color: #fff;
  3788. font-family: Consolas, "Courier New", Courier, monospace;
  3789. font-size: 0.9rem;
  3790. padding-left: 8px;
  3791. z-index: 1000;
  3792. text-wrap: nowrap;
  3793. }
  3794.  
  3795. .better-xcloud-stats-bar[data-position=top-left] {
  3796. left: 20px;
  3797. }
  3798.  
  3799. .better-xcloud-stats-bar[data-position=top-right] {
  3800. right: 0;
  3801. }
  3802.  
  3803. .better-xcloud-stats-bar[data-position=top-center] {
  3804. transform: translate(-50%, 0);
  3805. left: 50%;
  3806. }
  3807.  
  3808. .better-xcloud-stats-bar[data-transparent=true] {
  3809. background: none;
  3810. filter: drop-shadow(1px 0 0 #000) drop-shadow(-1px 0 0 #000) drop-shadow(0 1px 0 #000) drop-shadow(0 -1px 0 #000);
  3811. }
  3812.  
  3813. .better-xcloud-stats-bar label {
  3814. margin: 0 8px 0 0;
  3815. font-family: Bahnschrift, Arial, Helvetica, sans-serif;
  3816. font-size: inherit;
  3817. font-weight: bold;
  3818. vertical-align: middle;
  3819. }
  3820.  
  3821. .better-xcloud-stats-bar span {
  3822. min-width: 60px;
  3823. display: inline-block;
  3824. text-align: right;
  3825. padding-right: 8px;
  3826. margin-right: 8px;
  3827. border-right: 2px solid #fff;
  3828. vertical-align: middle;
  3829. }
  3830. .better-xcloud-stats-bar div {
  3831. min-width: 60px;
  3832. display: inline-block;
  3833. text-align: right;
  3834. padding-right: 8px;
  3835. margin-right: 8px;
  3836. border-right: 2px solid #fff;
  3837. vertical-align: middle;
  3838. float:left
  3839. }
  3840.  
  3841. .better-xcloud-stats-bar span[data-grade=good] {
  3842. color: #6bffff;
  3843. }
  3844.  
  3845. .better-xcloud-stats-bar span[data-grade=ok] {
  3846. color: #fff16b;
  3847. }
  3848.  
  3849. .better-xcloud-stats-bar span[data-grade=bad] {
  3850. color: #ff5f5f;
  3851. }
  3852.  
  3853. .better-xcloud-stats-bar span:first-of-type {
  3854. min-width: 30px;
  3855. }
  3856.  
  3857. .better-xcloud-stats-bar span:last-of-type {
  3858. border: 0;
  3859. margin-right: 0;
  3860. }
  3861. .better-xcloud-stats-bar div:last-of-type {
  3862. border: 0;
  3863. margin-right: 0;
  3864. }
  3865.  
  3866. .better-xcloud-stats-settings {
  3867. display: none;
  3868. position: fixed;
  3869. top: 50%;
  3870. left: 50%;
  3871. margin-right: -50%;
  3872. transform: translate(-50%, -50%);
  3873. width: 420px;
  3874. padding: 20px;
  3875. border-radius: 8px;
  3876. z-index: 500;
  3877. background: #1a1b1e;
  3878. color: #fff;
  3879. font-weight: 400;
  3880. font-size: 16px;
  3881. font-family: "Segoe UI", Arial, Helvetica, sans-serif;
  3882. box-shadow: 0 0 6px #000;
  3883. user-select: none;
  3884. overflow-y: auto;
  3885. }
  3886.  
  3887. .better-xcloud-stats-settings *:focus {
  3888. outline: none !important;
  3889. }
  3890.  
  3891. .better-xcloud-stats-settings > b {
  3892. color: #fff;
  3893. display: block;
  3894. font-family: Bahnschrift, Arial, Helvetica, sans-serif;
  3895. font-size: 26px;
  3896. font-weight: 400;
  3897. line-height: 32px;
  3898. margin-bottom: 12px;
  3899. }
  3900.  
  3901. .better-xcloud-stats-settings > div {
  3902. display: flex;
  3903. margin-bottom: 8px;
  3904. padding: 2px 4px;
  3905. }
  3906.  
  3907. .better-xcloud-stats-settings label {
  3908. flex: 1;
  3909. margin-bottom: 0;
  3910. align-self: center;
  3911. }
  3912.  
  3913. .better-xcloud-stats-settings button {
  3914. padding: 8px 32px;
  3915. margin: 20px auto 0;
  3916. border: none;
  3917. border-radius: 4px;
  3918. display: block;
  3919. background-color: #2d3036;
  3920. text-align: center;
  3921. color: white;
  3922. text-transform: uppercase;
  3923. font-family: Bahnschrift, Arial, Helvetica, sans-serif;
  3924. font-weight: 400;
  3925. line-height: 18px;
  3926. font-size: 14px;
  3927. }
  3928.  
  3929. @media (hover: hover) {
  3930. .better-xcloud-stats-settings button:hover {
  3931. background-color: #515863;
  3932. }
  3933. }
  3934.  
  3935. .better-xcloud-stats-settings button:focus {
  3936. background-color: #515863;
  3937. }
  3938.  
  3939. .better-xcloud-gone {
  3940. display: none !important;
  3941. }
  3942.  
  3943. .better-xcloud-quick-settings-bar {
  3944. display: none;
  3945. user-select: none;
  3946. -webkit-user-select: none;
  3947. position: fixed;
  3948. bottom: 10%;
  3949. left: 50%;
  3950. transform: translate(-50%, 0);
  3951. z-index: 9999;
  3952. padding: 16px;
  3953. width: 600px;
  3954. background: #1a1b1e;
  3955. color: #fff;
  3956. border-radius: 8px 8px 0 0;
  3957. font-weight: 400;
  3958. font-size: 14px;
  3959. font-family: Bahnschrift, Arial, Helvetica, sans-serif;
  3960. text-align: center;
  3961. box-shadow: 0px 0px 6px #000;
  3962. opacity: 0.95;
  3963. }
  3964.  
  3965. .better-xcloud-quick-settings-bar *:focus {
  3966. outline: none !important;
  3967. }
  3968.  
  3969. .better-xcloud-quick-settings-bar > div {
  3970. flex: 1;
  3971. }
  3972.  
  3973. .better-xcloud-quick-settings-bar label {
  3974. font-size: 16px;
  3975. display: block;
  3976. margin-bottom: 8px;
  3977. }
  3978.  
  3979. .better-xcloud-quick-settings-bar input {
  3980. width: 22px;
  3981. height: 22px;
  3982. }
  3983.  
  3984. .better-xcloud-quick-settings-bar button {
  3985. border: none;
  3986. width: 22px;
  3987. height: 22px;
  3988. margin: 0 4px;
  3989. line-height: 22px;
  3990. background-color: #515151;
  3991. color: #fff;
  3992. border-radius: 4px;
  3993. }
  3994.  
  3995. @media (hover: hover) {
  3996. .better-xcloud-quick-settings-bar button:hover {
  3997. background-color: #414141;
  3998. color: white;
  3999. }
  4000. }
  4001.  
  4002. .better-xcloud-quick-settings-bar button:active {
  4003. background-color: #414141;
  4004. color: white;
  4005. }
  4006.  
  4007. .better-xcloud-quick-settings-bar span {
  4008. display: inline-block;
  4009. width: 40px;
  4010. font-weight: bold;
  4011. font-family: Consolas, "Courier New", Courier, monospace;
  4012. }
  4013.  
  4014.  
  4015. .closeSetting1 {
  4016. color: #0099CC;
  4017. background: transparent;
  4018. border: 2px solid #0099CC;
  4019. border-radius: 6px;
  4020. border: none;
  4021. color: white;
  4022. padding: 3px 13px;
  4023. text-align: center;
  4024. display: inline-block;
  4025. font-size: 16px;
  4026. margin: 4px 2px;
  4027. -webkit-transition-duration: 0.4s; /* Safari */
  4028. transition-duration: 0.4s;
  4029. cursor: pointer;
  4030. text-decoration: none;
  4031. text-transform: uppercase;
  4032. }
  4033. .closeSetting2 {
  4034. background-color: white;
  4035. color: black;
  4036. border: 2px solid #008CBA;
  4037. display: block;
  4038. margin: 0 auto;
  4039. margin-top: 5px;
  4040. }
  4041. .closeSetting2:hover {
  4042. background-color: #008CBA;
  4043. color: white;
  4044. }
  4045. .settingsBackgroud{
  4046. position: fixed;
  4047. left: 0;
  4048. top: 0;
  4049. background: #0000;
  4050. width: 100%;
  4051. height: 90vh;
  4052. overflow: scroll;
  4053. z-index:8888;
  4054. }
  4055. .settingsBox{
  4056. position: relative;
  4057. background: wheat;
  4058. width: fit-content;
  4059. height: fit-content;
  4060. border-radius: 5px;
  4061. margin: 5% auto;
  4062. padding: 10px;
  4063. font-family: '微软雅黑';
  4064. line-height: 22px;
  4065. top:5%;
  4066. z-index:8889;
  4067. }
  4068. .settingsBoxInputRadio{
  4069. background-color: initial;
  4070. cursor: pointer;
  4071. appearance: auto;
  4072. box-sizing: border-box;
  4073. margin: 3px 3px 0px 5px;
  4074. padding: initial;
  4075. padding-top: initial;
  4076. padding-right: initial;
  4077. padding-bottom: initial;
  4078. padding-left: initial;
  4079. border: initial;
  4080. -webkit-appearance: checkbox;
  4081. accent-color: dodgerblue;
  4082. }
  4083.  
  4084. #StreamHud >div{
  4085. background-color:rgba(255,0,0,0)!important;
  4086. }
  4087.  
  4088. #StreamHud >button{
  4089. background-color:rgba(0,0,0,0)!important;
  4090. }
  4091. #StreamHud >button > div{
  4092. opacity:0.3!important;
  4093. }
  4094.  
  4095. #touchControllerEventArea {
  4096. pointer-events: auto;
  4097. position: fixed;
  4098. bottom: 0;
  4099. right: 0;
  4100. width: 33%;
  4101. height: 6vh;
  4102. z-index: 5678;
  4103. background-color: rgba(0, 0, 0, 0);
  4104. }
  4105. .better-xcloud-badges {
  4106. position: absolute;
  4107. margin-left: 0px;
  4108. user-select: none;
  4109. -webkit-user-select: none;
  4110. bottom: 0px;
  4111. display: none;
  4112. }
  4113. /* 横屏 */
  4114. @media screen and (orientation: landscape) {
  4115. .better-xcloud-badges {
  4116. display: block; /* 显示 */
  4117. }
  4118. }
  4119.  
  4120. /* 竖屏 */
  4121. @media screen and (orientation: portrait) {
  4122. .better-xcloud-badges {
  4123. display: none; /* 隐藏 */
  4124. }
  4125. }
  4126.  
  4127. button[class*=BaseItem-module__container] {
  4128. height:fit-content;
  4129. }
  4130. div[class*=Menu-module__scrollable] {
  4131. height:225px;
  4132. }
  4133. .better-xcloud-badge {
  4134. border: none;
  4135. display: inline-block;
  4136. line-height: 24px;
  4137. color: #fff;
  4138. font-family: Bahnschrift Semibold, Arial, Helvetica, sans-serif;
  4139. font-size: 14px;
  4140. font-weight: 400;
  4141. margin: 0 8px 8px 0;
  4142. box-shadow: 0px 0px 6px #000;
  4143. border-radius: 4px;
  4144. }
  4145.  
  4146. .better-xcloud-badge-name {
  4147. background-color: #2d3036;
  4148. display: inline-block;
  4149. padding: 2px 8px;
  4150. border-radius: 4px 0 0 4px;
  4151. text-transform: uppercase;
  4152. }
  4153.  
  4154. .better-xcloud-badge-value {
  4155. background-color: grey;
  4156. display: inline-block;
  4157. padding: 2px 8px;
  4158. border-radius: 0 4px 4px 0;
  4159. }
  4160.  
  4161. .better-xcloud-badge-battery[data-charging=true] span:first-of-type::after {
  4162. content: ' ⚡️';
  4163. }
  4164.  
  4165. div[class*=NotFocusedDialog-module__container] {
  4166. display:none
  4167. }
  4168. @keyframes blink {
  4169. 20% {color: blueviolet; }
  4170. 50% { color: blue; }
  4171. 100% { color: green; }
  4172. }
  4173.  
  4174. .blink-text {
  4175. font-size: 15px;
  4176. font-weight: bold;
  4177. animation: blink 3s infinite;
  4178. float: right;
  4179. cursor: pointer;
  4180. display:none
  4181. }
  4182. .remote-play-button {
  4183. background-color: transparent;
  4184. border: none;
  4185. color: white;
  4186. font-weight: bold;
  4187. line-height: 30px;
  4188. border-radius: 4px;
  4189. padding: 8px;
  4190. }
  4191.  
  4192. .remote-play-button:hover, .remote-play-button:focus {
  4193. background-color: #515863;
  4194. }
  4195. .bx-dialog-overlay {
  4196. position: fixed;
  4197. inset: 0;
  4198. z-index: var(--bx-dialog-overlay-z-index);
  4199. background: black;
  4200. opacity: 50%;
  4201. }
  4202.  
  4203. .bx-dialog {
  4204. display: flex;
  4205. flex-flow: column;
  4206. max-height: 90vh;
  4207. position: fixed;
  4208. top: 50%;
  4209. left: 50%;
  4210. margin-right: -50%;
  4211. transform: translate(-50%, -50%);
  4212. max-width: 410px;
  4213. width:95%;
  4214. padding: 20px;
  4215. border-radius: 8px;
  4216. z-index: var(--bx-dialog-z-index);
  4217. background: #1a1b1e;
  4218. color: #fff;
  4219. font-weight: 400;
  4220. font-size: 16px;
  4221. font-family: var(--bx-normal-font);
  4222. box-shadow: 0 0 6px #000;
  4223. user-select: none;
  4224. -webkit-user-select: none;
  4225. }
  4226.  
  4227. .bx-dialog *:focus {
  4228. outline: none !important;
  4229. }
  4230.  
  4231. .bx-dialog > b {
  4232. color: #fff;
  4233. display: block;
  4234. font-family: var(--bx-title-font);
  4235. font-size: 26px;
  4236. font-weight: 400;
  4237. line-height: 32px;
  4238. margin-bottom: 12px;
  4239. }
  4240.  
  4241. .bx-dialog > div {
  4242. overflow: auto;
  4243. padding: 2px 0;
  4244. }
  4245.  
  4246. .bx-dialog > button {
  4247. padding: 8px 32px;
  4248. margin: 20px auto 0;
  4249. border: none;
  4250. border-radius: 4px;
  4251. display: block;
  4252. background-color: #2d3036;
  4253. text-align: center;
  4254. color: white;
  4255. text-transform: uppercase;
  4256. font-family: var(--bx-title-font);
  4257. font-weight: 400;
  4258. line-height: 18px;
  4259. font-size: 14px;
  4260. }
  4261. .bx-gone {
  4262. display: none !important;
  4263. }
  4264. .bx-remote-play-settings {
  4265. margin-bottom: 12px;
  4266. padding-bottom: 12px;
  4267. border-bottom: 1px solid #2d2d2d;
  4268. }
  4269.  
  4270. .bx-remote-play-settings > div {
  4271. display: flex;
  4272. }
  4273.  
  4274. .bx-remote-play-settings label {
  4275. flex: 1;
  4276. }
  4277.  
  4278. .bx-remote-play-settings label p {
  4279. margin: 4px 0 0;
  4280. padding: 0;
  4281. color: #888;
  4282. font-size: 12px;
  4283. }
  4284.  
  4285. .bx-remote-play-settings input {
  4286. display: block;
  4287. margin: 0 auto;
  4288. }
  4289.  
  4290. .bx-remote-play-settings span {
  4291. font-weight: bold;
  4292. font-size: 18px;
  4293. display: block;
  4294. margin-bottom: 8px;
  4295. text-align: center;
  4296. }
  4297. .bx-remote-play-device-name {
  4298. font-size: 20px;
  4299. font-weight: bold;
  4300. display: inline-block;
  4301. vertical-align: middle;
  4302. }
  4303. .bx-remote-play-console-type {
  4304. font-size: 12px;
  4305. background: #888;
  4306. color: #fff;
  4307. display: inline-block;
  4308. border-radius: 14px;
  4309. padding: 2px 10px;
  4310. margin-left: 8px;
  4311. vertical-align: middle;
  4312. }
  4313.  
  4314. .bx-remote-play-power-state {
  4315. color: #888;
  4316. font-size: 14px;
  4317. }
  4318. .bx-remote-play-power-state {
  4319. color: #888;
  4320. font-size: 14px;
  4321. }
  4322. .bx-primary-button {
  4323. padding: 8px 32px;
  4324. margin: 10px auto 0;
  4325. border: none;
  4326. border-radius: 4px;
  4327. display: block;
  4328. background-color: #044e2a;
  4329. text-align: center;
  4330. color: white;
  4331. text-transform: uppercase;
  4332. font-family: var(--bx-title-font);
  4333. font-weight: 400;
  4334. font-size: 14px;
  4335. line-height: 24px;
  4336. }
  4337.  
  4338. @media (hover: hover) {
  4339. .bx-primary-button:hover {
  4340. background-color: #00753c;
  4341. }
  4342. }
  4343.  
  4344. .bx-primary-button:focus {
  4345. background-color: #00753c;
  4346. }
  4347.  
  4348. .bx-primary-button:active {
  4349. background-color: #00753c;
  4350. }
  4351.  
  4352. .bx-primary-button[disabled] {
  4353. background: #393939;
  4354. color: #a2a2a2;
  4355. }
  4356. .bx-no-margin {
  4357. margin: 0 !important;
  4358. }
  4359. .bx-remote-play-device-info {
  4360. flex: 1;
  4361. padding: 4px 0;
  4362. }
  4363. .bx-remote-play-device-wrapper {
  4364. display: flex;
  4365. margin-bottom: 8px;
  4366. }
  4367.  
  4368. .bx-remote-play-device-wrapper:not(:last-child) {
  4369. margin-bottom: 14px;
  4370. }
  4371. #xcloud_setting_STATS_BUTTON{
  4372. background-color: cadetblue;
  4373. width:100%;
  4374. }
  4375. .stats-container {
  4376. width: 30%;
  4377. border:1px solid white;
  4378. }
  4379.  
  4380. .drag-handle {
  4381. cursor: pointer;
  4382. margin: 5px;
  4383. padding: 6px;
  4384. border: 1px solid #ccc;
  4385. border-radius: 5px;
  4386. background-color: #f9f9f9;
  4387. transition: background-color 0.2s ease;
  4388. position: relative;
  4389. color:black;
  4390. height: 30px;
  4391. }
  4392.  
  4393. .drag-handle::after {
  4394. content: "≡";
  4395. position: absolute;
  4396. right: 5px;
  4397. top: 50%;
  4398. transform: translateY(-50%);
  4399. color: #999;
  4400. }
  4401.  
  4402. .drag-handle.drag-over {
  4403. background-color: #e0f0ff;
  4404. }
  4405.  
  4406. .drag-handle.dragging {
  4407. opacity: 0.5;
  4408. }
  4409.  
  4410. .drag-handle.stats-selected {
  4411. background-color: #cbffcd;
  4412. }
  4413. .drag-handle.stats-delete {
  4414. text-decoration: line-through;
  4415. }
  4416.  
  4417. .placeholder {
  4418. border: 1px dashed #ccc;
  4419. border-radius: 5px;
  4420. margin: 5px;
  4421. height: 30px;
  4422. display: none;
  4423. background-color:#c5c5c5!important;
  4424. }
  4425.  
  4426. /* 新增样式,用于被拖拽元素的副本 */
  4427. .dragged-copy-item {
  4428. position: absolute;
  4429. z-index: 1000;
  4430. pointer-events: none;
  4431. /* 防止被拖拽元素副本干扰其他事件 */
  4432. display: none;
  4433. width:100px
  4434. }
  4435. `;
  4436. if (NFTconfig['disableTouchControls'] == 1) {
  4437. popCss += `
  4438. #MultiTouchSurface, #BabylonCanvasContainer-main {
  4439. display: none !important;
  4440. }
  4441.  
  4442. `};
  4443.  
  4444. let xfbasicStyle = document.createElement('style');
  4445. xfbasicStyle.innerHTML = popCss;
  4446. let docxf = document.head || document.documentElement;
  4447. docxf.appendChild(xfbasicStyle);
  4448. }
  4449. addCss();
  4450. crturl = window.location.href;
  4451. console.log("all done");
  4452.  
  4453.  
  4454. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址