SaveTube

Download videos from video sharing web sites.

当前为 2021-04-09 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name SaveTube
  3. // @version 2021.03.23
  4. // @description Download videos from video sharing web sites.
  5. // @author sebaro
  6. // @namespace http://sebaro.pro/savetube
  7. // @icon https://gitlab.com/sebaro/savetube/raw/master/savetube.png
  8. // @include http://youtube.com*
  9. // @include http://www.youtube.com*
  10. // @include https://youtube.com*
  11. // @include https://www.youtube.com*
  12. // @include http://m.youtube.com*
  13. // @include https://m.youtube.com*
  14. // @include http://dailymotion.com*
  15. // @include http://www.dailymotion.com*
  16. // @include https://dailymotion.com*
  17. // @include https://www.dailymotion.com*
  18. // @include http://vimeo.com*
  19. // @include http://www.vimeo.com*
  20. // @include https://vimeo.com*
  21. // @include https://www.vimeo.com*
  22. // @include http://metacafe.com*
  23. // @include http://www.metacafe.com*
  24. // @include https://metacafe.com*
  25. // @include https://www.metacafe.com*
  26. // @include http://veoh.com*
  27. // @include http://www.veoh.com*
  28. // @include https://veoh.com*
  29. // @include https://www.veoh.com*
  30. // @include http://viki.com*
  31. // @include http://www.viki.com*
  32. // @include https://viki.com*
  33. // @include https://www.viki.com*
  34. // @include http://imdb.com*
  35. // @include http://www.imdb.com*
  36. // @include https://imdb.com*
  37. // @include https://www.imdb.com*
  38. // @noframes
  39. // @grant none
  40. // @run-at document-end
  41. // ==/UserScript==
  42.  
  43.  
  44. /*
  45.  
  46. Copyright (C) 2010 - 2021 Sebastian Luncan
  47.  
  48. This program is free software: you can redistribute it and/or modify
  49. it under the terms of the GNU General Public License as published by
  50. the Free Software Foundation, either version 3 of the License, or
  51. (at your option) any later version.
  52.  
  53. This program is distributed in the hope that it will be useful,
  54. but WITHOUT ANY WARRANTY; without even the implied warranty of
  55. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  56. GNU General Public License for more details.
  57.  
  58. You should have received a copy of the GNU General Public License
  59. along with this program. If not, see <http://www.gnu.org/licenses/>.
  60.  
  61. Website: http://sebaro.pro/savetube
  62. Contact: http://sebaro.pro/contact
  63.  
  64. */
  65.  
  66.  
  67. (function() {
  68.  
  69.  
  70. // Don't run on frames or iframes
  71. if (window.top != window.self) return;
  72.  
  73.  
  74. // ==========Variables========== //
  75.  
  76. // Userscript
  77. var userscript = 'SaveTube';
  78. var website = 'http://sebaro.pro/savetube';
  79. var contact = 'http://sebaro.pro/contact';
  80.  
  81. // Page
  82. var page = {win: window, doc: window.document, body: window.document.body, url: window.location.href, site: window.location.hostname.match(/([^.]+)\.[^.]+$/)[1]};
  83.  
  84. // Saver
  85. var saver = {};
  86. var panelHeight = 30;
  87.  
  88. // Features/Options
  89. var feature = {'definition': true, 'container': true, 'openpagelink': true, 'autosave': true, 'savedash': false, 'showsavelink': true};
  90. var option = {'definition': 'High Definition', 'container': 'MP4', 'openpagelink': false, 'autosave': false, 'savedash': false, 'showsavelink': false, 'hidden': false};
  91.  
  92. // Media
  93. var mediatypes = {'MP4': 'video/mp4', 'WebM': 'video/webm', 'M3U8': 'application/x-mpegURL', 'WebVTT': 'text/vtt'};
  94.  
  95. // Sources
  96. var sources = {};
  97.  
  98.  
  99. // ==========Functions========== //
  100.  
  101. function createMyElement(type, properties, event, listener) {
  102. var obj = page.doc.createElement(type);
  103. for (var propertykey in properties) {
  104. if (propertykey == 'target') obj.setAttribute('target', properties[propertykey]);
  105. else obj[propertykey] = properties[propertykey];
  106. }
  107. if (event && listener) {
  108. obj.addEventListener(event, listener, false);
  109. }
  110. return obj;
  111. }
  112.  
  113. function modifyMyElement(obj, properties, event, listener) {
  114. for (var propertykey in properties) {
  115. if (propertykey == 'target') obj.setAttribute('target', properties[propertykey]);
  116. else obj[propertykey] = properties[propertykey];
  117. }
  118. if (event && listener) {
  119. obj.addEventListener(event, listener, false);
  120. }
  121. }
  122.  
  123. function styleMyElement(obj, styles) {
  124. for (var stylekey in styles) {
  125. obj.style[stylekey] = styles[stylekey];
  126. }
  127. }
  128.  
  129. function cleanMyElement(obj, hide) {
  130. if (hide) {
  131. for (var i = 0; i < obj.children.length; i++) {
  132. styleMyElement(obj.children[i], {display: 'none'});
  133. }
  134. }
  135. else {
  136. if (obj.hasChildNodes()) {
  137. while (obj.childNodes.length >= 1) {
  138. obj.removeChild(obj.firstChild);
  139. }
  140. }
  141. }
  142. }
  143.  
  144. function getMyElement(obj, type, from, value, child, content) {
  145. var getObj, chObj, coObj;
  146. var pObj = (!obj) ? page.doc : obj;
  147. if (type == 'body') getObj = pObj.body;
  148. else {
  149. if (from == 'id') getObj = pObj.getElementById(value);
  150. else if (from == 'class') getObj = pObj.getElementsByClassName(value);
  151. else if (from == 'tag') getObj = pObj.getElementsByTagName(type);
  152. else if (from == 'ns') {
  153. if (pObj.getElementsByTagNameNS) getObj = pObj.getElementsByTagNameNS(value, type);
  154. }
  155. else if (from == 'query') {
  156. if (child > 0) {
  157. if (pObj.querySelectorAll) getObj = pObj.querySelectorAll(value);
  158. }
  159. else {
  160. if (pObj.querySelector) getObj = pObj.querySelector(value);
  161. }
  162. }
  163. }
  164. chObj = (getObj && child >= 0) ? getObj[child] : getObj;
  165. if (content && chObj) {
  166. if (type == 'html' || type == 'body' || type == 'div' || type == 'option') coObj = chObj.innerHTML;
  167. else if (type == 'object') coObj = chObj.data;
  168. else if (type == 'img' || type == 'video' || type == 'embed') coObj = chObj.src;
  169. else coObj = chObj.textContent;
  170. return coObj;
  171. }
  172. else {
  173. return chObj;
  174. }
  175. }
  176.  
  177. function appendMyElement(parent, child) {
  178. parent.appendChild(child);
  179. }
  180.  
  181. function removeMyElement(parent, child) {
  182. parent.removeChild(child);
  183. }
  184.  
  185. function replaceMyElement(parent, orphan, child) {
  186. parent.replaceChild(orphan, child);
  187. }
  188.  
  189. function cleanMyContent(content, unesc, extra) {
  190. if (unesc) content = unescape(content);
  191. content = content.replace(/\\u0025/g, '%');
  192. content = content.replace(/\\u0026/g, '&');
  193. content = content.replace(/\\u002F/g, '/');
  194. content = content.replace(/\\/g, '');
  195. content = content.replace(/\n/g, '');
  196. if (extra) {
  197. content = content.replace(/&quot;/g, '\'').replace(/&#34;/g, '\'').replace(/&#034;/g, '\'').replace(/"/g, '\'');
  198. content = content.replace(/&#39;/g, '\'').replace(/&#039;/g, '\'').replace(/'/g, '\'');
  199. content = content.replace(/&amp;/g, 'and').replace(/&/g, 'and');
  200. content = content.replace(/[\/\|]/g, '-');
  201. content = content.replace(/[#:\*\?]/g, '');
  202. content = content.replace(/^\s+|\s+$/, '').replace(/\.+$/g, '');
  203. }
  204. return content;
  205. }
  206.  
  207. function getMyContent(url, pattern, clean) {
  208. var myPageContent, myVideosParse, myVideosContent;
  209. if (!sources[url]) {
  210. var XHRequest = new XMLHttpRequest();
  211. XHRequest.open('GET', url, false);
  212. XHRequest.send();
  213. sources[url] = (XHRequest.responseText) ? XHRequest.responseText : XHRequest.responseXML;
  214. //console.log('Request: ' + url + ' ' + pattern);
  215. //console.log(sources[url]);
  216. }
  217. if (pattern == 'TEXT') {
  218. myVideosContent = sources[url];
  219. }
  220. else {
  221. myPageContent = (sources[url]) ? sources[url] : '';
  222. if (clean) myPageContent = cleanMyContent(myPageContent, true);
  223. myVideosParse = myPageContent.match(pattern);
  224. myVideosContent = (myVideosParse) ? myVideosParse[1] : null;
  225. }
  226. return myVideosContent;
  227. }
  228.  
  229. function createMySaver() {
  230. /* The Panel */
  231. saver['saverPanel'] = createMyElement('div');
  232. styleMyElement(saver['saverPanel'], {position: 'fixed', fontFamily: 'sans-serif', fontSize: '10px', minHeight: panelHeight + 'px', lineHeight: panelHeight + 'px', backgroundColor: '#FFFFFF', padding: '0px 10px 5px 10px', bottom: '0px', right: '25px', zIndex: '2000000000', borderTop: '1px solid #CCCCCC', borderLeft: '1px solid #CCCCCC', borderRight: '1px solid #CCCCCC', borderRadius: '5px 5px 0px 0px', textAlign: 'center', boxSizing: 'content-box'});
  233. appendMyElement(page.body, saver['saverPanel']);
  234.  
  235. /* Panel Hide/Show Toggle Button */
  236. saver['buttonHide'] = createMyElement('div', {title: '{Hide/Show: click to hide/show this panel}'}, 'click', function() {
  237. toggleMySaver();
  238. });
  239. styleMyElement(saver['buttonHide'], {width: '0px', height: '0px', display: 'inline-block', borderTop: '8px solid transparent', borderBottom: '8px solid transparent', borderLeft: '15px solid #777777', borderRight: '0px solid #777777', lineHeight: 'normal', verticalAlign: 'middle', marginLeft: '5px', marginRight: '10px', cursor: 'pointer', boxSizing: 'content-box'});
  240. appendMyElement(saver['saverPanel'], saver['buttonHide']);
  241.  
  242. /* Panel Logo */
  243. saver['panelLogo'] = createMyElement('div', {title: '{SaveTube: click to visit the script wesite}', textContent: userscript}, 'click', function() {
  244. page.win.location.href = website;
  245. });
  246. styleMyElement(saver['panelLogo'], {display: 'inline-block', color: '#32D132', fontSize: '14px', fontWeight: 'bold', border: '1px solid #32D132', borderRadius: '3px', padding: '0px 4px', marginRight: '10px', lineHeight: 'normal', verticalAlign: 'middle', cursor: 'pointer', boxSizing: 'content-box'});
  247. appendMyElement(saver['saverPanel'], saver['panelLogo']);
  248.  
  249. /* Warnings */
  250. if (saver['warnMess']) {
  251. saver['saverMessage'] = createMyElement('div');
  252. styleMyElement(saver['saverMessage'], {display: 'inline-block', fontSize: '12px', color: '#AD0000'});
  253. appendMyElement(saver['saverPanel'], saver['saverMessage']);
  254. if (saver['warnContent']) showMyMessage(saver['warnMess'], saver['warnContent']);
  255. else showMyMessage(saver['warnMess']);
  256. return;
  257. }
  258.  
  259. /* Panel Video Menu */
  260. saver['videoMenu'] = createMyElement('select', {title: '{Videos: select the video format for download}'}, 'change', function() {
  261. saver['videoSave'] = this.value;
  262. if (saver['isShowingLink']) {
  263. cleanMyElement(saver['buttonSaveLink'], false);
  264. saver['isShowingLink'] = false;
  265. }
  266. if (option['autosave']) {
  267. saveMyVideo();
  268. }
  269. });
  270. styleMyElement(saver['videoMenu'], {display: 'inline-block', width: 'auto', height: '20px', fontFamily: 'inherit', fontSize: '14px', fontWeight: 'bold', padding: '0px 3px', overflow: 'hidden', border: '1px solid #CCCCCC', color: '#777777', backgroundColor: '#FFFFFF', lineHeight: 'normal', verticalAlign: 'middle', cursor: 'pointer', boxSizing: 'content-box'});
  271. appendMyElement(saver['saverPanel'], saver['videoMenu']);
  272. if (feature['openpagelink']) {
  273. saver['videoList']['Page Link'] = page.url;
  274. }
  275. var videosProgressive = [];
  276. var videosAdaptiveHLS = [];
  277. var videosAdaptiveDASHVideo = [];
  278. var videosAdaptiveDASHAudio = [];
  279. var videosAdaptiveDASHMuxed = [];
  280. var videosExtra = [];
  281. for (var videoCode in saver['videoList']) {
  282. if (videoCode.indexOf('Video') != -1) videosAdaptiveDASHVideo.push(videoCode);
  283. else if (videoCode.indexOf('Audio') != -1) videosAdaptiveDASHAudio.push(videoCode);
  284. else if (saver['videoList'][videoCode] == 'DASH') videosAdaptiveDASHMuxed.push(videoCode);
  285. else if (videoCode.indexOf('M3U8') != -1) videosAdaptiveHLS.push(videoCode);
  286. else if (videoCode.indexOf('MP4') != -1 || videoCode.indexOf('WebM') != -1) videosProgressive.push(videoCode);
  287. else videosExtra.push(videoCode);
  288. }
  289. if (videosProgressive.length > 0) {
  290. for (var i = 0; i < videosProgressive.length; i++) {
  291. saver['videoItem'] = createMyElement('option', {value: videosProgressive[i], textContent: videosProgressive[i]});
  292. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
  293. appendMyElement(saver['videoMenu'], saver['videoItem']);
  294. }
  295. }
  296. if (videosAdaptiveHLS.length > 0) {
  297. saver['videoItem'] = createMyElement('option', {value: 'HLS', textContent: 'HLS'});
  298. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
  299. saver['videoItem'].disabled = 'disabled';
  300. appendMyElement(saver['videoMenu'], saver['videoItem']);
  301. for (var i = 0; i < videosAdaptiveHLS.length; i++) {
  302. saver['videoItem'] = createMyElement('option', {value: videosAdaptiveHLS[i], textContent: videosAdaptiveHLS[i]});
  303. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
  304. appendMyElement(saver['videoMenu'], saver['videoItem']);
  305. }
  306. }
  307. if (videosAdaptiveDASHVideo.length > 0) {
  308. saver['videoItem'] = createMyElement('option', {value: 'DASH (Video Only)', textContent: 'DASH (Video Only)'});
  309. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
  310. saver['videoItem'].disabled = 'disabled';
  311. appendMyElement(saver['videoMenu'], saver['videoItem']);
  312. for (var i = 0; i < videosAdaptiveDASHVideo.length; i++) {
  313. saver['videoItem'] = createMyElement('option', {value: videosAdaptiveDASHVideo[i], textContent: videosAdaptiveDASHVideo[i]});
  314. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
  315. appendMyElement(saver['videoMenu'], saver['videoItem']);
  316. }
  317. }
  318. if (videosAdaptiveDASHAudio.length > 0) {
  319. saver['videoItem'] = createMyElement('option', {value: 'DASH (Audio Only)', textContent: 'DASH (Audio Only)'});
  320. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
  321. saver['videoItem'].disabled = 'disabled';
  322. appendMyElement(saver['videoMenu'], saver['videoItem']);
  323. for (var i = 0; i < videosAdaptiveDASHAudio.length; i++) {
  324. saver['videoItem'] = createMyElement('option', {value: videosAdaptiveDASHAudio[i], textContent: videosAdaptiveDASHAudio[i]});
  325. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
  326. appendMyElement(saver['videoMenu'], saver['videoItem']);
  327. }
  328. }
  329. if (videosAdaptiveDASHMuxed.length > 0) {
  330. feature['savedash'] = true;
  331. if (option['savedash']) {
  332. saver['videoItem'] = createMyElement('option', {value: 'DASH (Video With Audio)', textContent: 'DASH (Video With Audio)'});
  333. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
  334. saver['videoItem'].disabled = 'disabled';
  335. appendMyElement(saver['videoMenu'], saver['videoItem']);
  336. for (var i = 0; i < videosAdaptiveDASHMuxed.length; i++) {
  337. saver['videoItem'] = createMyElement('option', {value: videosAdaptiveDASHMuxed[i], textContent: videosAdaptiveDASHMuxed[i]});
  338. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
  339. appendMyElement(saver['videoMenu'], saver['videoItem']);
  340. }
  341. }
  342. else {
  343. for (var videoCode in saver['videoList']) {
  344. if (saver['videoList'][videoCode] == 'DASH') delete saver['videoList'][videoCode];
  345. }
  346. }
  347. }
  348. if (videosExtra.length > 0) {
  349. saver['videoItem'] = createMyElement('option', {value: 'Extra', textContent: 'Extra'});
  350. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
  351. saver['videoItem'].disabled = 'disabled';
  352. appendMyElement(saver['videoMenu'], saver['videoItem']);
  353. for (var i = 0; i < videosExtra.length; i++) {
  354. saver['videoItem'] = createMyElement('option', {value: videosExtra[i], textContent: videosExtra[i]});
  355. styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
  356. appendMyElement(saver['videoMenu'], saver['videoItem']);
  357. }
  358. }
  359.  
  360. /* Panel Options Button */
  361. saver['buttonOptions'] = createMyElement('div', {title: '{Options: click to show the available options}'}, 'click', function() {
  362. if (saver['showsOptions']) {
  363. saver['showsOptions'] = false;
  364. styleMyElement(saver['optionsContent'], {display: 'none'})
  365. }
  366. else {
  367. saver['showsOptions'] = true;
  368. styleMyElement(saver['optionsContent'], {display: 'block'})
  369. }
  370. });
  371. styleMyElement(saver['buttonOptions'], {width: '1px', height: '14px', display: 'inline-block', paddingTop: '3px', borderLeft: '3px dotted #777777', lineHeight: 'normal', verticalAlign: 'middle', marginLeft: '20px', cursor: 'pointer', boxSizing: 'content-box'});
  372. appendMyElement(saver['saverPanel'], saver['buttonOptions']);
  373.  
  374. /* Panel Save Button */
  375. saver['buttonSave'] = createMyElement('div', {title: '{Save: click to download the selected video format}'}, 'click', function() {
  376. saveMyVideo();
  377. });
  378. styleMyElement(saver['buttonSave'], {width: '0px', height: '0px', display: 'inline-block', borderLeft: '8px solid transparent', borderRight: '8px solid transparent', borderTop: '15px solid #777777', borderBottom: '0px solid #777777', lineHeight: 'normal', verticalAlign: 'middle', marginTop: '2px', marginLeft: '20px', cursor: 'pointer', boxSizing: 'content-box'});
  379. appendMyElement(saver['saverPanel'], saver['buttonSave']);
  380.  
  381. /* Panel Save Button Link */
  382. saver['buttonSaveLink'] = createMyElement('div', {title: '{Save: right click & save as to download the selected video format}'});
  383. styleMyElement(saver['buttonSaveLink'], {display: 'inline-block', color: '#777777', fontSize: '14px', fontWeight: 'bold', lineHeight: 'normal', verticalAlign: 'middle', marginLeft: '5px', marginBottom: '2px', boxSizing: 'content-box'});
  384. appendMyElement(saver['saverPanel'], saver['buttonSaveLink']);
  385.  
  386. /* Disable Features */
  387. if (saver['videoDefinitions'].length < 2) feature['definition'] = false;
  388. if (saver['videoContainers'].length < 2) feature['container'] = false;
  389.  
  390. /* Select The Video */
  391. if (feature['definition'] || feature['container'] || feature['openpagelink']) {
  392. if (!option['definition'] || saver['videoDefinitions'].indexOf(option['definition']) == -1) option['definition'] = saver['videoSave'].replace(/Definition.*/, 'Definition');
  393. if (!option['container'] || saver['videoContainers'].indexOf(option['container']) == -1) option['container'] = saver['videoSave'].replace(/.*\s/, '');
  394. selectMyVideo();
  395. }
  396.  
  397. /* Save The Video On Autosave */
  398. if (option['autosave']) saveMyVideo();
  399.  
  400. /* Panel Options */
  401. saver['optionsContent'] = createMyElement('div');
  402. styleMyElement(saver['optionsContent'], {display: 'none', fontSize: '14px', fontWeight: 'bold', padding: '10px', textAlign: 'center', boxSizing: 'content-box'});
  403. appendMyElement(saver['saverPanel'], saver['optionsContent']);
  404.  
  405. /* Options Object => option: [label, options, new line, change video] */
  406. var options = {
  407. 'definition': ['Definition', saver['videoDefinitions'], true, true],
  408. 'container': ['Container', saver['videoContainers'], false, true],
  409. 'openpagelink': ['Open Page Link', ['On', 'Off'], true, true],
  410. 'autosave': ['Autosave', ['On', 'Off'], true, true],
  411. 'showsavelink': ['Show Save Link', ['On', 'Off'], false, true],
  412. 'savedash': ['Save DASH (Video With Audio)', ['On', 'Off'], true, false]
  413. };
  414.  
  415. /* Options */
  416. var optionsBox, optionBox, optionLabel, optionMenu, optionMenuItem;
  417. for (var o in options) {
  418. if (feature[o] === false) continue;
  419. if (options[o][2]) {
  420. optionsBox = createMyElement('div');
  421. styleMyElement(optionsBox, {display: 'block', padding: '5px 0px 5px 0px'});
  422. appendMyElement(saver['optionsContent'], optionsBox);
  423. }
  424. optionBox = createMyElement('div');
  425. styleMyElement(optionBox, {display: 'inline-block'});
  426. optionLabel = createMyElement('div', {textContent: options[o][0]});
  427. styleMyElement(optionLabel, {display: 'inline-block', color: '#777777', marginRight: '10px', verticalAlign: 'middle'});
  428. optionMenu = createMyElement('select', {id: 'savetube-option-' + o}, 'change', function() {
  429. var id = this.id.replace('savetube-option-', '');
  430. if (this.value == 'On' || this.value == 'Off') {
  431. option[id] = (this.value == 'On') ? true : false;
  432. }
  433. else {
  434. option[id] = this.value;
  435. }
  436. setMyOptions(id, option[id]);
  437. if (options[id][3]) {
  438. if (saver['isShowingLink']) {
  439. cleanMyElement(saver['buttonSaveLink'], false);
  440. saver['isShowingLink'] = false;
  441. }
  442. selectMyVideo();
  443. if (option['autosave']) {
  444. saveMyVideo();
  445. }
  446. }
  447. });
  448. styleMyElement(optionMenu, {display: 'inline-block', width: 'auto', height: '20px', color: '#777777', backgroundColor: '#FFFFFF', border: '1px solid #CCCCCC', fontFamily: 'inherit', fontSize: '14px', fontWeight: 'bold', marginRight: '10px', verticalAlign: 'middle'});
  449. appendMyElement(optionBox, optionLabel);
  450. appendMyElement(optionBox, optionMenu);
  451. appendMyElement(optionsBox, optionBox);
  452. for (var i = 0; i < options[o][1].length; i++) {
  453. optionMenuItem = createMyElement('option', {value: options[o][1][i], textContent: options[o][1][i]});
  454. styleMyElement(optionMenuItem, {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
  455. appendMyElement(optionMenu, optionMenuItem);
  456. }
  457. if (optionMenu.value == 'On' || optionMenu.value == 'Off') {
  458. if (option[o]) optionMenu.value = 'On';
  459. else optionMenu.value = 'Off';
  460. }
  461. else {
  462. optionMenu.value = option[o];
  463. }
  464. }
  465.  
  466. /* Hide */
  467. if (option['hidden']) {
  468. toggleMySaver('hide');
  469. }
  470. }
  471.  
  472. function toggleMySaver(toggle) {
  473. if (toggle == 'hide') {
  474. styleMyElement(saver['saverPanel'], {right: '-' + (saver['saverPanel'].offsetWidth - 40) + 'px', backgroundColor: 'transparent', borderColor: '#777777'});
  475. styleMyElement(saver['buttonHide'], {borderTop: '8px solid transparent', borderBottom: '8px solid transparent', borderLeft: '0px solid #777777', borderRight: '15px solid #777777'});
  476. }
  477. else {
  478. if (option['hidden']) {
  479. styleMyElement(saver['saverPanel'], {right: '25px', backgroundColor: '#FFFFFF', borderColor: '#CCCCCC', transition: 'right 2s, background-color 5s, border-color 5s'});
  480. styleMyElement(saver['buttonHide'], {borderTop: '8px solid transparent', borderBottom: '8px solid transparent', borderLeft: '15px solid #777777', borderRight: '0px solid #777777'});
  481. option['hidden'] = false;
  482. }
  483. else {
  484. styleMyElement(saver['saverPanel'], {right: '-' + (saver['saverPanel'].offsetWidth - 40) + 'px', backgroundColor: 'transparent', borderColor: '#777777', transition: 'right 2s, background-color 5s, border-color 5s'});
  485. styleMyElement(saver['buttonHide'], {borderTop: '8px solid transparent', borderBottom: '8px solid transparent', borderLeft: '0px solid #777777', borderRight: '15px solid #777777'});
  486. option['hidden'] = true;
  487. }
  488. setMyOptions('hidden', option['hidden']);
  489. }
  490. }
  491.  
  492. function setMyOptions(key, value) {
  493. key = page.site + '_' + userscript.toLowerCase() + '_' + key;
  494. try {
  495. localStorage.setItem(key, value);
  496. if (localStorage.getItem(key) == value) return;
  497. else throw false;
  498. }
  499. catch(e) {
  500. var date = new Date();
  501. date.setTime(date.getTime() + (356*24*60*60*1000));
  502. var expires = '; expires=' + date.toGMTString();
  503. page.doc.cookie = key + '=' + value + expires + '; path=/';
  504. }
  505. }
  506.  
  507. function getMyOptions() {
  508. for (var opt in option) {
  509. var key = page.site + '_' + userscript.toLowerCase() + '_' + opt;
  510. try {
  511. if (localStorage.getItem(key)) {
  512. option[opt] = localStorage.getItem(key);
  513. continue;
  514. }
  515. else throw false;
  516. }
  517. catch(e) {
  518. var cookies = page.doc.cookie.split(';');
  519. for (var i=0; i < cookies.length; i++) {
  520. var cookie = cookies[i];
  521. while (cookie.charAt(0) == ' ') cookie = cookie.substring(1, cookie.length);
  522. option[opt] = (cookie.indexOf(key) == 0) ? cookie.substring(key.length + 1, cookie.length) : option[opt];
  523. }
  524. }
  525. }
  526. var boolOptions = ['openpagelink', 'autosave', 'showsavelink', 'savedash', 'hidden'];
  527. for (var i = 0; i < boolOptions.length; i++) {
  528. option[boolOptions[i]] = (option[boolOptions[i]] === true || option[boolOptions[i]] == 'true') ? true : false;
  529. }
  530. }
  531.  
  532. function selectMyVideo() {
  533. if (option['openpagelink']) {
  534. saver['videoSave'] = 'Page Link';
  535. }
  536. else {
  537. var vdoCont = (option['container'] != 'Any') ? [option['container']] : saver['videoContainers'];
  538. var vdoDef = saver['videoDefinitions'];
  539. var vdoList = {};
  540. for (var vC = 0; vC < vdoCont.length; vC++) {
  541. if (vdoCont[vC] != 'Any') {
  542. for (var vD = 0; vD < vdoDef.length; vD++) {
  543. var format = vdoDef[vD] + ' ' + vdoCont[vC];
  544. if (!vdoList[vdoDef[vD]]) {
  545. for (var vL in saver['videoList']) {
  546. if (vL == format) {
  547. vdoList[vdoDef[vD]] = vL;
  548. break;
  549. }
  550. }
  551. }
  552. }
  553. }
  554. }
  555. var vdoDef2 = [];
  556. var keepDef = false;
  557. for (var vD = 0; vD < vdoDef.length; vD++) {
  558. if (vdoDef[vD] == option['definition'] && keepDef == false) keepDef = true;
  559. if (keepDef == true) vdoDef2.push(vdoDef[vD])
  560. }
  561. for (var vD = 0; vD < vdoDef2.length; vD++) {
  562. if (vdoList[vdoDef2[vD]]) {
  563. saver['videoSave'] = vdoList[vdoDef2[vD]];
  564. break;
  565. }
  566. }
  567. }
  568. saver['videoMenu'].value = saver['videoSave'];
  569. }
  570.  
  571. function saveMyVideo() {
  572. var vdoURL = saver['videoList'][saver['videoSave']];
  573. var vdoDef = ' (' + saver['videoSave'].split(' ').slice(0, -1).join('').match(/[A-Z]/g).join('') + ')';
  574. var vdoExt = saver['videoSave'].split(' ').slice(-1).join('');
  575. var vdoTle = (saver['videoTitle']) ? saver['videoTitle'] : page.url.replace(/https?:\/\//, '').replace(/[^0-9a-zA-Z]/g, '-');
  576. if (saver['videoSave'] == 'Page Link' || vdoURL == 'DASH' || (vdoExt == 'M3U8' && !option['showsavelink'])) {
  577. var vdoV, vdoA;
  578. if (saver['videoSave'] == 'Page Link' || vdoExt == 'M3U8') {
  579. vdoV = vdoURL;
  580. vdoA = '';
  581. vdoDef = '';
  582. }
  583. else {
  584. if (saver['videoSave'].indexOf('MP4') != -1) {
  585. vdoV = saver['videoList'][saver['videoSave'].replace('MP4', 'Video MP4')];
  586. vdoA = saver['videoList']['Medium Bitrate Audio MP4'] || saver['videoList'][saver['videoSave'].replace('MP4', 'Audio MP4')];
  587. }
  588. else {
  589. vdoV = saver['videoList'][saver['videoSave'].replace('WebM', 'Video WebM')];
  590. vdoA = saver['videoList']['High Bitrate Audio WebM'] || saver['videoList']['Medium Bitrate Audio WebM'] || saver['videoList']['Medium Bitrate Audio MP4'];
  591. }
  592. }
  593. var vdoT = vdoTle + vdoDef;
  594. page.win.location.href = 'savetube:' + vdoT + 'SEPARATOR' + vdoV + 'SEPARATOR' + vdoA;
  595. }
  596. else {
  597. var vdoLnk = createMyElement('a', {href: vdoURL, target: '_blank', textContent: '[Link]'});
  598. styleMyElement(vdoLnk, {color: '#777777', textDecoration: 'underline'});
  599. var vdoT = vdoTle + vdoDef;
  600. if (option['showsavelink'] || vdoExt == 'M3U8') {
  601. appendMyElement(saver['buttonSaveLink'], vdoLnk);
  602. saver['isShowingLink'] = true;
  603. if (page.site == 'youtube' && saver['videoSave'] == 'High Definition MP4') {
  604. if (!page.win.URL || !page.win.URL.createObjectURL) {
  605. page.win.location.href = vdoURL + '&title=' + vdoT;
  606. }
  607. }
  608. }
  609. else {
  610. if (!saver['isSaving']) {
  611. if (page.site == 'youtube' && saver['videoSave'] == 'High Definition MP4') {
  612. page.win.location.href = vdoURL + '&title=' + vdoT;
  613. }
  614. else {
  615. if (page.win.URL && page.win.URL.createObjectURL) {
  616. saver['isSaving'] = true;
  617. styleMyElement(saver['buttonSave'], {borderBottomWidth: '1px', cursor: 'none'});
  618. var vdoLnkBlob, vdoBlob, vdoBlobLnk;
  619. vdoLnkBlob = createMyElement('a');
  620. styleMyElement(vdoLnkBlob, {display: 'none'});
  621. appendMyElement(page.body, vdoLnkBlob);
  622. var XHRequest = new XMLHttpRequest();
  623. XHRequest.open('GET', vdoURL);
  624. XHRequest.responseType = 'arraybuffer';
  625. XHRequest.onload = function() {
  626. if (this.status === 200 && this.response) {
  627. vdoBlob = new Blob([this.response], {type: mediatypes[vdoExt]});
  628. vdoBlobLnk = page.win.URL.createObjectURL(vdoBlob);
  629. modifyMyElement(vdoLnkBlob, {href: vdoBlobLnk, target: '_blank', download: vdoT + '.' + vdoExt.toLowerCase()});
  630. vdoLnkBlob.click();
  631. page.win.URL.revokeObjectURL(vdoBlobLnk);
  632. removeMyElement(page.body, vdoLnkBlob);
  633. saver['isSaving'] = false;
  634. styleMyElement(saver['buttonSave'], {borderBottomWidth: '0px', cursor: 'pointer'});
  635. }
  636. else {
  637. saver['isSaving'] = false;
  638. styleMyElement(saver['buttonSave'], {borderBottomWidth: '0px', cursor: 'pointer'});
  639. if (!saver['isShowingLink']) {
  640. appendMyElement(saver['buttonSaveLink'], vdoLnk);
  641. saver['isShowingLink'] = true;
  642. }
  643. }
  644. }
  645. XHRequest.onerror = function() {
  646. saver['isSaving'] = false;
  647. styleMyElement(saver['buttonSave'], {borderBottomWidth: '0px', cursor: 'pointer'});
  648. if (!saver['isShowingLink']) {
  649. appendMyElement(saver['buttonSaveLink'], vdoLnk);
  650. saver['isShowingLink'] = true;
  651. }
  652. }
  653. XHRequest.send();
  654. }
  655. else {
  656. appendMyElement(saver['buttonSaveLink'], vdoLnk);
  657. saver['isShowingLink'] = true;
  658. }
  659. }
  660. }
  661. }
  662. }
  663. }
  664.  
  665. function showMyMessage(cause, content) {
  666. if (cause == '!content') {
  667. modifyMyElement(saver['saverMessage'], {innerHTML: 'Couldn\'t get the videos content. Please report it <a href="' + contact + '" style="color:#00892C">here</a>.'});
  668. }
  669. else if (cause == '!videos') {
  670. modifyMyElement(saver['saverMessage'], {innerHTML: 'Couldn\'t get any video. Please report it <a href="' + contact + '" style="color:#00892C">here</a>.'});
  671. }
  672. else if (cause == '!support') {
  673. modifyMyElement(saver['saverMessage'], {innerHTML: 'This video uses the RTMP protocol which is not supported.'});
  674. }
  675. else if (cause == 'embed') {
  676. modifyMyElement(saver['saverMessage'], {innerHTML: 'This is an embedded video. You can get it <a href="' + content + '" style="color:#00892C">here</a>.'});
  677. }
  678. else if (cause == 'other') {
  679. modifyMyElement(saver['saverMessage'], {innerHTML: content});
  680. }
  681. }
  682.  
  683.  
  684. // ==========Websites========== //
  685.  
  686. function SaveTube() {
  687.  
  688. // =====YouTube===== //
  689.  
  690. if (page.url.indexOf('youtube.com/watch') != -1) {
  691.  
  692. /* Video Availability */
  693. if (getMyContent(page.url, '"playabilityStatus":\\{"status":"(ERROR|UNPLAYABLE)"', false)) return;
  694. var ytVideoUnavailable = getMyElement('', 'div', 'id', 'player-unavailable', -1, false);
  695. if (ytVideoUnavailable) {
  696. if (ytVideoUnavailable.className.indexOf('hid') == -1) {
  697. var ytAgeGateContent = getMyElement('', 'div', 'id', 'watch7-player-age-gate-content', -1, true);
  698. if (!ytAgeGateContent) return;
  699. else {
  700. if(ytAgeGateContent.indexOf('feature=private_video') != -1) return;
  701. }
  702. }
  703. }
  704.  
  705. /* Decrypt Signature */
  706. var ytScriptSrc;
  707. function ytDecryptSignature(s) {return null;}
  708. function ytDecryptFunction() {
  709. var ytSignFuncName, ytSignFuncBody, ytSwapFuncName, ytSwapFuncBody, ytFuncMatch;
  710. ytScriptSrc = ytScriptSrc.replace(/(\r\n|\n|\r)/gm, '');
  711. ytSignFuncName = ytScriptSrc.match(/"signature"\s*,\s*([^\)]*?)\(/);
  712. if (!ytSignFuncName) ytSignFuncName = ytScriptSrc.match(/c&&.\.set\(b,(?:encodeURIComponent\()?.*?([$a-zA-Z0-9]+)\(/);
  713. if (!ytSignFuncName) ytSignFuncName = ytScriptSrc.match(/c&&\([a-zA-Z0-9$]+=([a-zA-Z0-9$]+)\(decodeURIComponent/);
  714. ytSignFuncName = (ytSignFuncName) ? ytSignFuncName[1] : null;
  715. if (ytSignFuncName) {
  716. ytFuncMatch = ';' + ytSignFuncName.replace(/\$/, '\\$') + '\\s*=\\s*function\\s*' + '\\s*\\(\\w+\\)\\s*\\{(.*?)\\}';
  717. ytSignFuncBody = ytScriptSrc.match(ytFuncMatch);
  718. ytSignFuncBody = (ytSignFuncBody) ? ytSignFuncBody[1] : null;
  719. if (ytSignFuncBody) {
  720. ytSwapFuncName = ytSignFuncBody.match(/((\$|_|\w)+)\.(\$|_|\w)+\(\w,[0-9]+\)/);
  721. ytSwapFuncName = (ytSwapFuncName) ? ytSwapFuncName[1] : null;
  722. if (ytSwapFuncName) {
  723. ytFuncMatch = 'var\\s+' + ytSwapFuncName.replace(/\$/, '\\$') + '=\\s*\\{(.*?)\\};';
  724. ytSwapFuncBody = ytScriptSrc.match(ytFuncMatch);
  725. ytSwapFuncBody = (ytSwapFuncBody) ? ytSwapFuncBody[1] : null;
  726. }
  727. if (ytSwapFuncBody) ytSignFuncBody = 'var ' + ytSwapFuncName + '={' + ytSwapFuncBody + '};' + ytSignFuncBody;
  728. ytSignFuncBody = 'try {' + ytSignFuncBody + '} catch(e) {return null}';
  729. ytDecryptSignature = new Function('a', ytSignFuncBody);
  730. }
  731. }
  732. }
  733.  
  734. /* Get Video Title */
  735. var ytVideoTitle = getMyContent(page.url, '"videoDetails".*?"title":"((\\\\"|[^"])*?)"', false);
  736. if (!ytVideoTitle) ytVideoTitle = getMyContent(page.url, '"title":\\{"runs":\\[\\{"text":"((\\\\"|[^"])*?)"', false);
  737. if (!ytVideoTitle) ytVideoTitle = getMyContent(page.url, 'meta\\s+property="og:title"\\s+content="(.*?)"', false);
  738. if (!ytVideoTitle) ytVideoTitle = getMyContent(page.url, 'meta\\s+itemprop="name"\\s+content="(.*?)"', false);
  739. if (ytVideoTitle) {
  740. var ytVideoAuthor = getMyContent(page.url, '"(?:author|name)":\\s*"((\\\\"|[^"])*?)"', false);
  741. if (ytVideoAuthor) ytVideoTitle = ytVideoTitle + ' by ' + ytVideoAuthor;
  742. ytVideoTitle = cleanMyContent(ytVideoTitle, false, true);
  743. }
  744.  
  745. /* Get Videos Content */
  746. var ytVideosEncodedFmts, ytVideosEncodedFmtsNew, ytVideosAdaptiveFmts, ytVideosAdaptiveFmtsNew, ytVideosContent, ytHLSVideos, ytHLSContent;
  747. ytVideosEncodedFmts = getMyContent(page.url, '"url_encoded_fmt_stream_map\\\\?":\\s*\\\\?"(.*?)\\\\?"', false);
  748. if (!ytVideosEncodedFmts) {
  749. ytVideosEncodedFmtsNew = getMyContent(page.url, '"formats\\\\?":\\s*(\\[.*?\\])', false);
  750. if (ytVideosEncodedFmtsNew) {
  751. ytVideosEncodedFmts = '';
  752. ytVideosEncodedFmtsNew = cleanMyContent(ytVideosEncodedFmtsNew, false);
  753. ytVideosEncodedFmtsNew = ytVideosEncodedFmtsNew.match(new RegExp('"(url|cipher|signatureCipher)":\s*".*?"', 'g'));
  754. if (ytVideosEncodedFmtsNew) {
  755. for (var i = 0 ; i < ytVideosEncodedFmtsNew.length; i++) {
  756. ytVideosEncodedFmts += ytVideosEncodedFmtsNew[i].replace(/"/g, '').replace('url:', 'url=').replace('cipher:', '').replace('signatureCipher:', '') + ',';
  757. }
  758. if (ytVideosEncodedFmts.indexOf('itag%3D') != -1) {
  759. ytVideosEncodedFmts = cleanMyContent(ytVideosEncodedFmts, true);
  760. }
  761. }
  762. }
  763. }
  764. ytVideosAdaptiveFmts = getMyContent(page.url, '"adaptive_fmts\\\\?":\\s*\\\\?"(.*?)\\\\?"', false);
  765. if (!ytVideosAdaptiveFmts) {
  766. ytVideosAdaptiveFmtsNew = getMyContent(page.url, '"adaptiveFormats\\\\?":\\s*(\\[.*?\\])', false);
  767. if (ytVideosAdaptiveFmtsNew) {
  768. ytVideosAdaptiveFmts = '';
  769. ytVideosAdaptiveFmtsNew = cleanMyContent(ytVideosAdaptiveFmtsNew, false);
  770. ytVideosAdaptiveFmtsNew = ytVideosAdaptiveFmtsNew.match(new RegExp('"(url|cipher|signatureCipher)":\s*".*?"', 'g'));
  771. if (ytVideosAdaptiveFmtsNew) {
  772. for (var i = 0 ; i < ytVideosAdaptiveFmtsNew.length; i++) {
  773. ytVideosAdaptiveFmts += ytVideosAdaptiveFmtsNew[i].replace(/"/g, '').replace('url:', 'url=').replace('cipher:', '').replace('signatureCipher:', '') + ',';
  774. }
  775. if (ytVideosAdaptiveFmts.indexOf('itag%3D') != -1) {
  776. ytVideosAdaptiveFmts = cleanMyContent(ytVideosAdaptiveFmts, true);
  777. }
  778. }
  779. }
  780. }
  781. if (!ytVideosAdaptiveFmts) {
  782. var ytDASHVideos, ytDASHContent;
  783. ytDASHVideos = getMyContent(page.url, '"dash(?:mpd|ManifestUrl)\\\\?":\\s*\\\\?"(.*?)\\\\?"', false);
  784. if (ytDASHVideos) {
  785. ytDASHVideos = cleanMyContent(ytDASHVideos, false);
  786. ytDASHContent = getMyContent(ytDASHVideos + '?pacing=0', 'TEXT', false);
  787. if (ytDASHContent) {
  788. var ytDASHVideo, ytDASHVideoParts, ytDASHVideoServer, ytDASHVideoParams;
  789. ytDASHVideos = ytDASHContent.match(new RegExp('<BaseURL>.*?</BaseURL>', 'g'));
  790. if (ytDASHVideos) {
  791. ytVideosAdaptiveFmts = '';
  792. for (var i = 0; i < ytDASHVideos.length; i++) {
  793. ytDASHVideo = ytDASHVideos[i].replace('<BaseURL>', '').replace('</BaseURL>', '');
  794. if (ytDASHVideo.indexOf('source/youtube') == -1) continue;
  795. ytDASHVideoParts = ytDASHVideo.split('videoplayback/');
  796. ytDASHVideoServer = ytDASHVideoParts[0] + 'videoplayback?';
  797. ytDASHVideoParams = ytDASHVideoParts[1].split('/');
  798. ytDASHVideo = '';
  799. for (var p = 0; p < ytDASHVideoParams.length; p++) {
  800. if (p % 2) ytDASHVideo += ytDASHVideoParams[p] + '&';
  801. else ytDASHVideo += ytDASHVideoParams[p] + '=';
  802. }
  803. ytDASHVideo = encodeURIComponent(ytDASHVideoServer + ytDASHVideo);
  804. ytDASHVideo = ytDASHVideo.replace('itag%3D', 'itag=');
  805. ytVideosAdaptiveFmts += ytDASHVideo + ',';
  806. }
  807. }
  808. }
  809. }
  810. }
  811. if (ytVideosEncodedFmts) {
  812. ytVideosContent = ytVideosEncodedFmts;
  813. }
  814. else {
  815. ytHLSVideos = getMyContent(page.url, '"hls(?:vp|ManifestUrl)\\\\?":\\s*\\\\?"(.*?)\\\\?"', false);
  816. if (ytHLSVideos) {
  817. ytHLSVideos = cleanMyContent(ytHLSVideos, false);
  818. if (ytHLSVideos.indexOf('keepalive/yes/') != -1) ytHLSVideos = ytHLSVideos.replace('keepalive/yes/', '');
  819. }
  820. else {
  821. var ytVideoID = page.url.match(/(\?|&)v=(.*?)(&|$)/);
  822. ytVideoID = (ytVideoID) ? ytVideoID[2] : null;
  823. if (ytVideoID) {
  824. var ytVideosInfoPage = page.win.location.protocol + '//' + page.win.location.hostname + '/get_video_info?video_id=' + ytVideoID + '&eurl=https://youtube.googleapis.com/v/';
  825. ytVideosEncodedFmts = getMyContent(ytVideosInfoPage, 'url_encoded_fmt_stream_map=(.*?)&', false);
  826. if (ytVideosEncodedFmts) {
  827. ytVideosEncodedFmts = cleanMyContent(ytVideosEncodedFmts, true);
  828. ytVideosContent = ytVideosEncodedFmts;
  829. }
  830. else {
  831. ytVideosEncodedFmtsNew = getMyContent(ytVideosInfoPage, 'formats%22%3A(%5B.*?%5D)', false);
  832. if (ytVideosEncodedFmtsNew) {
  833. ytVideosEncodedFmts = '';
  834. ytVideosEncodedFmtsNew = cleanMyContent(ytVideosEncodedFmtsNew, true);
  835. ytVideosEncodedFmtsNew = ytVideosEncodedFmtsNew.match(new RegExp('"(url|cipher|signatureCipher)":\s*".*?"', 'g'));
  836. if (ytVideosEncodedFmtsNew) {
  837. for (var i = 0 ; i < ytVideosEncodedFmtsNew.length; i++) {
  838. ytVideosEncodedFmts += ytVideosEncodedFmtsNew[i].replace(/"/g, '').replace('url:', 'url=').replace('cipher:', '').replace('signatureCipher:', '') + ',';
  839. }
  840. if (ytVideosEncodedFmts.indexOf('itag%3D') != -1) {
  841. ytVideosEncodedFmts = cleanMyContent(ytVideosEncodedFmts, true);
  842. }
  843. ytVideosContent = ytVideosEncodedFmts;
  844. }
  845. }
  846. }
  847. if (!ytVideosAdaptiveFmts) {
  848. ytVideosAdaptiveFmts = getMyContent(ytVideosInfoPage, 'adaptive_fmts=(.*?)&', false);
  849. if (ytVideosAdaptiveFmts) {
  850. ytVideosAdaptiveFmts = cleanMyContent(ytVideosAdaptiveFmts, true);
  851. }
  852. else {
  853. ytVideosAdaptiveFmtsNew = getMyContent(ytVideosInfoPage, 'adaptiveFormats%22%3A(%5B.*?%5D)', false);
  854. if (ytVideosAdaptiveFmtsNew) {
  855. ytVideosAdaptiveFmts = '';
  856. ytVideosAdaptiveFmtsNew = cleanMyContent(ytVideosAdaptiveFmtsNew, true);
  857. ytVideosAdaptiveFmtsNew = ytVideosAdaptiveFmtsNew.match(new RegExp('"(url|cipher|signatureCipher)":\s*".*?"', 'g'));
  858. if (ytVideosAdaptiveFmtsNew) {
  859. for (var i = 0 ; i < ytVideosAdaptiveFmtsNew.length; i++) {
  860. ytVideosAdaptiveFmts += ytVideosAdaptiveFmtsNew[i].replace(/"/g, '').replace('url:', 'url=').replace('cipher:', '').replace('signatureCipher:', '') + ',';
  861. }
  862. if (ytVideosAdaptiveFmts.indexOf('itag%3D') != -1) {
  863. ytVideosAdaptiveFmts = cleanMyContent(ytVideosAdaptiveFmts, true);
  864. }
  865. }
  866. }
  867. }
  868. }
  869. }
  870. }
  871. }
  872. if (ytVideosAdaptiveFmts && !ytHLSVideos) {
  873. if (ytVideosContent) ytVideosContent += ',' + ytVideosAdaptiveFmts;
  874. else ytVideosContent = ytVideosAdaptiveFmts;
  875. }
  876.  
  877. /* Create Saver */
  878. var ytDefaultVideo = 'Low Definition MP4';
  879. function ytSaver() {
  880. saver = {
  881. 'videoList': ytVideoList,
  882. 'videoDefinitions': ['Ultra High Definition', 'Quad High Definition', 'Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
  883. 'videoContainers': ['MP4', 'WebM', 'M3U8', 'Any'],
  884. 'videoSave': ytDefaultVideo,
  885. 'videoTitle': ytVideoTitle
  886. };
  887. }
  888.  
  889. /* Parse Videos */
  890. function ytVideos() {
  891. var ytVideoFormats = {
  892. '18': 'Low Definition MP4',
  893. '22': 'High Definition MP4',
  894. '43': 'Low Definition WebM',
  895. '133': 'Very Low Definition Video MP4',
  896. '134': 'Low Definition Video MP4',
  897. '135': 'Standard Definition Video MP4',
  898. '136': 'High Definition Video MP4',
  899. '137': 'Full High Definition Video MP4',
  900. '140': 'Medium Bitrate Audio MP4',
  901. '242': 'Very Low Definition Video WebM',
  902. '243': 'Low Definition Video WebM',
  903. '244': 'Standard Definition Video WebM',
  904. '247': 'High Definition Video WebM',
  905. '248': 'Full High Definition Video WebM',
  906. '249': 'Low Bitrate Audio WebM',
  907. '250': 'Medium Bitrate Audio WebM',
  908. '251': 'High Bitrate Audio WebM',
  909. '264': 'Quad High Definition Video MP4',
  910. '271': 'Quad High Definition Video WebM',
  911. '272': 'Ultra High Definition Video WebM',
  912. '298': 'High Definition Video MP4',
  913. '299': 'Full High Definition Video MP4',
  914. '302': 'High Definition Video WebM',
  915. '303': 'Full High Definition Video WebM',
  916. '308': 'Quad High Definition Video WebM',
  917. '313': 'Ultra High Definition Video WebM',
  918. '315': 'Ultra High Definition Video WebM',
  919. '333': 'Standard Definition Video WebM',
  920. '334': 'High Definition Video WebM',
  921. '335': 'Full High Definition Video WebM',
  922. '337': 'Ultra High Definition Video WebM'
  923. };
  924. var ytVideoFound = false;
  925. var ytVideos = ytVideosContent.split(',');
  926. var ytVideoParse, ytVideoCodeParse, ytVideoCode, myVideoCode, ytVideo, ytSign, ytSignP;
  927. for (var i = 0; i < ytVideos.length; i++) {
  928. ytVideo = ytVideos[i];
  929. ytVideoCodeParse = ytVideo.match(/itag=(\d{1,3})/);
  930. ytVideoCode = (ytVideoCodeParse) ? ytVideoCodeParse[1] : null;
  931. if (!ytVideoCode) continue;
  932. myVideoCode = ytVideoFormats[ytVideoCode];
  933. if (!myVideoCode) continue;
  934. if (!ytVideo.match(/^url/)) {
  935. ytVideoParse = ytVideo.match(/(.*)(url=.*$)/);
  936. if (ytVideoParse) ytVideo = ytVideoParse[2] + '&' + ytVideoParse[1];
  937. }
  938. ytVideo = cleanMyContent(ytVideo, true);
  939. if (myVideoCode.indexOf('Video') != -1) {
  940. if (ytVideo.indexOf('source=yt_otf') != -1) continue;
  941. }
  942. ytVideo = ytVideo.replace(/url=/, '').replace(/&$/, '');
  943. if (ytVideo.match(/itag=/) && ytVideo.match(/itag=/g).length > 1) {
  944. if (ytVideo.match(/itag=\d{1,3}&/)) ytVideo = ytVideo.replace(/itag=\d{1,3}&/, '');
  945. else if (ytVideo.match(/&itag=\d{1,3}/)) ytVideo = ytVideo.replace(/&itag=\d{1,3}/, '');
  946. }
  947. if (ytVideo.match(/clen=/) && ytVideo.match(/clen=/g).length > 1) {
  948. if (ytVideo.match(/clen=\d+&/)) ytVideo = ytVideo.replace(/clen=\d+&/, '');
  949. else if (ytVideo.match(/&clen=\d+/)) ytVideo = ytVideo.replace(/&clen=\d+/, '');
  950. }
  951. if (ytVideo.match(/lmt=/) && ytVideo.match(/lmt=/g).length > 1) {
  952. if (ytVideo.match(/lmt=\d+&/)) ytVideo = ytVideo.replace(/lmt=\d+&/, '');
  953. else if (ytVideo.match(/&lmt=\d+/)) ytVideo = ytVideo.replace(/&lmt=\d+/, '');
  954. }
  955. if (ytVideo.match(/type=(video|audio).*?&/)) ytVideo = ytVideo.replace(/type=(video|audio).*?&/, '');
  956. else ytVideo = ytVideo.replace(/&type=(video|audio).*$/, '');
  957. if (ytVideo.match(/xtags=[^%=]*&/)) ytVideo = ytVideo.replace(/xtags=[^%=]*?&/, '');
  958. else if (ytVideo.match(/&xtags=[^%=]*$/)) ytVideo = ytVideo.replace(/&xtags=[^%=]*$/, '');
  959. if (ytVideo.match(/&sig=/) && !ytVideo.match(/&lsig=/)) ytVideo = ytVideo.replace(/&sig=/, '&signature=');
  960. else if (ytVideo.match(/&s=/)) {
  961. ytSign = ytVideo.match(/&s=(.*?)(&|$)/);
  962. ytSign = (ytSign) ? ytSign[1] : null;
  963. if (ytSign) {
  964. ytSign = ytDecryptSignature(ytSign);
  965. if (ytSign) {
  966. ytSignP = ytVideo.match(/&sp=(.*?)(&|$)/);
  967. ytSignP = (ytSignP) ? ytSignP[1] : ((ytVideo.match(/&lsig=/)) ? 'sig' : 'signature');
  968. ytVideo = ytVideo.replace(/&s=.*?(&|$)/, '&' + ytSignP + '=' + ytSign + '$1');
  969. }
  970. else ytVideo = '';
  971. }
  972. else ytVideo = '';
  973. }
  974. ytVideo = cleanMyContent(ytVideo, true);
  975. if (ytVideo.indexOf('ratebypass') == -1) ytVideo += '&ratebypass=yes';
  976. if (ytVideo && ytVideo.indexOf('http') == 0) {
  977. if (!ytVideoFound) ytVideoFound = true;
  978. ytVideoList[myVideoCode] = ytVideo;
  979. }
  980. }
  981.  
  982. if (ytVideoFound) {
  983. /* Save DASH */
  984. if (ytVideoList['Medium Bitrate Audio MP4'] || ytVideoList['Medium Bitrate Audio WebM']) {
  985. for (var myVideoCode in ytVideoList) {
  986. if (myVideoCode.indexOf('Video') != -1) {
  987. if (!ytVideoList[myVideoCode.replace(' Video', '')]) {
  988. ytVideoList[myVideoCode.replace(' Video', '')] = 'DASH';
  989. }
  990. }
  991. }
  992. }
  993. /* Create Saver */
  994. ytSaver();
  995. createMySaver();
  996. }
  997. else {
  998. saver = {};
  999. if (ytVideosContent.indexOf('conn=rtmp') != -1) saver['warnMess'] = '!support';
  1000. else saver['warnMess'] = '!videos';
  1001. createMySaver();
  1002. }
  1003. }
  1004.  
  1005. /* Parse HLS */
  1006. function ytHLS() {
  1007. var ytHLSFormats = {
  1008. '92': 'Very Low Definition M3U8',
  1009. '93': 'Low Definition M3U8',
  1010. '94': 'Standard Definition M3U8',
  1011. '95': 'High Definition M3U8',
  1012. '96': 'Full High Definition M3U8'
  1013. };
  1014. ytVideoList["Multi Definition M3U8"] = ytHLSVideos;
  1015. if (ytHLSContent) {
  1016. var ytHLSVideo, ytVideoCodeParse, ytVideoCode, myVideoCode;
  1017. var ytHLSMatcher = new RegExp('(http.*?m3u8)', 'g');
  1018. ytHLSVideos = ytHLSContent.match(ytHLSMatcher);
  1019. if (ytHLSVideos) {
  1020. for (var i = 0; i < ytHLSVideos.length; i++) {
  1021. ytHLSVideo = ytHLSVideos[i];
  1022. ytVideoCodeParse = ytHLSVideo.match(/\/itag\/(\d{1,3})\//);
  1023. ytVideoCode = (ytVideoCodeParse) ? ytVideoCodeParse[1] : null;
  1024. if (ytVideoCode) {
  1025. myVideoCode = ytHLSFormats[ytVideoCode];
  1026. if (myVideoCode && ytHLSVideo) {
  1027. ytVideoList[myVideoCode] = ytHLSVideo;
  1028. }
  1029. }
  1030. }
  1031. }
  1032. }
  1033. /* Create Saver */
  1034. ytDefaultVideo = 'Multi Definition M3U8';
  1035. ytSaver();
  1036. createMySaver();
  1037. }
  1038.  
  1039. /* Get Videos */
  1040. var ytVideoList = {};
  1041. if (ytVideosContent) {
  1042. if (ytVideosContent.match(/^s=/) || ytVideosContent.match(/&s=/) || ytVideosContent.match(/,s=/) || ytVideosContent.match(/u0026s=/)) {
  1043. var ytScriptURL = getMyContent(page.url, '"js(?:Url)?":\\s*"(.*?)"', true);
  1044. if (!ytScriptURL) ytScriptURL = getMyContent(page.url.replace(/watch.*?v=/, 'embed/').replace(/&.*$/, ''), '"js(?:Url)?":\\s*"(.*?)"', true);
  1045. if (ytScriptURL) {
  1046. ytScriptURL = page.win.location.protocol + '//' + page.win.location.hostname + ytScriptURL;
  1047. ytScriptSrc = getMyContent(ytScriptURL, 'TEXT', false);
  1048. if (ytScriptSrc) ytDecryptFunction();
  1049. ytVideos();
  1050. }
  1051. else {
  1052. saver = {
  1053. 'warnMess': 'other',
  1054. 'warnContent': '<b>SaveTube:</b> Couldn\'t get the signature link. Please report it <a href="' + contact + '" style="color:#00892C">here</a>.'
  1055. };
  1056. createMySaver();
  1057. }
  1058. }
  1059. else {
  1060. ytVideos();
  1061. }
  1062. }
  1063. else {
  1064. if (ytHLSVideos) {
  1065. ytHLSContent = getMyContent(ytHLSVideos, 'TEXT', false);
  1066. ytHLS();
  1067. }
  1068. else {
  1069. saver = {'warnMess': '!content'};
  1070. createMySaver();
  1071. }
  1072. }
  1073.  
  1074. }
  1075.  
  1076. // =====DailyMotion===== //
  1077.  
  1078. else if (page.url.indexOf('dailymotion.com/video') != -1) {
  1079.  
  1080. /* Video Availability */
  1081. if (getMyContent(page.url.replace(/\/video\//, "/embed/video/"), '"error":\\{"title":"(.*?)"', false)) return;
  1082. if (getMyContent(page.url.replace(/\/video\//, "/embed/video/"), '"error_title":"(.*?)"', false)) return;
  1083.  
  1084. /* Get Video Title */
  1085. var dmVideoTitle = getMyContent(page.url.replace(/\/video\//, "/embed/video/"), '"title":"((\\\\"|[^"])*?)"', false);
  1086. if (dmVideoTitle) {
  1087. var dmVideoAuthor = getMyContent(page.url.replace(/\/video\//, "/embed/video/"), '"screenname":"((\\\\"|[^"])*?)"', false);
  1088. if (dmVideoAuthor) dmVideoTitle = dmVideoTitle + ' by ' + dmVideoAuthor;
  1089. dmVideoTitle = cleanMyContent(dmVideoTitle, false, true);
  1090. }
  1091.  
  1092. /* Get Videos Content */
  1093. var dmVideosContent = getMyContent(page.url.replace(/\/video\//, "/embed/video/"), '"qualities":\\{(.*?)\\]\\},', false);
  1094.  
  1095. /* Get Videos */
  1096. if (dmVideosContent) {
  1097. var dmVideoFormats = {'auto': 'Low Definition MP4', '240': 'Very Low Definition MP4', '380': 'Low Definition MP4',
  1098. '480': 'Standard Definition MP4', '720': 'High Definition MP4', '1080': 'Full High Definition MP4'};
  1099. var dmVideoList = {};
  1100. var dmVideoFound = false;
  1101. var dmVideoParser, dmVideoParse, myVideoCode, dmVideo;
  1102. for (var dmVideoCode in dmVideoFormats) {
  1103. dmVideoParser = '"' + dmVideoCode + '".*?"type":"video.*?mp4","url":"(.*?)"';
  1104. dmVideoParse = dmVideosContent.match(dmVideoParser);
  1105. dmVideo = (dmVideoParse) ? dmVideoParse[1] : null;
  1106. if (dmVideo) {
  1107. if (!dmVideoFound) dmVideoFound = true;
  1108. dmVideo = cleanMyContent(dmVideo, true);
  1109. myVideoCode = dmVideoFormats[dmVideoCode];
  1110. if (!dmVideoList[myVideoCode]) dmVideoList[myVideoCode] = dmVideo;
  1111. }
  1112. }
  1113. if (!dmVideoFound) {
  1114. dmVideoParser = '"' + dmVideoCode + '".*?"type":"application.*?mpegURL","url":"(.*?)"';
  1115. dmVideoParse = dmVideosContent.match(dmVideoParser);
  1116. if (dmVideoParse) {
  1117. dmVideo = (dmVideoParse) ? dmVideoParse[1] : null;
  1118. if (!dmVideoFound) dmVideoFound = true;
  1119. dmVideo = cleanMyContent(dmVideo, true);
  1120. dmVideoList["Multi Definition M3U8"] = dmVideo;
  1121. var dmManifestSource = getMyContent(dmVideo, 'TEXT', false);
  1122. if (dmManifestSource) {
  1123. dmManifestSource = cleanMyContent(dmManifestSource, false);
  1124. for (var dmVideoCode in dmVideoFormats) {
  1125. dmVideoParser = 'RESOLUTION=\\d*x\\d*?,NAME="' + dmVideoCode + '",PROGRESSIVE-URI="(.*?)"([^"]*)(#|$)';
  1126. dmVideoParse = dmManifestSource.match(dmVideoParser);
  1127. dmVideo = (dmVideoParse) ? dmVideoParse[1] : null;
  1128. if (dmVideo) {
  1129. myVideoCode = dmVideoFormats[dmVideoCode];
  1130. if (!dmVideoList[myVideoCode]) dmVideoList[myVideoCode] = dmVideo;
  1131. }
  1132. dmVideo = (dmVideoParse) ? dmVideoParse[2] : null;
  1133. if (dmVideo) {
  1134. myVideoCode = dmVideoFormats[dmVideoCode].replace('MP4', 'M3U8');
  1135. if (!dmVideoList[myVideoCode]) dmVideoList[myVideoCode] = dmVideo;
  1136. }
  1137. }
  1138. }
  1139. }
  1140. }
  1141.  
  1142. if (dmVideoFound) {
  1143. /* Create Saver */
  1144. var dmDefaultVideo = 'Low Definition MP4';
  1145. if (!dmVideoList[dmDefaultVideo]) dmDefaultVideo = 'Low Definition M3U8';
  1146. saver = {
  1147. 'videoList': dmVideoList,
  1148. 'videoDefinitions': ['Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
  1149. 'videoContainers': ['MP4'],
  1150. 'videoSave': dmDefaultVideo,
  1151. 'videoTitle': dmVideoTitle
  1152. };
  1153. createMySaver();
  1154. }
  1155. else {
  1156. saver = {'warnMess': '!videos'};
  1157. createMySaver();
  1158. }
  1159. }
  1160. else {
  1161. saver = {'warnMess': '!content'};
  1162. createMySaver();
  1163. }
  1164.  
  1165. }
  1166.  
  1167. // =====Vimeo===== //
  1168.  
  1169. else if (page.url.indexOf('vimeo.com/') != -1) {
  1170.  
  1171. /* Page Type */
  1172. var viPageType = getMyContent(page.url, 'meta\\s+property="og:type"\\s+content="(.*?)"', false);
  1173. if (!viPageType || (viPageType.indexOf('video') == -1 && viPageType.indexOf('profile') == -1)) return;
  1174.  
  1175. /* Get Video Title */
  1176. var viVideoTitle;
  1177. if (viPageType == 'video') {
  1178. viVideoTitle = getMyContent(page.url, 'meta\\s+property="og:title"\\s+content="(.*?)"', false);
  1179. }
  1180. else {
  1181. viVideoTitle = getMyContent(page.url, '"title":"((\\\\"|[^"])*?)"', false);
  1182. }
  1183. if (viVideoTitle) {
  1184. viVideoTitle = viVideoTitle.replace(/\s*on\s*Vimeo$/, '');
  1185. var viVideoAuthor = getMyContent(page.url, '"display_name":"((\\\\"|[^"])*?)"', false);
  1186. if (viVideoAuthor) viVideoTitle = viVideoTitle + ' by ' + viVideoAuthor;
  1187. viVideoTitle = cleanMyContent(viVideoTitle, false, true);
  1188. }
  1189.  
  1190. /* Get Content Source */
  1191. var viVideoSource = getMyContent(page.url, 'config_url":"(.*?)"', false);
  1192. if (viVideoSource) viVideoSource = cleanMyContent(viVideoSource, false);
  1193. else {
  1194. viVideoSource = getMyContent(page.url, 'data-config-url="(.*?)"', false);
  1195. if (viVideoSource) viVideoSource = viVideoSource.replace(/&amp;/g, '&');
  1196. }
  1197.  
  1198. /* Get Videos Content */
  1199. var viVideosContent;
  1200. if (viVideoSource) {
  1201. viVideosContent = getMyContent(viVideoSource, '"progressive":\\[(.*?)\\]', false);
  1202. }
  1203.  
  1204. /* Get Videos */
  1205. if (viVideosContent) {
  1206. var viVideoFormats = {'1440p': 'Quad High Definition MP4', '1080p': 'Full High Definition MP4', '720p': 'High Definition MP4', '540p': 'Standard Definition MP4',
  1207. '480p': 'Standard Definition MP4', '360p': 'Low Definition MP4', '270p': 'Very Low Definition MP4', '240p': 'Very Low Definition MP4'};
  1208. var viVideoList = {};
  1209. var viVideoFound = false;
  1210. var viVideo, myVideoCode;
  1211. var viVideos = viVideosContent.split('},');
  1212. for (var i = 0; i < viVideos.length; i++) {
  1213. for (var viVideoCode in viVideoFormats) {
  1214. if (viVideos[i].indexOf('"quality":"' + viVideoCode + '"') != -1) {
  1215. viVideo = viVideos[i].match(/"url":"(.*?)"/);
  1216. viVideo = (viVideo) ? viVideo[1] : null;
  1217. if (viVideo) {
  1218. if (!viVideoFound) viVideoFound = true;
  1219. myVideoCode = viVideoFormats[viVideoCode];
  1220. viVideoList[myVideoCode] = viVideo;
  1221. }
  1222. }
  1223. }
  1224. }
  1225.  
  1226. if (viVideoFound) {
  1227. /* Create Saver */
  1228. var viDefaultVideo = 'Low Definition MP4';
  1229. saver = {
  1230. 'videoList': viVideoList,
  1231. 'videoDefinitions': ['Quad High Definition', 'Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
  1232. 'videoContainers': ['MP4'],
  1233. 'videoSave': viDefaultVideo,
  1234. 'videoTitle': viVideoTitle
  1235. };
  1236. createMySaver();
  1237. }
  1238. else {
  1239. saver = {'warnMess': '!videos'};
  1240. createMySaver();
  1241. }
  1242. }
  1243. else {
  1244. saver = {'warnMess': '!content'};
  1245. createMySaver();
  1246. }
  1247.  
  1248. }
  1249.  
  1250. // =====MetaCafe===== //
  1251.  
  1252. else if (page.url.indexOf('metacafe.com/watch') != -1) {
  1253.  
  1254. /* Get Video Title */
  1255. var mcVideoTitle = getMyContent(page.url, 'meta\\s+property="og:title"\\s+content="(.*?)"', false);
  1256. if (mcVideoTitle) mcVideoTitle = cleanMyContent(mcVideoTitle, false, true);
  1257.  
  1258. /* Get Videos Content */
  1259. var mcVideosContent = getMyContent(page.url, 'flashvars\\s*=\\s*\\{(.*?)\\};', false);
  1260.  
  1261. /* Get Videos */
  1262. if (mcVideosContent) {
  1263. var mcVideoList = {};
  1264. var mcVideoFound = false;
  1265. var mcVideoFormats = {'video_alt_url2': 'High Definition M3U8', 'video_alt_url': 'Low Definition M3U8', 'video_url': 'Very Low Definition M3U8'};
  1266. var mcVideoFormatz = {'video_alt_url2': '_720p', 'video_alt_url': '_360p', 'video_url': '_240p'};
  1267. var mcVideoHLS = mcVideosContent.match(/"src":"(.*?)"/);
  1268. mcVideoHLS = (mcVideoHLS) ? cleanMyContent(mcVideoHLS[1], false) : null;
  1269. if (mcVideoHLS) {
  1270. var mcVideoParser, mcVideoParse, myVideoCode, mcVideo;
  1271. for (var mcVideoCode in mcVideoFormats) {
  1272. mcVideoParser = '"' + mcVideoCode + '":"(.*?)"';
  1273. mcVideoParse = mcVideosContent.match(mcVideoParser);
  1274. mcVideo = (mcVideoParse) ? mcVideoParse[1] : null;
  1275. if (mcVideo) {
  1276. if (!mcVideoFound) mcVideoFound = true;
  1277. myVideoCode = mcVideoFormats[mcVideoCode];
  1278. mcVideoList[myVideoCode] = mcVideoHLS.replace('.m3u8', mcVideoFormatz[mcVideoCode] + '.m3u8');
  1279. }
  1280. }
  1281. }
  1282.  
  1283. if (mcVideoFound) {
  1284. /* Create Saver */
  1285. var mcDefaultVideo = 'Low Definition M3U8';
  1286. saver = {
  1287. 'videoList': mcVideoList,
  1288. 'videoDefinitions': ['High Definition', 'Low Definition', 'Very Low Definition'],
  1289. 'videoContainers': ['M3U8'],
  1290. 'videoSave': mcDefaultVideo,
  1291. 'videoTitle': mcVideoTitle
  1292. };
  1293. createMySaver();
  1294. }
  1295. else {
  1296. saver = {'warnMess': '!videos'};
  1297. createMySaver();
  1298. }
  1299. }
  1300. else {
  1301. saver = {};
  1302. var ytVideoId = page.url.match(/\/yt-(.*?)\//);
  1303. if (ytVideoId && ytVideoId[1]) {
  1304. var ytVideoLink = 'http://youtube.com/watch?v=' + ytVideoId[1];
  1305. saver['warnMess'] = 'embed';
  1306. saver['warnContent'] = ytVideoLink;
  1307. }
  1308. else saver['warnMess'] = '!videos';
  1309. createMySaver();
  1310. }
  1311.  
  1312. }
  1313.  
  1314. // =====Veoh===== //
  1315.  
  1316. else if (page.url.indexOf('veoh.com/watch') != -1) {
  1317.  
  1318. /* Get Video Availability */
  1319. if (getMyElement('', 'div', 'class', 'veoh-video-player-error', 0, false)) return;
  1320.  
  1321. /* Get Video Title */
  1322. var veVideoTitle = getMyContent(page.url, 'meta\\s+name="og:title"\\s+content="(.*?)"', false);
  1323. if (!veVideoTitle) {
  1324. veVideoTitle = getMyContent(page.url.replace(/\/watch\//, '/watch/getVideo/'), '"title":"((\\\\"|[^"])*?)"', false);
  1325. }
  1326. if (veVideoTitle) veVideoTitle = cleanMyContent(veVideoTitle, false, true);
  1327.  
  1328. /* Get Videos Content */
  1329. var veVideosContent = getMyContent(page.url.replace(/\/watch\//, '/watch/getVideo/'), '"src"\\s*:\\s*\\{(.*?)\\}', false);
  1330.  
  1331. /* Get Videos */
  1332. if (veVideosContent) {
  1333. var veVideoFormats = {'Regular': 'Low Definition MP4', 'HQ': 'Standard Definition MP4'};
  1334. var veVideoList = {};
  1335. var veVideoFound = false;
  1336. var veVideoParser, veVideoParse, veVideo, myVideoCode;
  1337. for (var veVideoCode in veVideoFormats) {
  1338. veVideoParser = veVideoCode + '":"(.*?)"';
  1339. veVideoParse = veVideosContent.match(veVideoParser);
  1340. veVideo = (veVideoParse) ? veVideoParse[1] : null;
  1341. if (veVideo) {
  1342. if (!veVideoFound) veVideoFound = true;
  1343. myVideoCode = veVideoFormats[veVideoCode];
  1344. veVideoList[myVideoCode] = cleanMyContent(veVideo, false);
  1345. }
  1346. }
  1347.  
  1348. if (veVideoFound) {
  1349. /* Create Saver */
  1350. var veDefaultVideo = 'Low Definition MP4';
  1351. saver = {
  1352. 'videoList': veVideoList,
  1353. 'videoDefinitions': ['Standard Definition', 'Low Definition'],
  1354. 'videoContainers': ['MP4'],
  1355. 'videoSave': veDefaultVideo,
  1356. 'videoTitle': veVideoTitle
  1357. };
  1358. createMySaver();
  1359. }
  1360. else {
  1361. saver = {};
  1362. var ytVideoId = getMyContent(page.url, 'youtube.com/embed/(.*?)("|\\?)', false);
  1363. if (!ytVideoId) ytVideoId = getMyContent(page.url, '"videoId":"yapi-(.*?)"', false);
  1364. if (ytVideoId) {
  1365. var ytVideoLink = 'http://youtube.com/watch?v=' + ytVideoId;
  1366. saver['warnMess'] = 'embed';
  1367. saver['warnContent'] = ytVideoLink;
  1368. }
  1369. else saver['warnMess'] = '!videos';
  1370. createMySaver();
  1371. }
  1372. }
  1373. else {
  1374. saver = {'warnMess': '!content'};
  1375. createMySaver();
  1376. }
  1377.  
  1378. }
  1379.  
  1380. // =====Viki===== //
  1381.  
  1382. else if (page.url.indexOf('viki.com/videos') != -1) {
  1383.  
  1384. /* Get Video Title */
  1385. var vkVideoTitle = getMyContent(page.url, 'meta\\s+property="og:title"\\s+content="(.*?)"', false);
  1386. if (vkVideoTitle) vkVideoTitle = cleanMyContent(vkVideoTitle, false, true);
  1387.  
  1388. /* Get Video ID */
  1389. var vkVideoID = page.url.match(/videos\/(\d+v)/);
  1390. vkVideoID = (vkVideoID) ? vkVideoID[1] : null;
  1391.  
  1392. /* Get Videos Content */
  1393. var vkVideosContent;
  1394. if (vkVideoID) {
  1395. /* SHA-1
  1396. Copyright 2008-2018 Brian Turek, 1998-2009 Paul Johnston & Contributors
  1397. Distributed under the BSD License
  1398. See https://caligatio.github.com/jsSHA/ for more information
  1399. */
  1400. var SHA1FuncBody;
  1401. var SHA1Key = 'sha1js';
  1402. try {
  1403. if (localStorage.getItem(SHA1Key)) {
  1404. SHA1FuncBody = localStorage.getItem(SHA1Key);
  1405. if (SHA1FuncBody.indexOf('SHA-1') == -1) throw false;
  1406. }
  1407. else throw false;
  1408. }
  1409. catch(e) {
  1410. SHA1FuncBody = getMyContent('https://raw.githack.com/Caligatio/jsSHA/master/dist/sha1.js', 'TEXT', false);
  1411. localStorage.setItem(SHA1Key, SHA1FuncBody);
  1412. }
  1413. var SHA1Func = new Function('a', SHA1FuncBody);
  1414. var SHA1 = new SHA1Func();
  1415. var shaObj = (SHA1.jsSHA) ? new SHA1.jsSHA("SHA-1", "TEXT") : new jsSHA("SHA-1", "TEXT");
  1416. var vkTimestamp = parseInt(Date.now() / 1000);
  1417. var vkQuery = "/v5/videos/" + vkVideoID + "/streams.json?app=100005a&t=" + vkTimestamp + "&site=www.viki.com"
  1418. var vkToken = "MM_d*yP@`&1@]@!AVrXf_o-HVEnoTnm$O-ti4[G~$JDI/Dc-&piU&z&5.;:}95\=Iad";
  1419. shaObj.setHMACKey(vkToken, "TEXT");
  1420. shaObj.update(vkQuery);
  1421. var vkSig = shaObj.getHMAC("HEX");
  1422. var vkSource = "https://api.viki.io" + vkQuery + "&sig=" + vkSig;
  1423. vkVideosContent = getMyContent(vkSource, 'TEXT', false);
  1424. }
  1425.  
  1426. /* Get Videos */
  1427. if (vkVideosContent) {
  1428. var vkVideoList = {};
  1429. var vkVideoFormats = {'1080p': 'Full High Definition MP4', '720p': 'High Definition MP4', '480p': 'Standard Definition MP4',
  1430. '360p': 'Low Definition MP4', '240p': 'Very Low Definition MP4'};
  1431. var vkVideoFound = false;
  1432. var vkVideoParser, vkVideoParse, vkVideo, myVideoCode;
  1433. for (var vkVideoCode in vkVideoFormats) {
  1434. vkVideoParser = '"' + vkVideoCode + '".*?"https":\{"url":"(.*?)"';
  1435. vkVideoParse = vkVideosContent.match(vkVideoParser);
  1436. vkVideo = (vkVideoParse) ? vkVideoParse[1] : null;
  1437. if (vkVideo) {
  1438. if (!vkVideoFound) vkVideoFound = true;
  1439. myVideoCode = vkVideoFormats[vkVideoCode];
  1440. vkVideoList[myVideoCode] = vkVideo;
  1441. }
  1442. }
  1443.  
  1444. // Unauthorized
  1445. var vkUnauthorized = (vkVideosContent.indexOf('unauthorized') != -1) ? true : false;
  1446.  
  1447. // DASH/HLS/Subtitles
  1448. vkVideosContent = getMyContent(page.url.replace('/videos/', '/player5_fragment/'), 'TEXT', false);
  1449. if (vkVideosContent) {
  1450. vkVideoEncHLS = vkVideosContent.match(/x-mpegURL".*?stream=(.*?)"/);
  1451. vkVideoEncHLS = (vkVideoEncHLS) ? vkVideoEncHLS[1] : null;
  1452. vkVideoEncDASH = vkVideosContent.match(/dash\+xml".*?stream=(.*?)"/);
  1453. vkVideoEncDASH = (vkVideoEncDASH) ? vkVideoEncDASH[1] : null;
  1454. if (vkVideoEncHLS || vkVideoEncDASH) {
  1455. vkVideoEncKey = vkVideosContent.match(/chabi:\s*'(.*?)'/);
  1456. vkVideoEncKey = (vkVideoEncKey) ? vkVideoEncKey[1] : null;
  1457. vkVideoEncIV = vkVideosContent.match(/ecta:\s*'(.*?)'/);
  1458. vkVideoEncIV = (vkVideoEncIV) ? vkVideoEncIV[1] : null;
  1459. if (vkVideoEncKey && vkVideoEncIV) {
  1460. /* AES
  1461. Copyright 2015-2018 Richard Moore
  1462. MIT License.
  1463. See https://github.com/ricmoo/aes-js/ for more information
  1464. */
  1465. var AESFuncBody;
  1466. var AESFuncKey = 'aesjs';
  1467. try {
  1468. if (localStorage.getItem(AESFuncKey)) {
  1469. AESFuncBody = localStorage.getItem(AESFuncKey);
  1470. }
  1471. else throw false;
  1472. }
  1473. catch(e) {
  1474. AESFuncBody = getMyContent('https://raw.githack.com/ricmoo/aes-js/master/index.js', 'TEXT', false);
  1475. localStorage.setItem(AESFuncKey, AESFuncBody);
  1476. }
  1477. var AESFunc = new Function('a', AESFuncBody);
  1478. var AES = new AESFunc();
  1479. var AESKey = AES.aesjs.utils.utf8.toBytes(vkVideoEncKey);
  1480. var AESIV = AES.aesjs.utils.utf8.toBytes(vkVideoEncIV);
  1481. var encryptedBytes, decryptedBytes;
  1482. // HLS
  1483. if (vkVideoEncHLS) {
  1484. encryptedBytes = AES.aesjs.utils.hex.toBytes(vkVideoEncHLS);
  1485. AESCBC = new AES.aesjs.ModeOfOperation.cbc(AESKey, AESIV);
  1486. decryptedBytes = AESCBC.decrypt(encryptedBytes);
  1487. var vkHLSManifest = AES.aesjs.utils.utf8.fromBytes(decryptedBytes);
  1488. if (vkHLSManifest) {
  1489. if (!vkVideoFound) vkVideoFound = true;
  1490. vkVideoList['Multi Definition M3U8'] = vkHLSManifest;
  1491. }
  1492. }
  1493. // DASH
  1494. if (vkVideoEncDASH) {
  1495. encryptedBytes = AES.aesjs.utils.hex.toBytes(vkVideoEncDASH);
  1496. AESCBC = new AES.aesjs.ModeOfOperation.cbc(AESKey, AESIV);
  1497. decryptedBytes = AESCBC.decrypt(encryptedBytes);
  1498. var vkDASHManifest = AES.aesjs.utils.utf8.fromBytes(decryptedBytes);
  1499. if (vkDASHManifest) {
  1500. var vkDASHDomain = vkDASHManifest.split('/').splice(0, 5).join('/');
  1501. var vkDASHContent = getMyContent(vkDASHManifest, 'TEXT', false);
  1502. if (vkDASHContent) {
  1503. var vkDASHVideo;
  1504. var vkDASHVideos = vkDASHContent.match(new RegExp('<BaseURL>.*?</BaseURL>', 'g'));
  1505. if (vkDASHVideos) {
  1506. for (var i = 0; i < vkDASHVideos.length; i++) {
  1507. vkDASHVideo = vkDASHVideos[i].replace('<BaseURL>', '').replace('</BaseURL>', '');
  1508. if (vkDASHVideo.indexOf('http') != 0) vkDASHVideo = vkDASHDomain + '/' + vkDASHVideo;
  1509. for (var vkVideoCode in vkVideoFormats) {
  1510. if (vkDASHVideo.indexOf(vkVideoCode) != -1) {
  1511. myVideoCode = vkVideoFormats[vkVideoCode];
  1512. if (vkDASHVideo.indexOf('track1') != -1 || vkDASHVideo.indexOf('video') != -1) {
  1513. if (!vkVideoFound) vkVideoFound = true;
  1514. if (!vkVideoList[myVideoCode]) {
  1515. if (vkDASHVideo.indexOf('drm') != -1) {
  1516. vkVideoList[myVideoCode.replace('MP4', 'Video MP4 [DRM]')] = vkDASHVideo;
  1517. }
  1518. else {
  1519. vkVideoList[myVideoCode.replace('MP4', 'Video MP4')] = vkDASHVideo;
  1520. }
  1521. }
  1522. }
  1523. if (vkDASHVideo.indexOf('track2') != -1 || vkDASHVideo.indexOf('audio') != -1) {
  1524. if (!vkVideoList[myVideoCode]) {
  1525. if (vkDASHVideo.indexOf('drm') != -1) {
  1526. vkVideoList[myVideoCode.replace('MP4', 'Audio MP4 [DRM]')] = vkDASHVideo;
  1527. }
  1528. else {
  1529. vkVideoList[myVideoCode.replace('MP4', 'Audio MP4')] = vkDASHVideo;
  1530. }
  1531. }
  1532. }
  1533. }
  1534. }
  1535. }
  1536. }
  1537. for (var vkVideoCode in vkVideoFormats) {
  1538. myVideoCode = vkVideoFormats[vkVideoCode];
  1539. if (!vkVideoList[myVideoCode]) {
  1540. if (vkVideoList[myVideoCode.replace('MP4', 'Video MP4')] && vkVideoList[myVideoCode.replace('MP4', 'Audio MP4')]) {
  1541. vkVideoList[myVideoCode] = 'DASH';
  1542. }
  1543. }
  1544. }
  1545. }
  1546. }
  1547. }
  1548. }
  1549. }
  1550. // Subtitles
  1551. var vkSubtitles = vkVideosContent.match(/"srclang":"en".*?"src":"(.*?)"/);
  1552. vkSubtitles = (vkSubtitles) ? vkSubtitles[1] : null;
  1553. if (vkSubtitles) vkVideoList['EN Subtitles WebVTT'] = vkSubtitles;
  1554. }
  1555.  
  1556. /* Create Saver */
  1557. if (vkVideoFound) {
  1558. var vkDefaultVideo = 'Low Definition MP4';
  1559. saver = {
  1560. 'videoList': vkVideoList,
  1561. 'videoDefinitions': ['Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
  1562. 'videoContainers': ['MP4', 'M3U8', 'Any'],
  1563. 'videoSave': vkDefaultVideo,
  1564. 'videoTitle': vkVideoTitle
  1565. };
  1566. createMySaver();
  1567. }
  1568. else {
  1569. if (vkUnauthorized) {
  1570. saver = {
  1571. 'warnMess': 'other',
  1572. 'warnContent': '<b>SaveTube:</b> Authorization required!'
  1573. };
  1574. }
  1575. else saver = {'warnMess': '!videos'};
  1576. createMySaver();
  1577. }
  1578. }
  1579. else {
  1580. saver = {'warnMess': '!content'};
  1581. createMySaver();
  1582. }
  1583.  
  1584. }
  1585.  
  1586. // =====IMDB===== //
  1587.  
  1588. else if (page.url.indexOf('imdb.com') != -1) {
  1589.  
  1590. /* Redirect To Video Page */
  1591. if (page.url.indexOf('/video/') == -1 && page.url.indexOf('/videoplayer/') == -1) {
  1592. page.doc.addEventListener('click', function(e) {
  1593. var p = e.target.parentNode;
  1594. while (p) {
  1595. if (p.tagName === 'A' && p.href.indexOf('/video/imdb') != -1) {
  1596. page.win.location.href = p.href.replace(/imdb\/inline.*/, '');
  1597. }
  1598. p = p.parentNode;
  1599. }
  1600. }, false);
  1601. return;
  1602. }
  1603.  
  1604. /* Get Video Title */
  1605. var imdbVideoTitle = getMyContent(page.url, 'meta\\s+property="og:title"\\s+content="(.*?)"', false);
  1606. if (imdbVideoTitle) imdbVideoTitle = cleanMyContent(imdbVideoTitle, false, true);
  1607.  
  1608. /* Get Data Key */
  1609. var imdbVideoId = page.url.replace(/^.*?\/(vi\d+).*/, '$1');
  1610. var imdbDataJSON = '{"type": "VIDEO_PLAYER", "subType": "FORCE_LEGACY", "id": "' + imdbVideoId + '"}';
  1611. var imdbDataKey = btoa(imdbDataJSON);
  1612.  
  1613. /* Get Videos Content */
  1614. var imdbVideosContent = getMyContent(page.url.replace(/video\/.*/, 've/data/VIDEO_PLAYBACK_DATA?key=' + imdbDataKey), '"videoLegacyEncodings":\\[(.*?)\\]', false);
  1615.  
  1616. /* Get Videos */
  1617. var imdbVideoList = {};
  1618. if (imdbVideosContent) {
  1619. var imdbVideoFormats = {'1080p': 'Full High Definition MP4', '720p': 'High Definition MP4', '480p': 'Standard Definition MP4',
  1620. '360p': 'Low Definition MP4', 'SD': 'Low Definition MP4', '240p': 'Very Low Definition MP4', 'AUTO': 'Multi Definition M3U8'};
  1621. var imdbVideoFound = false;
  1622. var imdbVideoParser, imdbVideoParse, myVideoCode, imdbVideo;
  1623. for (var imdbVideoCode in imdbVideoFormats) {
  1624. imdbVideoParser = '"definition":"' + imdbVideoCode + '".*?"url":"(.*?)"';
  1625. imdbVideoParse = imdbVideosContent.match(imdbVideoParser);
  1626. imdbVideo = (imdbVideoParse) ? imdbVideoParse[1] : null;
  1627. if (imdbVideo) {
  1628. imdbVideo = cleanMyContent(imdbVideo, false);
  1629. if (!imdbVideoFound) imdbVideoFound = true;
  1630. myVideoCode = imdbVideoFormats[imdbVideoCode];
  1631. if (!imdbVideoList[myVideoCode]) imdbVideoList[myVideoCode] = imdbVideo;
  1632. }
  1633. }
  1634.  
  1635. if (imdbVideoFound) {
  1636. /* Create Saver */
  1637. var imdbDefaultVideo = 'Low Definition MP4';
  1638. saver = {
  1639. 'videoList': imdbVideoList,
  1640. 'videoDefinitions': ['Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
  1641. 'videoContainers': ['MP4', 'M3U8', 'Any'],
  1642. 'videoSave': imdbDefaultVideo,
  1643. 'videoTitle': imdbVideoTitle
  1644. };
  1645. createMySaver();
  1646. }
  1647. else {
  1648. saver = {'warnMess': '!videos'};
  1649. createMySaver();
  1650. }
  1651. }
  1652. else {
  1653. saver = {'warnMess': '!content'};
  1654. createMySaver();
  1655. }
  1656.  
  1657. }
  1658.  
  1659. }
  1660.  
  1661.  
  1662. // ==========Run========== //
  1663.  
  1664. getMyOptions();
  1665. SaveTube();
  1666.  
  1667. page.win.setInterval(function() {
  1668. if (page.url != page.win.location.href) {
  1669. if (saver['saverPanel'] && saver['saverPanel'].parentNode) {
  1670. removeMyElement(saver['saverPanel'].parentNode, saver['saverPanel']);
  1671. }
  1672. page.doc = page.win.document;
  1673. page.body = page.doc.body;
  1674. page.url = page.win.location.href;
  1675. SaveTube();
  1676. }
  1677. }, 500);
  1678.  
  1679. })();