Greasy Fork镜像 支持简体中文。

Youtube double language subtitle / Youtube 双语字幕

Youtube double language subtitle / Youtube 双语字幕. 如果不能自动加载,请关闭字幕再次打开即可。默认语言为浏览器首选语言。已知Bug:默认为zh-TW繁体中文时,即使请求简体中文翻译,Youtube依旧会返回繁体,已向 Youtube 提交 Issue,有新动向此处代码也会同步修改。

目前為 2020-03-28 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name Youtube double language subtitle / Youtube 双语字幕
  3. // @version 1.4
  4. // @description Youtube double language subtitle / Youtube 双语字幕. 如果不能自动加载,请关闭字幕再次打开即可。默认语言为浏览器首选语言。已知Bug:默认为zh-TW繁体中文时,即使请求简体中文翻译,Youtube依旧会返回繁体,已向 Youtube 提交 Issue,有新动向此处代码也会同步修改。
  5. // @author Coink
  6. // @match *://www.youtube.com/watch?v=*
  7. // @match *://www.youtube.com
  8. // @match *://www.youtube.com/*
  9. // @require https://unpkg.com/xhook@latest/dist/xhook.min.js
  10. // @grant none
  11. // @namespace https://github.com/CoinkWang/Y2BDoubleSubs
  12. // ==/UserScript==
  13.  
  14. (function() {
  15. let localeLang = navigator.language ? navigator.language : 'en'
  16. // localeLang = 'zh' // uncomment this line to define the language you wish here
  17. xhook.after(function (request, response) {
  18. if (request.url.includes('/api/timedtext') && !request.url.includes('&translate_h00ked')) {
  19. let xhr = new XMLHttpRequest();
  20. // Use RegExp to clean '&tlang=...' in our xhr request params while using Y2B auto translate.
  21. let reg = new RegExp("(^|[&?])tlang=([^&]*)",'g');
  22. xhr.open('GET', `${request.url.replace(reg,'')}&tlang=${localeLang}&translate_h00ked`, false);
  23. xhr.send();
  24. let defaultJson = null, localeJson = null;
  25. if (response.data && JSON.parse(response.data).events){
  26. defaultJson = JSON.parse(response.data)
  27. }
  28. localeJson = JSON.parse(xhr.response)
  29. // Merge default subs with locale language subs
  30. if (defaultJson.events.length === localeJson.events.length) {
  31. // when length of segments are the same
  32. for (let i = 0,len = defaultJson.events.length; i<len; i++) {
  33. defaultJson.events[i].segs[0].utf8 += ('\n' + localeJson.events[i].segs[0].utf8)
  34. response.text = JSON.stringify(defaultJson)
  35. }
  36. } else {
  37. // when length of segments are not the same (e.g. automatic generated english subs)
  38. let pureEvents = defaultJson.events.filter(event => event.aAppend !== 1 && event.segs)
  39. for (let i = 0,len = localeJson.events.length; i<len; i++) {
  40. let currentLocaleEvent = localeJson.events[i]
  41. let currentRawEvents = pureEvents.filter(pe => currentLocaleEvent.tStartMs <= pe.tStartMs && pe.tStartMs < currentLocaleEvent.tStartMs + currentLocaleEvent.dDurationMs)
  42. let line = '';
  43. currentRawEvents.forEach(ev => {
  44. ev.segs.forEach(seg => line += seg.utf8);
  45. line += ' '; // add space to avoid words stick together
  46. })
  47. localeJson.events[i].segs[0].utf8 = line + '\n' + localeJson.events[i].segs[0].utf8
  48. response.text = JSON.stringify(localeJson)
  49. }
  50. }
  51. }
  52. });
  53. })();

QingJ © 2025

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