Youtube double language subtitle / Youtube 双语字幕

Youtube double language subtitle / Youtube 双语字幕. 如果不能自动加载,请关闭字幕再次打开即可。默认语言为浏览器首选语言。

  1. // ==UserScript==
  2. // @name Youtube double language subtitle / Youtube 双语字幕
  3. // @version 2023.1.2
  4. // @description Youtube double language subtitle / Youtube 双语字幕. 如果不能自动加载,请关闭字幕再次打开即可。默认语言为浏览器首选语言。
  5. // @author Coink
  6. // @match *://www.youtube.com/watch?v=*
  7. // @match *://www.youtube.com
  8. // @match *://www.youtube.com/*
  9. // @match *://m.youtube.com/watch?v=*
  10. // @require https://unpkg.com/ajax-hook@2.1.3/dist/ajaxhook.min.js
  11. // @grant none
  12. // @run-at document-start
  13. // @namespace https://github.com/CoinkWang/Y2BDoubleSubs
  14. // ==/UserScript==
  15.  
  16. (function() {
  17. let localeLang = document.documentElement.lang || navigator.language ||
  18. // localeLang = 'zh' // uncomment this line to define the language you wish here
  19. 'en' // follow the language used in YouTube Page
  20. ah.proxy({
  21. onRequest: (config, handler) => {
  22. handler.next(config);
  23. },
  24. onResponse: (response, handler) => {
  25. if (response.config.url.includes('/api/timedtext') && !response.config.url.includes('&translate_h00ked')) {
  26. let xhr = new XMLHttpRequest();
  27. // Use RegExp to clean '&tlang=...' in our xhr request params while using Y2B auto translate.
  28. let url = response.config.url
  29. url = url.replace(/(^|[&?])tlang=[^&]*/g, '')
  30. url = `${url}&tlang=${localeLang}&translate_h00ked`
  31. xhr.open('GET', url, false);
  32. xhr.send();
  33. let defaultJson = null
  34. if (response.response) {
  35. const jsonResponse = JSON.parse(response.response)
  36. if (jsonResponse.events) defaultJson = jsonResponse
  37. }
  38. const localeJson = JSON.parse(xhr.response)
  39. let isOfficialSub = true;
  40. for (const defaultJsonEvent of defaultJson.events) {
  41. if (defaultJsonEvent.segs && defaultJsonEvent.segs.length > 1) {
  42. isOfficialSub = false;
  43. break;
  44. }
  45. }
  46. // Merge default subs with locale language subs
  47. if (isOfficialSub) {
  48. // when length of segments are the same
  49. for (let i = 0, len = defaultJson.events.length; i < len; i++) {
  50. const defaultJsonEvent = defaultJson.events[i]
  51. if (!defaultJsonEvent.segs) continue
  52. const localeJsonEvent = localeJson.events[i]
  53. if (`${defaultJsonEvent.segs[0].utf8}`.trim() !== `${localeJsonEvent.segs[0].utf8}`.trim()) {
  54. // avoid merge subs while the are the same
  55. defaultJsonEvent.segs[0].utf8 = localeJsonEvent.segs[0].utf8 + '\n' + defaultJsonEvent.segs[0].utf8
  56. }
  57. }
  58. response.response = JSON.stringify(defaultJson)
  59. } else {
  60. // when length of segments are not the same (e.g. automatic generated english subs)
  61. let pureLocalEvents = localeJson.events.filter(event => event.aAppend !== 1 && event.segs)
  62. for (const defaultJsonEvent of defaultJson.events) {
  63. if (!defaultJsonEvent.segs) continue
  64. let currentStart = defaultJsonEvent.tStartMs,
  65. currentEnd = currentStart + defaultJsonEvent.dDurationMs
  66. for (const localeJsonEvent of pureLocalEvents) {
  67. if (localeJsonEvent.tStartMs === currentStart) {
  68. defaultJsonEvent.segs[0].utf8 = localeJsonEvent.segs[0].utf8 + '\n' + defaultJsonEvent.segs[0].utf8
  69. break;
  70. }
  71. }
  72. }
  73. response.response = JSON.stringify(defaultJson)
  74. }
  75. }
  76. handler.resolve(response)
  77. }
  78. })
  79. })();

QingJ © 2025

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