阿里云盘字幕

让你的视频文件和字幕文件梦幻联动!

  1. // ==UserScript==
  2. // @name 阿里云盘字幕
  3. // @namespace http://tampermonkey.net/
  4. // @version 0.7
  5. // @description 让你的视频文件和字幕文件梦幻联动!
  6. // @author polygon
  7. // @match https://www.aliyundrive.com/drive*
  8. // @icon 
  9. // @grant GM_addStyle
  10. // @grant GM_xmlhttpRequest
  11. // @runat document-end
  12. // ==/UserScript==
  13. !function(f){"object"==typeof exports&&"undefined"!=typeof module?module.exports=f():"function"==typeof define&&define.amd?define([],f):("undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this).languageEncoding=f()}(function(){return function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var p="function"==typeof require&&require;if(!f&&p)return p(i,!0);if(u)return u(i,!0);throw(p=new Error("Cannot find module '"+i+"'")).code="MODULE_NOT_FOUND",p}p=n[i]={exports:{}},e[i][0].call(p.exports,function(r){return o(e[i][1][r]||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}({1:[function(require,module,exports){const byteOrderMarks=require("../config/byteOrderMarkObject.js");module.exports=uInt8Start=>{for(const element of byteOrderMarks)if(element.regex.test(uInt8Start))return element.encoding;return null}},{"../config/byteOrderMarkObject.js":6}],2:[function(require,module,exports){module.exports=content=>{for(let b=0;b<content.length;b++)if("�"===content[b])return!1;return!0}},{}],3:[function(require,module,exports){const countAllMatches=require("./processing-content/countAllMatches.js"),calculateConfidenceScore=require("./processing-content/calculateConfidenceScore.js");module.exports=(data,fileInfo)=>{data.languageArr=countAllMatches(data,fileInfo.encoding),fileInfo.language=data.languageArr.reduce((acc,val)=>acc.count>val.count?acc:val).name,data.pos=data.languageArr.findIndex(elem=>elem.name===fileInfo.language),fileInfo.encoding||(fileInfo.encoding=data.languageArr[data.pos].encoding);var calculations=calculateConfidenceScore(data,fileInfo);return data.testFile?calculations:(fileInfo.confidence.encoding||(fileInfo.confidence.encoding=calculations),fileInfo.confidence.language=calculations,data.languageArr[data.pos].count||(fileInfo.language=null,fileInfo.confidence.language=null),fileInfo)}},{"./processing-content/calculateConfidenceScore.js":4,"./processing-content/countAllMatches.js":5}],4:[function(require,module,exports){module.exports=(surplus,fileInfo)=>{var confidenceRaise=new RegExp(/\d|\n|\s|\-|\.|\,|\:|\;|\?|\!|\<|\>|\[|\]|\{|\}|\&|\=|\|/,"g"),totalCharacters=surplus.content.replace(confidenceRaise,"").length;const langArr=surplus.languageArr;var pos=surplus.pos;const testFile=surplus.testFile;var secondLanguage=langArr.reduce((acc,val)=>acc.name!==fileInfo.language&&(val.name===fileInfo.language||acc.count>=val.count)?acc:val);const languageRatio=langArr[pos].count/(secondLanguage.count+langArr[pos].count),characterWordRatio=langArr[pos].count/totalCharacters;let lowerLimit=null,upperLimit=null;upperLimit="UTF-8"===fileInfo.encoding||"UTF-16LE"===fileInfo.encoding?(lowerLimit=langArr[pos].utfFrequency?.8*langArr[pos].utfFrequency.low:null,langArr[pos].utfFrequency?(langArr[pos].utfFrequency.low+langArr[pos].utfFrequency.high)/2:null):(lowerLimit=langArr[pos].isoFrequency?.8*langArr[pos].isoFrequency.low:null,langArr[pos].isoFrequency?(langArr[pos].isoFrequency.low+langArr[pos].isoFrequency.high)/2:null);let confidenceScore;return confidenceScore=lowerLimit&&upperLimit?characterWordRatio>=upperLimit?1:characterWordRatio>lowerLimit?(confidenceRaise=upperLimit-lowerLimit,surplus=characterWordRatio-lowerLimit,confidenceRaise=(1-languageRatio)*(surplus/confidenceRaise),Number((languageRatio+confidenceRaise).toFixed(2))):Number((languageRatio*(characterWordRatio/lowerLimit)).toFixed(2)):null,testFile?{name:testFile.substr(testFile.lastIndexOf("/")+1),path:testFile,encoding:fileInfo.encoding,language:fileInfo.language,languageConfidence:confidenceScore,ratio:Number(languageRatio.toFixed(2)),count:langArr[pos].count,totalCharacters:totalCharacters,characterWordRatio:characterWordRatio.toFixed(6),secondLanguage:{name:secondLanguage.name,count:secondLanguage.count}}:confidenceScore}},{}],5:[function(require,module,exports){const languageArr=require("../../config/languageObject.js");module.exports=(data,encoding)=>{const newLanguageArr=[];languageArr.forEach(obj=>{const updatedLangObj={};Object.keys(obj).forEach(key=>{"count"!==key?updatedLangObj[key]=obj[key]:updatedLangObj.count=0}),newLanguageArr.push(updatedLangObj)});const regex=encoding?"utfRegex":"isoRegex";return newLanguageArr.forEach(lang=>{var matches;!lang[regex]||(matches=data.content.match(lang[regex]))&&(lang.count=matches.length)}),newLanguageArr}},{"../../config/languageObject.js":7}],6:[function(require,module,exports){module.exports=[{encoding:"UTF-EBCDIC",regex:new RegExp("221 115 102 115")},{encoding:"GB-18030",regex:new RegExp("132 49 149 51")},{encoding:"UTF-32LE",regex:new RegExp("255 254 0 0")},{encoding:"UTF-32BE",regex:new RegExp("0 0 254 255")},{encoding:"UTF-8",regex:new RegExp("239 187 191")},{encoding:"UTF-7",regex:new RegExp("43 47 118")},{encoding:"UTF-1",regex:new RegExp("247 100 76")},{encoding:"SCSU",regex:new RegExp("14 254 255")},{encoding:"BOCU-1",regex:new RegExp("251 238 40")},{encoding:"UTF-16BE",regex:new RegExp("254 255")},{encoding:"UTF-16LE",regex:new RegExp("255 254")}]},{}],7:[function(require,module,exports){var sharedRegex={czech:new RegExp(/jsem|jsi/,"gi"),hungarian:new RegExp(/\snem\s/,"gi"),slovak:new RegExp(/poriadku|myslím|\ssme\s/,"gi"),slovenian:new RegExp(/\skaj\s|lahko|zdaj/,"gi"),albanian:new RegExp(/nuk/,"gi"),english:new RegExp(/ the /,"gi"),french:new RegExp(/c'est/,"gi"),portuguese:new RegExp(/ não /,"gi"),spanish:new RegExp(/estaba|\smuy\s|siempre|ahora/,"gi"),german:new RegExp(/\sdas\s/,"gi"),italian:new RegExp(/\sche\s/,"gi"),danish:new RegExp(/hvad|noget/,"gi"),norwegian:new RegExp(/deg/,"gi"),swedish:new RegExp(/ jag /,"gi"),dutch:new RegExp(/ het /,"gi"),finnish:new RegExp(/hän/,"gi"),"serbo-croatian":new RegExp(/ sam | kako /,"gi"),estonian:new RegExp(/\sseda\s|\spole\s|midagi/,"gi"),icelandic:new RegExp(/Það/,"gi"),"malay-indonesian":new RegExp(/tidak/,"gi"),turkish:new RegExp(/ bir /,"gi"),lithuanian:new RegExp(/taip|\stai\s/,"gi"),bengali:new RegExp(/এটা/,"gi"),hindi:new RegExp(/हैं/,"gi"),urdu:new RegExp(/ایک/,"gi"),vietnamese:new RegExp(/ không /,"gi")},sharedFrequency={polish:{low:.004355,high:.005102},czech:{low:.004433,high:.007324},hungarian:{low:.004994,high:.005183},romanian:{low:.003319,high:.00419},slovak:{low:.001736,high:.002557},slovenian:{low:.004111,high:.004959},albanian:{low:.003773,high:.007313},ukrainian:{low:.002933,high:.005389},english:{low:.004679,high:.00758},french:{low:.003016,high:.004825},portuguese:{low:.003406,high:.005032},spanish:{low:.002348,high:.002881},german:{low:.004044,high:.004391},italian:{low:.003889,high:.005175},danish:{low:.00363,high:.004189},norwegian:{low:.00241,high:.003918},swedish:{low:.004916,high:.007221},dutch:{low:.003501,high:.00415},finnish:{low:.003308,high:.005135},"serbo-croatian":{low:.002568,high:.005182},estonian:{low:.002892,high:.003963},icelandic:{low:.004366,high:.004366},"malay-indonesian":{low:.002825,high:.003932},greek:{low:.00344,high:.004862},turkish:{low:.002915,high:.004588},hebrew:{low:.003663,high:.004666},lithuanian:{low:.003277,high:.003768},bengali:{low:.003155,high:.005236},hindi:{low:.004159,high:.006478},urdu:{low:.004118,high:.005851},vietnamese:{low:.003387,high:.005191}};module.exports=[{name:"polish",count:0,utfRegex:new RegExp(/się/,"gi"),isoRegex:new RegExp(/siê/,"gi"),encoding:"CP1250",utfFrequency:sharedFrequency.polish,isoFrequency:sharedFrequency.polish},{name:"czech",count:0,utfRegex:sharedRegex.czech,isoRegex:sharedRegex.czech,encoding:"CP1250",utfFrequency:sharedFrequency.czech,isoFrequency:sharedFrequency.czech},{name:"hungarian",count:0,utfRegex:sharedRegex.hungarian,isoRegex:sharedRegex.hungarian,encoding:"CP1250",utfFrequency:sharedFrequency.hungarian,isoFrequency:sharedFrequency.hungarian},{name:"romanian",count:0,utfRegex:new RegExp(/sunt|eşti/,"gi"),isoRegex:new RegExp(/sunt|eºti/,"gi"),encoding:"CP1250",utfFrequency:sharedFrequency.romanian,isoFrequency:sharedFrequency.romanian},{name:"slovak",count:0,utfRegex:sharedRegex.slovak,isoRegex:sharedRegex.slovak,encoding:"CP1250",utfFrequency:sharedFrequency.slovak,isoFrequency:sharedFrequency.slovak},{name:"slovenian",count:0,utfRegex:sharedRegex.slovenian,isoRegex:sharedRegex.slovenian,encoding:"CP1250",utfFrequency:sharedFrequency.slovenian,isoFrequency:sharedFrequency.slovenian},{name:"albanian",count:0,utfRegex:sharedRegex.albanian,isoRegex:sharedRegex.albanian,encoding:"CP1250",utfFrequency:sharedFrequency.albanian,isoFrequency:sharedFrequency.albanian},{name:"russian",count:0,utfRegex:new RegExp(/что/,"gi"),isoRegex:new RegExp(/÷òî/,"gi"),encoding:"CP1251",utfFrequency:{low:.004965,high:.005341},isoFrequency:{low:.003884,high:.003986}},{name:"ukrainian",count:0,utfRegex:new RegExp(/він|але/,"gi"),isoRegex:new RegExp(/â³í|àëå/,"gi"),encoding:"CP1251",utfFrequency:sharedFrequency.ukrainian,isoFrequency:sharedFrequency.ukrainian},{name:"bulgarian",count:0,utfRegex:new RegExp(/това|какво/,"gi"),isoRegex:new RegExp(/òîâà|äîáðå|êaêâo/,"gi"),encoding:"CP1251",utfFrequency:{low:.005225,high:.005628},isoFrequency:{low:.002767,high:.004951}},{name:"english",count:0,utfRegex:sharedRegex.english,isoRegex:sharedRegex.english,encoding:"CP1252",utfFrequency:sharedFrequency.english,isoFrequency:sharedFrequency.english},{name:"french",count:0,utfRegex:sharedRegex.french,isoRegex:sharedRegex.french,encoding:"CP1252",utfFrequency:sharedFrequency.french,isoFrequency:sharedFrequency.french},{name:"portuguese",count:0,utfRegex:sharedRegex.portuguese,isoRegex:sharedRegex.portuguese,encoding:"CP1252",utfFrequency:sharedFrequency.portuguese,isoFrequency:sharedFrequency.portuguese},{name:"spanish",count:0,utfRegex:sharedRegex.spanish,isoRegex:sharedRegex.spanish,encoding:"CP1252",utfFrequency:sharedFrequency.spanish,isoFrequency:sharedFrequency.spanish},{name:"german",count:0,utfRegex:sharedRegex.german,isoRegex:sharedRegex.german,encoding:"CP1252",utfFrequency:sharedFrequency.german,isoFrequency:sharedFrequency.german},{name:"italian",count:0,utfRegex:sharedRegex.italian,isoRegex:sharedRegex.italian,encoding:"CP1252",utfFrequency:sharedFrequency.italian,isoFrequency:sharedFrequency.italian},{name:"danish",count:0,utfRegex:sharedRegex.danish,isoRegex:sharedRegex.danish,encoding:"CP1252",utfFrequency:sharedFrequency.danish,isoFrequency:sharedFrequency.danish},{name:"norwegian",count:0,utfRegex:sharedRegex.norwegian,isoRegex:sharedRegex.norwegian,encoding:"CP1252",utfFrequency:sharedFrequency.norwegian,isoFrequency:sharedFrequency.norwegian},{name:"swedish",count:0,utfRegex:sharedRegex.swedish,isoRegex:sharedRegex.swedish,encoding:"CP1252",utfFrequency:sharedFrequency.swedish,isoFrequency:sharedFrequency.swedish},{name:"dutch",count:0,utfRegex:sharedRegex.dutch,isoRegex:sharedRegex.dutch,encoding:"CP1252",utfFrequency:sharedFrequency.dutch,isoFrequency:sharedFrequency.dutch},{name:"finnish",count:0,utfRegex:sharedRegex.finnish,isoRegex:sharedRegex.finnish,encoding:"CP1252",utfFrequency:sharedFrequency.finnish,isoFrequency:sharedFrequency.finnish},{name:"serbo-croatian",count:0,utfRegex:sharedRegex["serbo-croatian"],isoRegex:sharedRegex["serbo-croatian"],encoding:"CP1252",utfFrequency:sharedFrequency["serbo-croatian"],isoFrequency:sharedFrequency["serbo-croatian"]},{name:"estonian",count:0,utfRegex:sharedRegex.estonian,isoRegex:sharedRegex.estonian,encoding:"CP1252",utfFrequency:sharedFrequency.estonian,isoFrequency:sharedFrequency.estonian},{name:"icelandic",count:0,utfRegex:sharedRegex.icelandic,isoRegex:sharedRegex.icelandic,encoding:"CP1252",utfFrequency:sharedFrequency.icelandic,isoFrequency:sharedFrequency.icelandic},{name:"malay-indonesian",count:0,utfRegex:sharedRegex["malay-indonesian"],isoRegex:sharedRegex["malay-indonesian"],encoding:"CP1252",utfFrequency:sharedFrequency["malay-indonesian"],isoFrequency:sharedFrequency["malay-indonesian"]},{name:"greek",count:0,utfRegex:new RegExp(/είναι/,"gi"),isoRegex:new RegExp(/åßíáé/,"gi"),encoding:"CP1253",utfFrequency:sharedFrequency.greek,isoFrequency:sharedFrequency.greek},{name:"turkish",count:0,utfRegex:sharedRegex.turkish,isoRegex:sharedRegex.turkish,encoding:"CP1254",utfFrequency:sharedFrequency.turkish,isoFrequency:sharedFrequency.turkish},{name:"hebrew",count:0,utfRegex:new RegExp(/אתה/,"gi"),isoRegex:new RegExp(/àúä/,"gi"),encoding:"CP1255",utfFrequency:sharedFrequency.hebrew,isoFrequency:sharedFrequency.hebrew},{name:"arabic",count:0,utfRegex:new RegExp(/هذا/,"gi"),isoRegex:new RegExp(/åðç/,"gi"),encoding:"CP1256",utfFrequency:{low:.003522,high:.004348},isoFrequency:{low:.003773,high:.005559}},{name:"farsi-persian",count:0,utfRegex:new RegExp(/اون/,"gi"),isoRegex:new RegExp(/çíä/,"gi"),encoding:"CP1256",utfFrequency:{low:.002761,high:.004856},isoFrequency:{low:.00301,high:.006646}},{name:"lithuanian",count:0,utfRegex:sharedRegex.lithuanian,isoRegex:sharedRegex.lithuanian,encoding:"CP1257",utfFrequency:sharedFrequency.lithuanian,isoFrequency:sharedFrequency.lithuanian},{name:"chinese-simplified",count:0,utfRegex:new RegExp(/么/,"gi"),isoRegex:new RegExp(/´ó|¶¯|Å®/,"gi"),encoding:"GB18030",utfFrequency:{low:.009567,high:.011502},isoFrequency:{low:.003137,high:.005009}},{name:"chinese-traditional",count:0,utfRegex:new RegExp(/們/,"gi"),isoRegex:new RegExp(/¦b/,"gi"),encoding:"BIG5",utfFrequency:{low:.012484,high:.014964},isoFrequency:{low:.005063,high:.005822}},{name:"japanese",count:0,utfRegex:new RegExp(/ど/,"gi"),isoRegex:new RegExp(/‚»/,"gi"),encoding:"Shift-JIS",utfFrequency:{low:.004257,high:.006585},isoFrequency:{low:.004286,high:.004653}},{name:"korean",count:0,utfRegex:new RegExp(/도/,"gi"),isoRegex:new RegExp(/àö¾î|å¾ß|¡¼­/,"gi"),encoding:"EUC-KR",utfFrequency:{low:.01091,high:.01367},isoFrequency:{low:.004118,high:.004961}},{name:"thai",count:0,utfRegex:new RegExp(/แฮร์รี่|พอตเตอร์/,"gi"),isoRegex:new RegExp(/áîãìãõè|¾íµàµíãì­/,"gi"),encoding:"TIS-620",utfFrequency:{low:.003194,high:.003468},isoFrequency:{low:.002091,high:.002303}},{name:"bengali",count:0,utfRegex:sharedRegex.bengali,isoRegex:sharedRegex.bengali,utfFrequency:sharedFrequency.bengali,isoFrequency:sharedFrequency.bengali},{name:"hindi",count:0,utfRegex:sharedRegex.hindi,isoRegex:sharedRegex.hindi,utfFrequency:sharedFrequency.hindi,isoFrequency:sharedFrequency.hindi},{name:"urdu",count:0,utfRegex:sharedRegex.urdu,isoRegex:sharedRegex.urdu,utfFrequency:sharedFrequency.urdu,isoFrequency:sharedFrequency.urdu},{name:"vietnamese",count:0,utfRegex:sharedRegex.vietnamese,isoRegex:sharedRegex.vietnamese,utfFrequency:sharedFrequency.vietnamese,isoFrequency:sharedFrequency.vietnamese}]},{}],8:[function(require,module,exports){const checkUTF=require("./components/checkUTF.js"),processContent=require("./components/processContent.js"),checkByteOrderMark=require("./components/checkByteOrderMark.js");module.exports=file=>new Promise((resolve,reject)=>{const fileInfo={encoding:null,language:null,confidence:{encoding:null,language:null}},data={},byteOrderMarkBuffer=new FileReader;byteOrderMarkBuffer.onload=()=>{var byteOrderMark=new Uint8Array(byteOrderMarkBuffer.result).slice(0,4).join(" "),byteOrderMark=checkByteOrderMark(byteOrderMark);if(byteOrderMark){fileInfo.encoding=byteOrderMark,fileInfo.confidence.encoding=1;const byteOrderMarkReader=new FileReader;byteOrderMarkReader.onload=()=>{data.content=byteOrderMarkReader.result,resolve(processContent(data,fileInfo))},byteOrderMarkReader.onerror=err=>{reject(err)},byteOrderMarkReader.readAsText(file,fileInfo.encoding)}else{const utfReader=new FileReader;utfReader.onload=()=>{var utfContent=utfReader.result,utf8=checkUTF(utfContent);if(utf8&&(fileInfo.encoding="UTF-8",fileInfo.confidence.encoding=1),utf8)data.content=utfContent,resolve(processContent(data,fileInfo));else{const isoReader=new FileReader;isoReader.onload=()=>{data.content=isoReader.result,resolve(processContent(data,fileInfo))},isoReader.readAsText(file,"ISO-8859-1")}},utfReader.onerror=err=>{reject(err)},utfReader.readAsText(file,"UTF-8")}},byteOrderMarkBuffer.onerror=err=>{reject(err)},byteOrderMarkBuffer.readAsArrayBuffer(file)})},{"./components/checkByteOrderMark.js":1,"./components/checkUTF.js":2,"./components/processContent.js":3}]},{},[8])(8)});
  14.  
  15. const notification = (function() {
  16. 'use strict';
  17. GM_addStyle(`
  18. #notification {
  19. box-sizing: border-box;
  20. position: fixed;
  21. left: calc(50% - 365.65px / 2);
  22. display: flex;
  23. flex-direction: row;
  24. align-items: center;
  25. justify-content: center;
  26. height: 50px;
  27. background-color: #ff7675;
  28. border-radius: 50px;
  29. padding: 0 0px 0px 20px;
  30. top: -50px;
  31. transition: top .5s ease-out;
  32. z-index: 9999999999;
  33. }
  34. #notification .content {
  35. display: flex;
  36. align-items: center;
  37. justify-content: center;
  38. color: white;
  39. font-size: 25px;
  40. }
  41. #notification .closeBox {
  42. margin: 0 10px;
  43. transform: rotate(90deg);
  44. cursor: pointer;
  45. }
  46. #notification .closeBox .progress {
  47. margin: 0 10px;
  48. cursor: pointer;
  49. }
  50. #notification .closeBox .progress .circle {
  51. stroke-dasharray: 100;
  52. animation: progressOffset 0s linear;
  53. }
  54. @keyframes progressOffset {
  55. from {
  56. stroke-dashoffset: 100;
  57. }
  58. to {
  59. stroke-dashoffset: 0;
  60. }
  61. }
  62. `)
  63. return {
  64. open(info, timeout, autoClose=true) {
  65. let eles = document.querySelectorAll('#notification')
  66. for (let i=0;i<eles.length;i++) {
  67. document.body.removeChild(eles[i])
  68. }
  69. this.box = document.createElement('div')
  70. this.box.setAttribute('id', 'notification')
  71. this.box.innerHTML = `
  72. <div class="content"></div>
  73. <svg class="closeBox" width="40" height="40">
  74. <g class="close" style="stroke: white; stroke-width: 2; stroke-linecap: round;">
  75. <line x1="13" y1="13" x2="27" y2="27"/>
  76. <line x1="13" y1="27" x2="27" y2="13"/>
  77. </g>
  78. <g class="progress" fill="transparent" stroke-width="3">
  79. <circle class="background" cx="20" cy="20" r="16" stroke="rgba(255,255,255,0.15)"/>
  80. <circle class="circle" cx="20" cy="20" r="16" stroke="rgba(255,255,255,1)"/>
  81. </g>
  82. </svg>
  83. `
  84. document.body.appendChild(this.box)
  85. this.box.querySelector('.content').innerHTML = info
  86. let width = getComputedStyle(this.box).width
  87. this.box.style.left = `clac(50%-${width}/2)`
  88. this.box.querySelector('.closeBox .progress .circle').style['animation-duration'] = `${timeout}s`
  89. this.box.style.top = '100px'
  90. this.box.querySelector('.closeBox .progress').addEventListener('click', () => {
  91. console.log('you close...')
  92. this.close()
  93. console.log('you clear...')
  94. })
  95. if (autoClose) {
  96. setTimeout(() => {
  97. console.log('timeout close...')
  98. this.close()
  99. console.log('timeout clear ...')
  100. }, timeout * 1000)
  101. }
  102. },
  103. close() {
  104. this.box.style['transition-duration'] = '.23s'
  105. this.box.style['transition-timing-function'] = 'eaer-out'
  106. this.box.style.top = '-50px'
  107. setTimeout(() => {
  108. try {
  109. document.body.removeChild(this.box)
  110. } catch {
  111. console.log('clear')
  112. }
  113. }, 1000)
  114. }
  115. }
  116. })();
  117.  
  118. (function() {
  119. 'use strict'
  120. // create new XMLHttpRequest
  121. const subtitleParser = {
  122. ass: {
  123. getItems(text) { return text.match(/Dialogue:.+/g) },
  124. getInfo(item) {
  125. let [from, to, content] = /Dialogue: 0,(.+?),(.+?),.*?,.*?,.*?,.*?,.*?,.*?,([^\n]+)/.exec(item).slice(1)
  126. return {
  127. from: toSeconds(from),
  128. to: toSeconds(to),
  129. content: content.replace(/{[\s\S]*?}/g, '').replaceAll('\\N', '<br/>').replaceAll('\\h', " ")
  130. }
  131. },
  132. },
  133. ssa: {
  134. getItems(text) { return text.match(/Dialogue:.+/g) },
  135. getInfo(item) {
  136. let [from, to, content] = /Dialogue: Marked=\d*?,(.+?),(.+?),.*?,.*?,.*?,.*?,.*?,.*?,([^\n]+)/.exec(item).slice(1)
  137. return {
  138. from: toSeconds(from),
  139. to: toSeconds(to),
  140. content: content.replace(/{[\s\S]*?}/g, '').replaceAll('\\N', '<br/>')
  141. }
  142. },
  143. },
  144. srt: {
  145. getItems(text) { return text.split(/(\r\n|\n)\1+/) },
  146. getInfo(item) {
  147. let lineArray = item.match(/(.+)/g)
  148. let [from, to] = lineArray[1].split(' --> ')
  149. return {
  150. from: toSeconds(from),
  151. to: toSeconds(to),
  152. content: lineArray.slice(2).join('<br/>').replaceAll(/{[\s\S]*?}/g, '')
  153. }
  154. },
  155. },
  156. }
  157. let subtitleType
  158. let toSeconds = (timeStr) => {
  159. let timeArr = timeStr.replace(',', '.').split(':')
  160. let timeSec = 0
  161. for (let i = 0; i < timeArr.length; i++) {
  162. timeSec += 60 ** (timeArr.length - i - 1) * parseFloat(timeArr[i])
  163. }
  164. return timeSec
  165. }
  166. // read
  167. function read(blob) {
  168. return new Promise(function(resolve, reject) {
  169. let reader = new FileReader()
  170. reader.onload = function(e) {
  171. let text = reader.result
  172. // 可能错误编码方式
  173. if (text.indexOf("�") !== -1) {
  174. console.log('ERROR in UTF-8')
  175. console.log(`GBK ${fileInfo.name}`)
  176. return reader.readAsText(blob, 'GBK')
  177. }
  178. // 可能错误编码方式
  179. if (text.indexOf("") !== -1) {
  180. console.log('ERROR in UTF-8')
  181. console.log(`ISO-8859-1 ${fileInfo.name}`)
  182. languageEncoding(blob)
  183. .then(info => reader.readAsText(blob, info.encoding))
  184. }
  185. resolve(reader)
  186. }
  187. reader.onerror = reject
  188. reader.readAsText(blob, 'UTF-8')
  189. })
  190. }
  191. // parse subtitle
  192. async function parseTextToArray(fileInfo) {
  193. if (Object.keys(fileInfo).includes("text")) return
  194. console.log('fetch ' + fileInfo.name)
  195. let reader = await fetch(fileInfo.url, {headers: {Referer: 'https://www.aliyundrive.com/'}})
  196. .then(e => e.blob())
  197. .then(blob => {
  198. console.log(`UTF-8 ${fileInfo.name}`)
  199. return read(blob)
  200. })
  201. let text = reader.result
  202. let itemArray = subtitleParser[subtitleType].getItems(text)
  203. console.log(itemArray)
  204. let InfoArray = []
  205. itemArray.forEach((item) => {
  206. try {
  207. let info = subtitleParser[subtitleType].getInfo(item)
  208. InfoArray.push(info)
  209. }
  210. catch {
  211. console.log(`[ERROR] ${item}`)
  212. }
  213. })
  214. return InfoArray
  215. }
  216.  
  217. // add subtitle to video
  218. let addSubtitle = (subtitles) => {
  219. console.log('add subtitle...')
  220. window.startTime = 0
  221. window.endTime = 0
  222. const fontsize = 4.23
  223. // 00:00
  224. let percentNode = document.querySelector("[class^=modal] [class^=progress-bar] [class^=current]")
  225. let totalTimeNode = document.querySelector("[class^=modal] [class^=progress-bar] span:last-child")
  226. // create a subtitle div
  227. const videoStageNode = document.querySelector("[class^=video-stage]")
  228. subtitleNode && subtitleNode.parentNode && subtitleNode.parentNode.removeChild(subtitleNode)
  229. subtitleNode = document.createElement('div')
  230. subtitleNode.setAttribute('id', 'subtitle')
  231. GM_addStyle(`
  232. #subtitle {
  233. position: absolute;
  234. display: flex;
  235. flex-direction: column-reverse;
  236. align-items: flex-end;
  237. color: white;
  238. width: 100%;
  239. height: 100%;
  240. bottom: 4vh;
  241. transition: bottom .2s linear;
  242. z-index: 9;
  243. }
  244. #subtitle .subtitleText {
  245. position: absolute;
  246. display: flex;
  247. align-items: center;
  248. justify-content: center;
  249. text-align: center;
  250. width: 100%;
  251. color: white;
  252. bottom: 0vh;
  253. text-shadow: 0 0 1px var(--basic_black);
  254. font-size: ${fontsize}vh;
  255. visibility: hidden;
  256. }
  257. @keyframes subtitle {
  258. from {
  259. visibility: visible;
  260. }
  261. to {
  262. visibility: visible;
  263. }
  264. }
  265. `)
  266. videoStageNode.appendChild(subtitleNode)
  267. console.log('add subtitleNode')
  268. // 观察变化
  269. let video = document.querySelector('video')
  270. const totalSec = video.duration
  271. console.log(`total time is ${totalSec}s`)
  272. let insertSubtitle = function (mutationsList, observer) {
  273. // 00:00:00 => 秒
  274. let timeSec = video.currentTime
  275. // 保护时间,防止重复
  276. if (timeSec > window.endTime || timeSec < window.startTime) {
  277. // 此时用户可能在拖动进度条,反之拖动后重叠,清空subtitleNode
  278. subtitleNode.innerHTML = ""
  279. } else {
  280. let pTags = subtitleNode.querySelectorAll('[animationend]')
  281. for (let i=0;i<pTags.length;i++) {
  282. subtitleNode.removeChild(pTags[i])
  283. }
  284. }
  285. let existIndex = (index) => {
  286. if (subtitleNode.childNodes.length) {
  287. for (let i=0; i<subtitleNode.childNodes.length; i++) {
  288. if (subtitleNode.childNodes[i].getAttribute('index') == String(index)) {
  289. return true
  290. }
  291. }
  292. }
  293. return false
  294. }
  295. let searchIndex = function (target, arr) {
  296. let indexes = []
  297. for (let index=0;index<arr.length;index++) {
  298. let e = arr[index]
  299. if (target >= e.from && target <= e.to) {
  300. if (!existIndex(index)) {
  301. indexes.push(index)
  302. }
  303. }
  304. }
  305. return indexes
  306. }
  307. searchIndex(timeSec, subtitles).forEach((index) => {
  308. let oneSubtitle = subtitles[index]
  309. let subtitleText = document.createElement('p')
  310. subtitleText.setAttribute('class', 'subtitleText')
  311. subtitleText.setAttribute('index', String(index))
  312. subtitleText.innerHTML = oneSubtitle.content
  313. let duration = oneSubtitle.to - oneSubtitle.from - (timeSec - oneSubtitle.from)
  314. subtitleText.addEventListener('animationend', function() {
  315. subtitleText.setAttribute('animationend', '')
  316. })
  317. // 合适位置插入
  318. if (subtitleNode.childNodes.length) {
  319. // debugger
  320. let bottom = '0px'
  321. let i = 0
  322. while (true) {
  323. if (subtitleNode.childNodes[i]) {
  324. if (parseFloat(bottom.replace('px', '')) < parseFloat(subtitleNode.childNodes[i].style.bottom.replace('px', ''))) {
  325. subtitleText.style.bottom = bottom
  326. subtitleNode.insertBefore(subtitleText, subtitleNode.childNodes[i])
  327. break
  328. } else {
  329. bottom = parseFloat(bottom.replace('px', '')) + parseFloat(getComputedStyle(subtitleNode.childNodes[i]).height.replace('px', '')) + 'px'
  330. i ++
  331. continue
  332. }
  333. } else {
  334. // px -> vh 相对高度,调整窗口自适应
  335. subtitleText.style.bottom = parseFloat(bottom.replace('px', '')) / parseFloat(window.innerHeight) * 100 + 'vh'
  336. subtitleNode.appendChild(subtitleText)
  337. break
  338. }
  339. }
  340. } else {
  341. subtitleNode.appendChild(subtitleText)
  342. }
  343. subtitleText.style.animation = `subtitle ${duration}s linear`
  344. // 记录结束时间
  345. window.startTime = oneSubtitle.from
  346. window.endTime = oneSubtitle.to
  347. return true
  348. })
  349. }
  350. var config = { attributes: true, childList: true, subtree: true }
  351. var observer = new MutationObserver(insertSubtitle)
  352. observer.observe(percentNode, config)
  353. // 暂停播放事件
  354. let playBtnEvent = () => {
  355. subtitleNode.innerHTML = ""
  356. while (true) {
  357. if (!insertSubtitle(null, null)) {
  358. break
  359. }
  360. }
  361. subtitleNode.childNodes.forEach((p) => {
  362. p.style.visibility = 'visible'
  363. })
  364. }
  365. window.addEventListener('keydown', () => {
  366. if (window.event.which == 32 | window.event.which == 39 | window.event.which == 37) {
  367. playBtnEvent()
  368. }
  369. })
  370. document.querySelector('[class^=video-player]').addEventListener('click', () => {
  371. playBtnEvent()
  372. }, false)
  373. return observer
  374. }
  375. // observer root
  376. const rootNode = document.querySelector('#root')
  377. // no root, exist
  378. if (!rootNode) { return }
  379. let obsArray = [], subtitleNode
  380. const callback = function (mutationList, observer) {
  381. // add subtitle
  382. subtitleNode = document.querySelector('#subtitle')
  383. if (subtitleNode) {subtitleNode.parentNode.removeChild(subtitleNode)}
  384. let Node = mutationList[0].addedNodes[0]
  385. if (!Node || !Node.getAttribute('class').includes('modal')) { return }
  386. // clear observer
  387. obsArray.forEach(obs => {
  388. console.log(obs)
  389. console.log('disconnect')
  390. obs.disconnect()
  391. })
  392. obsArray = []
  393. console.log('add a video modal')
  394.  
  395. /////////////////////
  396. // 获取文件夹内文件
  397. let fileInfoList = null
  398. let params = JSON.parse(window.localStorage.token)
  399. let drive_id = params.default_drive_id
  400. let access_token = params.access_token
  401. let parent_file_id = window.location.pathname.split("/").slice(-1)[0]
  402. console.log(drive_id, access_token, parent_file_id)
  403. GM_xmlhttpRequest({
  404. method: "POST",
  405. url: "https://api.aliyundrive.com/adrive/v3/file/list?jsonmask=next_marker,items(name,file_id,drive_id,url,file_extension)",
  406. data: JSON.stringify({
  407. drive_id: drive_id,
  408. parent_file_id: parent_file_id
  409. }),
  410. headers: {
  411. "authorization": `Bearer ${access_token}`,
  412. },
  413. responseType: "json",
  414. onload: function(res) {
  415. console.log(res.response)
  416. fileInfoList = res.response["items"]
  417. }
  418. })
  419. let id = setInterval(
  420. async function() {
  421. let modal = Node
  422. // find title name
  423. if (modal.querySelectorAll('[class^=filename] span').length == 0 || fileInfoList == null) return
  424. // 检查text是否加载完毕
  425. clearInterval(id)
  426. let filename = modal.querySelector('[class^=filename] span').innerText
  427. let title = filename.split('.').slice(0, -1).join('.')
  428. console.log(title)
  429. console.log(fileInfoList)
  430. // search the corresponding ass url
  431. let isInclude = (c1, c2) => {
  432. return c1.match(new RegExp((c2.split('.').slice(0, -1).join('.') || c2).replace('(', '\\(').replace(')', '\\)'), 'i'))
  433. }
  434. // 先过滤出适配的字幕格式文件
  435. fileInfoList = fileInfoList.filter((fileInfo) => {
  436. if (Object.keys(subtitleParser).includes(fileInfo.name.split('.').slice(-1)[0])) {
  437. return true
  438. }
  439. })
  440. let fileInfo = fileInfoList.filter((fileInfo) => {
  441. if (!fileInfo.name.includes('.')) return false
  442. if (fileInfo.name == filename) return false
  443. // 你中有我,或我中有你
  444. let flag = isInclude(fileInfo.name, title) || isInclude(title, fileInfo.name)
  445. // S01E01样式匹配
  446. const reg = /s\d+e\d+/i
  447. let subtitleMatch = fileInfo.name.match(reg)
  448. let videoMatch = title.match(reg)
  449. if (!flag) {
  450. flag = subtitleMatch && videoMatch && subtitleMatch[0].toUpperCase() == videoMatch[0].toUpperCase()
  451. }
  452. // 单音频单字幕文件直接匹配,因为没得选
  453. if (fileInfoList.length == 1) {
  454. flag = true
  455. }
  456. return flag
  457. })
  458. console.log(fileInfo)
  459. // no file, exit
  460. if (!fileInfo.length) {console.log('subtitle exit...'); return}
  461. fileInfo = fileInfo[0]
  462. console.log(fileInfo)
  463. subtitleType = fileInfo.name.split('.').slice(-1)
  464. console.log(`[subtitleType] ${subtitleType}`)
  465. let subtitles = await parseTextToArray(fileInfo)
  466. obsArray.push(addSubtitle(subtitles))
  467. console.log(`${subtitles.length}条字幕添加成功`)
  468. notification.open(`${subtitles.length}条字幕添加成功`, 3)
  469. // 是否变更视频
  470. let obs = new MutationObserver((mutationList, obs) => {
  471. let filenameNode = modal.querySelector('[class^=header] [class^=text]')
  472. if (filenameNode && filenameNode.innerText !== filename) {
  473. setTimeout(() => {
  474. callback([{addedNodes: [modal]}], null)
  475. }, 0)
  476. }
  477. })
  478. obs.observe(modal, {subtree: true, childList: true})
  479. obsArray.push(obs)
  480. // 是否触发控制条
  481. let playerTool = document.querySelector('[class^=video-player]')
  482. let offsetSubtitle = (mutationList, obs) => {
  483. // let subtitleNode = document.querySelector('#subtitle')
  484. if (subtitleNode && mutationList[0].attributeName == 'class') {
  485. if (mutationList[0].target.classList.length == 2 && document.fullscreenElement) {
  486. subtitleNode.style['bottom'] = '13vh'
  487. } else {
  488. subtitleNode.style['bottom'] = '4vh'
  489. }
  490. }
  491. }
  492. obs = new MutationObserver(offsetSubtitle)
  493. obs.observe(playerTool, {attributes: true, childList: true})
  494. offsetSubtitle([{attributeName: 'class', target: playerTool}])
  495. obsArray.push(obs)
  496. document.onfullscreenchange = () => {
  497. offsetSubtitle([{attributeName: 'class', target: playerTool}], obs)
  498. }
  499. }, 10)
  500. }
  501. const observer = new MutationObserver(callback)
  502. observer.observe(rootNode, {childList: true})
  503. })();