Xbox Cloud Gaming 游戏振动支持

让 Xbox Cloud Gaming 支持游戏力反馈(振动)功能

  1. // ==UserScript==
  2. // @name Xbox Cloud Gaming Vibration
  3. // @name:zh-CN Xbox Cloud Gaming 游戏振动支持
  4. // @name:zh-TW Xbox Cloud Gaming 游戲振動支持
  5. // @namespace http://tampermonkey.net/
  6. // @version 1.4
  7. // @description Add game force feedback (vibration or rumble) support for Xbox Cloud Gaming
  8. // @description:zh-CN 让 Xbox Cloud Gaming 支持游戏力反馈(振动)功能
  9. // @description:zh-TW 將 Xbox Cloud Gaming 支援游戲力回饋(振動)功能
  10. // @author TGSAN
  11. // @match https://www.xbox.com/*/play*
  12. // @icon 
  13. // @inject-into page
  14. // @run-at document-start
  15. // @grant unsafeWindow
  16. // @grant GM_setValue
  17. // @grant GM_getValue
  18. // @grant GM_registerMenuCommand
  19. // @grant GM_unregisterMenuCommand
  20. // ==/UserScript==
  21.  
  22. (function() {
  23. 'use strict';
  24.  
  25. const useControllerVibration = true;
  26. const useMobileVibration = true;
  27. const lang = navigator.language.toLowerCase();
  28.  
  29. let windowCtx = self.window;
  30. if (self.unsafeWindow) {
  31. console.log("[Xbox Cloud Gaming Vibration] use unsafeWindow mode");
  32. windowCtx = self.unsafeWindow;
  33. } else {
  34. console.log("[Xbox Cloud Gaming Vibration] use window mode (your userscript extensions not support unsafeWindow)");
  35. }
  36.  
  37. let configList = {
  38. "XCLOUD_HAPTIC_IMPULSE_TRIGGERS_EMU": {
  39. "desc": {
  40. "en": "Impulse Triggers Haptic Emulation",
  41. "zh": "脈衝發射鍵觸覺回饋仿真",
  42. "zh-cn": "脉冲扳机触感反馈模拟",
  43. },
  44. "value": "1"
  45. },
  46. "XCLOUD_HAPTIC_CONTROLLER_ENABLE": {
  47. "desc": {
  48. "en": "Gamepad Haptic ",
  49. "zh": "游戲控制器觸覺回饋",
  50. "zh-cn": "游戏控制器触感反馈",
  51. },
  52. "value": "1"
  53. },
  54. "XCLOUD_HAPTIC_DEVICE_ENABLE": {
  55. "desc": {
  56. "en": "Device Haptic (Tablet or Mobile)",
  57. "zh": "裝置觸覺回饋(平板電腦或手機)",
  58. "zh-cn": "设备触感反馈(平板电脑或手机)",
  59. },
  60. "value": "1"
  61. },
  62. "XCLOUD_HAPTIC_DEVICE_AUTO_DISABLE": {
  63. "desc": {
  64. "en": "Disable Device Haptic When Using Gamepad",
  65. "zh": "使用游戲控制器時停用裝置觸覺回饋",
  66. "zh-cn": "使用游戏控制器时禁用设备触感反馈",
  67. },
  68. "value": "1"
  69. }
  70. }
  71. let menuItemList = [];
  72.  
  73. function checkSelected(key) {
  74. let value = GM_getValue(key);
  75. if (value === undefined) {
  76. GM_setValue(key, configList[key].value);
  77. }
  78. return value == "1";
  79. }
  80.  
  81. function registerSwitchMenuItem(key) {
  82. let configItem = configList[key];
  83. let name = configItem["desc"]["en"];
  84. let blurMatch = configItem["desc"][lang.substr(0, 2)];
  85. let match = configItem["desc"][lang];
  86. if (match) {
  87. name = match;
  88. } else if (blurMatch) {
  89. name = blurMatch;
  90. }
  91. let isSelected = checkSelected(key);
  92. return GM_registerMenuCommand((isSelected ? "✅" : "🔲") + " " + name, function() {
  93. GM_setValue(key, isSelected ? "0" : "1");
  94. loadAndUpdateSwitchMenuItem();
  95. });
  96. }
  97.  
  98. async function loadAndUpdateSwitchMenuItem() {
  99. for(let command of menuItemList) {
  100. await GM_unregisterMenuCommand(command);
  101. }
  102. menuItemList = [];
  103. let configKeys = Object.keys(configList);
  104. for(let configKey of configKeys) {
  105. configList[configKey].value = checkSelected(configKey) ? "1" : "0";
  106. menuItemList.push(await registerSwitchMenuItem(configKey));
  107. }
  108. // Apply
  109. haptic.enableControllerHaptic = checkSelected("XCLOUD_HAPTIC_CONTROLLER_ENABLE");
  110. haptic.enableDeviceHaptic = checkSelected("XCLOUD_HAPTIC_DEVICE_ENABLE");
  111. haptic.alwaysEnableDeviceHaptic = !checkSelected("XCLOUD_HAPTIC_DEVICE_AUTO_DISABLE");
  112. }
  113.  
  114. let haptic = null;
  115. const xinputMaxHaptic = 65535;
  116.  
  117. windowCtx.RTCPeerConnection.prototype.originalCreateDataChannelXCGV = windowCtx.RTCPeerConnection.prototype.createDataChannel;
  118. windowCtx.RTCPeerConnection.prototype.createDataChannel = function (...params) {
  119. let dc = this.originalCreateDataChannelXCGV(...params);
  120. if (dc.label == "input") {
  121. dc.addEventListener("message", function (de) {
  122. if (typeof(de.data) == "object") {
  123. let dataBytes = new Uint8Array(de.data);
  124. if (dataBytes[0] == 128) {
  125. const leftM = dataBytes[3] / 255;
  126. const rightM = dataBytes[4] / 255;
  127. const leftT = dataBytes[5] / 255;
  128. const rightT = dataBytes[6] / 255;
  129. let wLeftMotorSpeed = leftM * xinputMaxHaptic;
  130. let wRightMotorSpeed = rightM * xinputMaxHaptic;
  131. if (checkSelected("XCLOUD_HAPTIC_IMPULSE_TRIGGERS_EMU")) {
  132. wRightMotorSpeed = Math.max(wRightMotorSpeed, leftT * xinputMaxHaptic, rightT * xinputMaxHaptic);
  133. }
  134. if (haptic) {
  135. haptic.SetState(wLeftMotorSpeed, wRightMotorSpeed);
  136. }
  137. }
  138. }
  139. });
  140. dc.addEventListener("close", function () {
  141. if (haptic) haptic.SetState(0, 0);
  142. });
  143. }
  144. return dc;
  145. }
  146.  
  147. // WebHaptic.ts Compile with Webpack, using Polify, disable UglifyJS
  148. var __classPrivateFieldGet = this && this.__classPrivateFieldGet || function (t, e, i, a) {
  149. if (i === "a" && !a) throw new TypeError("Private accessor was defined without a getter");
  150. if (typeof e === "function" ? t !== e || !a : !e.has(t)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
  151. return i === "m" ? a : i === "a" ? a.call(t) : a ? a.value : e.get(t)
  152. };
  153. var __classPrivateFieldSet = this && this.__classPrivateFieldSet || function (t, e, i, a, s) {
  154. if (a === "m") throw new TypeError("Private method is not writable");
  155. if (a === "a" && !s) throw new TypeError("Private accessor was defined without a setter");
  156. if (typeof e === "function" ? t !== e || !s : !e.has(t)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
  157. return a === "a" ? s.call(t, i) : s ? s.value = i : e.set(t, i), i
  158. };
  159. var _WebHapticV2_enableControllerHaptic, _WebHapticV2_enableDeviceHaptic;
  160. class WebHapticV2 {
  161. set enableControllerHaptic(t) {
  162. var e;
  163. if (__classPrivateFieldGet(this, _WebHapticV2_enableControllerHaptic, "f") != t) {
  164. __classPrivateFieldSet(this, _WebHapticV2_enableControllerHaptic, t, "f");
  165. if (t) {
  166. this.controllerHaptic = new WebControllerHaptic
  167. } else {
  168. (e = this.controllerHaptic) === null || e === void 0 ? void 0 : e.Dispose();
  169. this.controllerHaptic = undefined
  170. }
  171. }
  172. }
  173. get enableControllerHaptic() {
  174. return __classPrivateFieldGet(this, _WebHapticV2_enableControllerHaptic, "f")
  175. }
  176. set enableDeviceHaptic(t) {
  177. var e;
  178. if (__classPrivateFieldGet(this, _WebHapticV2_enableDeviceHaptic, "f") != t) {
  179. __classPrivateFieldSet(this, _WebHapticV2_enableDeviceHaptic, t, "f");
  180. if (t) {
  181. this.deviceHaptic = new WebDeviceHaptic
  182. } else {
  183. (e = this.deviceHaptic) === null || e === void 0 ? void 0 : e.Dispose();
  184. this.deviceHaptic = undefined
  185. }
  186. }
  187. }
  188. get enableDeviceHaptic() {
  189. return __classPrivateFieldGet(this, _WebHapticV2_enableDeviceHaptic, "f")
  190. }
  191. constructor(t = 0) {
  192. _WebHapticV2_enableControllerHaptic.set(this, false);
  193. _WebHapticV2_enableDeviceHaptic.set(this, false);
  194. this.alwaysEnableDeviceHaptic = false;
  195. this.updateTimeoutMs = t;
  196. this.enableDeviceHaptic = false;
  197. this.enableControllerHaptic = false
  198. }
  199. SetState(t, e) {
  200. if (this.updateTimeoutId) {
  201. clearTimeout(this.updateTimeoutId)
  202. }
  203. let i = false;
  204. if (this.controllerHaptic !== undefined) {
  205. i = this.controllerHaptic.GetHapticGamepadsCount() > 0;
  206. this.controllerHaptic.SetState(t, e)
  207. }
  208. if (this.deviceHaptic !== undefined) {
  209. if (this.alwaysEnableDeviceHaptic || !i) {
  210. this.deviceHaptic.SetState(t, e)
  211. } else {
  212. this.deviceHaptic.SetState(0, 0)
  213. }
  214. }
  215. if (this.updateTimeoutMs > 0) {
  216. if (t > 0 || e > 0) {
  217. this.updateTimeoutId = setTimeout(() => {
  218. this.updateTimeoutId = undefined;
  219. this.SetState(0, 0)
  220. }, this.updateTimeoutMs)
  221. }
  222. }
  223. }
  224. Dispose() {
  225. this.SetState(0, 0);
  226. this.enableControllerHaptic = false;
  227. this.enableDeviceHaptic = false
  228. }
  229. }
  230. _WebHapticV2_enableControllerHaptic = new WeakMap, _WebHapticV2_enableDeviceHaptic = new WeakMap;
  231. class WebDeviceHaptic {
  232. constructor() {
  233. this.tickSliceCount = 100;
  234. this.tickSliceMs = 10;
  235. this.rangeTirm = 8;
  236. this.supportDeviceHaptic = false;
  237. this.pwmTerminateTick = 0;
  238. this.supportDeviceHaptic = WebDeviceHaptic.IsSupport()
  239. }
  240. Dispose() {
  241. this.SetState(0, 0)
  242. }
  243. SetState(t, e) {
  244. this.SetWebHapticState(t, e)
  245. }
  246. getAdvancedVibrateMotorPercent(t) {
  247. const e = .75;
  248. const i = -.1;
  249. const a = 1 / (e + i * t);
  250. return Math.pow(t, a)
  251. }
  252. SetWebHapticState(a, s) {
  253. if (this.supportDeviceHaptic) {
  254. let t = .5;
  255. let e = 65535;
  256. let i = Math.max(a, s * t);
  257. if (i > 0) {
  258. let t = this.getAdvancedVibrateMotorPercent(i / e);
  259. this.pwmTerminateTick = Math.round(this.tickSliceCount / this.rangeTirm * t);
  260. const n = this.tickSliceCount * this.tickSliceMs * this.rangeTirm;
  261. if (this.hapticPwmIntervalId === undefined) {
  262. let t = 0;
  263. this.hapticPwmIntervalId = setInterval(() => {
  264. if (t == 0) {
  265. window.navigator.vibrate(n)
  266. }
  267. if (t < this.pwmTerminateTick) {
  268. t++
  269. } else {
  270. t = 0
  271. }
  272. }, this.tickSliceMs)
  273. }
  274. } else {
  275. if (this.hapticPwmIntervalId !== undefined) {
  276. clearInterval(this.hapticPwmIntervalId);
  277. this.hapticPwmIntervalId = undefined
  278. }
  279. window.navigator.vibrate(0)
  280. }
  281. }
  282. }
  283. static IsSupport() {
  284. if (!!window.navigator.vibrate) {
  285. return true
  286. } else {
  287. return false
  288. }
  289. }
  290. }
  291. class WebControllerHaptic {
  292. constructor() {
  293. this.magnitudeDurationMs = 1e3;
  294. this.supportControllerHaptic = false;
  295. this.gamepads = [];
  296. this.hapticGamepadsCount = 0;
  297. this.supportControllerHaptic = WebControllerHaptic.IsSupport();
  298. this.onGamepadConnected = t => {
  299. console.log("A gamepad was connected:" + t.gamepad.id);
  300. this.UpdateGamepads()
  301. };
  302. this.onGamepadDisonnected = t => {
  303. console.log("A gamepad was disconnected:" + t.gamepad.id);
  304. this.UpdateGamepads()
  305. };
  306. if (this.supportControllerHaptic) {
  307. window.addEventListener("gamepadconnected", this.onGamepadConnected);
  308. window.addEventListener("gamepaddisconnected", this.onGamepadDisonnected);
  309. this.UpdateGamepads()
  310. }
  311. }
  312. GetHapticGamepadsCount() {
  313. return this.hapticGamepadsCount
  314. }
  315. Dispose() {
  316. this.SetState(0, 0);
  317. if (this.supportControllerHaptic) {
  318. window.removeEventListener("gamepadconnected", this.onGamepadConnected);
  319. window.removeEventListener("gamepaddisconnected", this.onGamepadDisonnected)
  320. }
  321. }
  322. SetState(t, e) {
  323. this.SetControllerState(t, e)
  324. }
  325. SetControllerState(a, s) {
  326. var n, o, r;
  327. if (this.hapticTimeoutId != undefined) {
  328. clearTimeout(this.hapticTimeoutId);
  329. this.hapticTimeoutId = undefined
  330. }
  331. if (this.supportControllerHaptic) {
  332. let t = 65535;
  333. let e = a / t;
  334. let i = s / t;
  335. for (const [c, l] of Object.entries(this.gamepads)) {
  336. if (l != null) {
  337. (n = l === null || l === void 0 ? void 0 : l.vibrationActuator) === null || n === void 0 ? void 0 : n.playEffect("dual-rumble", {
  338. duration: this.magnitudeDurationMs,
  339. strongMagnitude: e,
  340. weakMagnitude: i
  341. });
  342. if (l.hapticActuators != null) {
  343. (o = l.hapticActuators[0]) === null || o === void 0 ? void 0 : o.pulse(e, this.magnitudeDurationMs);
  344. (r = l.hapticActuators[1]) === null || r === void 0 ? void 0 : r.pulse(i, this.magnitudeDurationMs)
  345. }
  346. }
  347. }
  348. if (a > 0 || s > 0) {
  349. this.hapticTimeoutId = setTimeout(() => {
  350. this.hapticTimeoutId = undefined;
  351. this.SetControllerState(a, s)
  352. }, this.magnitudeDurationMs + 15)
  353. }
  354. }
  355. }
  356. UpdateGamepads() {
  357. this.gamepads = navigator.getGamepads();
  358. let e = 0;
  359. this.gamepads.forEach(t => {
  360. if (t != null) {
  361. if (t.vibrationActuator != null) {
  362. e++
  363. } else if (t.hapticActuators != null && t.hapticActuators.length > 0) {
  364. e++
  365. }
  366. }
  367. });
  368. this.hapticGamepadsCount = e
  369. }
  370. static IsSupport() {
  371. var t, e, i, a;
  372. if (!!window.Gamepad && (((e = (t = window.GamepadHapticActuator) === null || t === void 0 ? void 0 : t.prototype) === null || e === void 0 ? void 0 : e.hasOwnProperty("playEffect")) || ((a = (i = window.GamepadHapticActuator) === null || i === void 0 ? void 0 : i.prototype) === null || a === void 0 ? void 0 : a.hasOwnProperty("pulse")))) {
  373. return true
  374. } else {
  375. return false
  376. }
  377. }
  378. }
  379.  
  380. windowCtx.xcloudHaptic = new WebHapticV2();
  381. haptic = windowCtx.xcloudHaptic;
  382.  
  383. loadAndUpdateSwitchMenuItem();
  384. })();

QingJ © 2025

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