- // ==UserScript==
- // @name SaveTube
- // @version 2025.04.23
- // @description Download videos from video sharing web sites.
- // @author sebaro
- // @namespace http://sebaro.pro/savetube
- // @icon https://gitlab.com/sebaro/savetube/raw/master/savetube.png
- // @include http://youtube.com*
- // @include http://www.youtube.com*
- // @include https://youtube.com*
- // @include https://www.youtube.com*
- // @include http://m.youtube.com*
- // @include https://m.youtube.com*
- // @exclude http://youtube.com/shorts*
- // @exclude http://www.youtube.com/shorts*
- // @exclude https://youtube.com/shorts*
- // @exclude https://www.youtube.com/shorts*
- // @include http://dailymotion.com*
- // @include http://www.dailymotion.com*
- // @include https://dailymotion.com*
- // @include https://www.dailymotion.com*
- // @include http://vimeo.com*
- // @include http://www.vimeo.com*
- // @include https://vimeo.com*
- // @include https://www.vimeo.com*
- // @include http://imdb.com*
- // @include http://www.imdb.com*
- // @include https://imdb.com*
- // @include https://www.imdb.com*
- // @noframes
- // @grant none
- // @run-at document-end
- // ==/UserScript==
-
-
- /*
-
- Copyright (C) 2010 - 2025 Sebastian Luncan
-
- This program is free software: you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation, either version 3 of the License, or
- (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
-
- You should have received a copy of the GNU General Public License
- along with this program. If not, see <http://www.gnu.org/licenses/>.
-
- Website: http://sebaro.pro/savetube
- Contact: http://sebaro.pro/contact
-
- */
-
-
- (function() {
-
-
- // Don't run on frames or iframes
- if (window.top != window.self) return;
-
-
- // ==========Variables========== //
-
- // Userscript
- var userscript = 'SaveTube';
- var website = 'http://sebaro.pro/savetube';
- var contact = 'http://sebaro.pro/contact';
-
- // Page
- var page = {win: window, doc: window.document, body: window.document.body, url: window.location.href, site: window.location.hostname.match(/([^.]+)\.[^.]+$/)[1]};
-
- // Saver
- var saver = {};
- var panelHeight = 30;
-
- // Features/Options
- var feature = {'definition': true, 'container': true, 'openpagelink': true, 'autosave': true, 'savedash': false, 'showsavelink': true};
- var option = {'definition': 'High Definition', 'container': 'MP4', 'openpagelink': false, 'autosave': false, 'savedash': false, 'showsavelink': false, 'hidden': false};
-
- // Media
- var mediatypes = {'MP4': 'video/mp4', 'WebM': 'video/webm', 'M3U8': 'application/x-mpegURL', 'WebVTT': 'text/vtt'};
-
- // Sources
- var sources = {};
-
-
- // ==========Functions========== //
-
- function createMyElement(type, properties, event, listener) {
- var obj = page.doc.createElement(type);
- for (var propertykey in properties) {
- if (propertykey == 'target') obj.setAttribute('target', properties[propertykey]);
- else if (propertykey == 'innerHTML') {
- try {
- obj[propertykey] = properties[propertykey];
- }
- catch(e) {
- if (window.trustedTypes) {
- if (!window.trustedTypes.defaultPolicy) {
- if (window.trustedTypes.createPolicy) {
- window.trustedTypes.createPolicy('default', {
- createHTML: (string, sink) => string
- });
- }
- }
- if (window.trustedTypes.defaultPolicy) {
- obj[propertykey] = window.trustedTypes.defaultPolicy.createHTML(properties[propertykey]);
- }
- }
- }
- }
- else obj[propertykey] = properties[propertykey];
- }
- if (event && listener) {
- obj.addEventListener(event, listener, false);
- }
- return obj;
- }
-
- function modifyMyElement(obj, properties, event, listener) {
- for (var propertykey in properties) {
- if (propertykey == 'target') obj.setAttribute('target', properties[propertykey]);
- else if (propertykey == 'innerHTML') {
- try {
- obj[propertykey] = properties[propertykey];
- }
- catch(e) {
- if (window.trustedTypes) {
- if (!window.trustedTypes.defaultPolicy) {
- if (window.trustedTypes.createPolicy) {
- window.trustedTypes.createPolicy('default', {
- createHTML: (string, sink) => string
- });
- }
- }
- if (window.trustedTypes.defaultPolicy) {
- obj[propertykey] = window.trustedTypes.defaultPolicy.createHTML(properties[propertykey]);
- }
- }
- }
- }
- else obj[propertykey] = properties[propertykey];
- }
- if (event && listener) {
- obj.addEventListener(event, listener, false);
- }
- }
-
- function styleMyElement(obj, styles) {
- for (var stylekey in styles) {
- obj.style[stylekey] = styles[stylekey];
- }
- }
-
- function cleanMyElement(obj, hide) {
- if (hide) {
- for (var i = 0; i < obj.children.length; i++) {
- styleMyElement(obj.children[i], {display: 'none'});
- }
- }
- else {
- if (obj.hasChildNodes()) {
- while (obj.childNodes.length >= 1) {
- obj.removeChild(obj.firstChild);
- }
- }
- }
- }
-
- function getMyElement(obj, type, from, value, child, content) {
- var getObj, chObj, coObj;
- var pObj = (!obj) ? page.doc : obj;
- if (type == 'body') {
- getObj = pObj.body;
- }
- else {
- if (from == 'id') getObj = pObj.getElementById(value);
- else if (from == 'class') getObj = pObj.getElementsByClassName(value);
- else if (from == 'tag') getObj = pObj.getElementsByTagName(type);
- else if (from == 'ns') {
- if (pObj.getElementsByTagNameNS) getObj = pObj.getElementsByTagNameNS(value, type);
- }
- else if (from == 'query') {
- if (child > 0) {
- if (pObj.querySelectorAll) getObj = pObj.querySelectorAll(value);
- }
- else {
- if (pObj.querySelector) getObj = pObj.querySelector(value);
- }
- }
- }
- chObj = (getObj && child >= 0) ? getObj[child] : getObj;
- if (content && chObj) {
- if (type == 'html' || type == 'body' || type == 'div' || type == 'option') coObj = chObj.innerHTML;
- else if (type == 'object') coObj = chObj.data;
- else if (type == 'img' || type == 'video' || type == 'embed') coObj = chObj.src;
- else coObj = chObj.textContent;
- return coObj;
- }
- else {
- return chObj;
- }
- }
-
- function appendMyElement(parent, child) {
- parent.appendChild(child);
- }
-
- function removeMyElement(parent, child) {
- parent.removeChild(child);
- }
-
- function replaceMyElement(parent, orphan, child) {
- parent.replaceChild(orphan, child);
- }
-
- function cleanMyContent(content, unesc, extra) {
- if (unesc) content = unescape(content);
- content = content.replace(/\\u0025/g, '%');
- content = content.replace(/\\u0026/g, '&');
- content = content.replace(/\\u002F/g, '/');
- content = content.replace(/\\/g, '');
- content = content.replace(/\n/g, '');
- if (extra) {
- content = content.replace(/"/g, '\'').replace(/"/g, '\'').replace(/"/g, '\'').replace(/["“”„‘’]/g, '\'');
- content = content.replace(/'/g, '\'').replace(/'/g, '\'').replace(/'/g, '\'');
- content = content.replace(/&/g, 'and').replace(/&/g, 'and');
- //content = content.replace(/[^\x20-\xFF]/g, '');
- content = content.replace(/[\/\|]/g, '-');
- content = content.replace(/[<>#:\*\?]/g, '');
- content = content.replace(/^\s+|\s+$/, '').replace(/\s+/g, ' ').replace(/\.+$/g, '');
- }
- return content;
- }
-
- function parseMyContent(content, pattern) {
- var parse, response;
- content = content.replace(/(\r\n|\n|\r)/gm, '');
- parse = content.match(pattern);
- if (parse) {
- response = (/g$/.test(pattern)) ? parse : parse[1];
- }
- return response;
- }
-
- function getMyContent(url, pattern, data, headers) {
- var urle, xhr, response;
- if (data) {
- if (typeof data === 'object') data = JSON.stringify(data);
- urle = btoa(url + data + JSON.stringify(headers));
- if (!sources[urle]) {
- //console.log('SaveTube: POST\nURL: ' + url + '\nData: ' + data + '\nHeaders: ' + JSON.stringify(headers));
- xhr = new XMLHttpRequest();
- xhr.open('POST', url, false);
- if (data.indexOf('{') != -1) {
- xhr.setRequestHeader('Content-Type', 'application/json');
- }
- else {
- xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
- }
- if (headers) {
- if (headers['withCredentials']) {
- xhr.withCredentials = true;
- delete headers['withCredentials']
- }
- for (var header in headers) {
- xhr.setRequestHeader(header, headers[header]);
- }
- }
- try {
- xhr.send(data);
- }
- catch(e) {
- }
- sources[urle] = (xhr.responseText) ? xhr.responseText : xhr.responseXML;
- }
- }
- else {
- urle = btoa(url + JSON.stringify(headers));
- if (!sources[urle]) {
- //console.log('SaveTube: GET\nURL: ' + url + '\nHeaders: ' + JSON.stringify(headers));
- xhr = new XMLHttpRequest();
- xhr.open('GET', url, false);
- if (headers) {
- if (headers['withCredentials']) {
- xhr.withCredentials = true;
- delete headers['withCredentials']
- }
- for (var header in headers) {
- xhr.setRequestHeader(header, headers[header]);
- }
- }
- try {
- xhr.send();
- }
- catch(e) {
- }
- sources[urle] = (xhr.responseText) ? xhr.responseText : xhr.responseXML;
- }
- }
- if (sources[urle]) {
- response = sources[urle];
- if (pattern) {
- response = parseMyContent(response, pattern);
- }
- }
- return response;
- }
-
- function createMySaver() {
- /* The Panel */
- saver['saverPanel'] = createMyElement('div');
- 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'});
- appendMyElement(page.body, saver['saverPanel']);
-
- /* Panel Hide/Show Toggle Button */
- saver['buttonHide'] = createMyElement('div', {title: '{Hide/Show: click to hide/show this panel}'}, 'click', function() {
- toggleMySaver();
- });
- 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'});
- appendMyElement(saver['saverPanel'], saver['buttonHide']);
-
- /* Panel Logo */
- saver['panelLogo'] = createMyElement('div', {title: '{SaveTube: click to visit the script wesite}', textContent: userscript}, 'click', function() {
- page.win.location.href = website;
- });
- 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'});
- appendMyElement(saver['saverPanel'], saver['panelLogo']);
-
- /* Warnings */
- if (saver['warnMess']) {
- saver['saverMessage'] = createMyElement('div');
- styleMyElement(saver['saverMessage'], {display: 'inline-block', fontSize: '12px', color: '#AD0000'});
- appendMyElement(saver['saverPanel'], saver['saverMessage']);
- if (saver['warnContent']) showMyMessage(saver['warnMess'], saver['warnContent']);
- else showMyMessage(saver['warnMess']);
- return;
- }
-
- /* Panel Video Menu */
- saver['videoMenu'] = createMyElement('select', {title: '{Videos: select the video format for download}'}, 'change', function() {
- saver['videoSave'] = this.value;
- if (saver['isShowingLink']) {
- cleanMyElement(saver['buttonSaveLink'], false);
- saver['isShowingLink'] = false;
- }
- if (option['autosave']) {
- saveMyVideo();
- }
- });
- 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'});
- appendMyElement(saver['saverPanel'], saver['videoMenu']);
- if (feature['openpagelink']) {
- saver['videoList']['Page Link'] = page.url;
- }
- var videosProgressive = [];
- var videosAdaptiveHLS = [];
- var videosAdaptiveDASHVideo = [];
- var videosAdaptiveDASHAudio = [];
- var videosAdaptiveDASHMuxed = [];
- var videosExtra = [];
- for (var videoCode in saver['videoList']) {
- if (videoCode.indexOf('Video') != -1) videosAdaptiveDASHVideo.push(videoCode);
- else if (videoCode.indexOf('Audio') != -1) videosAdaptiveDASHAudio.push(videoCode);
- else if (saver['videoList'][videoCode] == 'DASH') videosAdaptiveDASHMuxed.push(videoCode);
- else if (videoCode.indexOf('M3U8') != -1) videosAdaptiveHLS.push(videoCode);
- else if (videoCode.indexOf('MP4') != -1 || videoCode.indexOf('WebM') != -1) videosProgressive.push(videoCode);
- else videosExtra.push(videoCode);
- }
- if (videosProgressive.length > 0) {
- for (var i = 0; i < videosProgressive.length; i++) {
- saver['videoItem'] = createMyElement('option', {value: videosProgressive[i], textContent: videosProgressive[i]});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- }
- }
- if (videosAdaptiveHLS.length > 0) {
- saver['videoItem'] = createMyElement('option', {value: 'HLS', textContent: 'HLS'});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
- saver['videoItem'].disabled = 'disabled';
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- for (var i = 0; i < videosAdaptiveHLS.length; i++) {
- saver['videoItem'] = createMyElement('option', {value: videosAdaptiveHLS[i], textContent: videosAdaptiveHLS[i]});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- }
- }
- if (videosAdaptiveDASHVideo.length > 0) {
- saver['videoItem'] = createMyElement('option', {value: 'DASH (Video Only)', textContent: 'DASH (Video Only)'});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
- saver['videoItem'].disabled = 'disabled';
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- for (var i = 0; i < videosAdaptiveDASHVideo.length; i++) {
- saver['videoItem'] = createMyElement('option', {value: videosAdaptiveDASHVideo[i], textContent: videosAdaptiveDASHVideo[i]});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- }
- }
- if (videosAdaptiveDASHAudio.length > 0) {
- saver['videoItem'] = createMyElement('option', {value: 'DASH (Audio Only)', textContent: 'DASH (Audio Only)'});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
- saver['videoItem'].disabled = 'disabled';
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- for (var i = 0; i < videosAdaptiveDASHAudio.length; i++) {
- saver['videoItem'] = createMyElement('option', {value: videosAdaptiveDASHAudio[i], textContent: videosAdaptiveDASHAudio[i]});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- }
- }
- if (videosAdaptiveDASHMuxed.length > 0) {
- feature['savedash'] = true;
- if (option['savedash']) {
- saver['videoItem'] = createMyElement('option', {value: 'DASH (Video With Audio)', textContent: 'DASH (Video With Audio)'});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
- saver['videoItem'].disabled = 'disabled';
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- for (var i = 0; i < videosAdaptiveDASHMuxed.length; i++) {
- saver['videoItem'] = createMyElement('option', {value: videosAdaptiveDASHMuxed[i], textContent: videosAdaptiveDASHMuxed[i]});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- }
- }
- else {
- for (var videoCode in saver['videoList']) {
- if (saver['videoList'][videoCode] == 'DASH') delete saver['videoList'][videoCode];
- }
- }
- }
- if (videosExtra.length > 0) {
- saver['videoItem'] = createMyElement('option', {value: 'Extra', textContent: 'Extra'});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', color: '#FF0000'});
- saver['videoItem'].disabled = 'disabled';
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- for (var i = 0; i < videosExtra.length; i++) {
- saver['videoItem'] = createMyElement('option', {value: videosExtra[i], textContent: videosExtra[i]});
- styleMyElement(saver['videoItem'], {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
- appendMyElement(saver['videoMenu'], saver['videoItem']);
- }
- }
-
- /* Panel Options Button */
- saver['buttonOptions'] = createMyElement('div', {title: '{Options: click to show the available options}'}, 'click', function() {
- if (saver['showsOptions']) {
- saver['showsOptions'] = false;
- styleMyElement(saver['optionsContent'], {display: 'none'})
- }
- else {
- saver['showsOptions'] = true;
- styleMyElement(saver['optionsContent'], {display: 'block'})
- }
- });
- styleMyElement(saver['buttonOptions'], {width: '1px', height: '8px', borderTop: '4px dotted #777777', borderBottom: '4px dotted #777777', borderLeft: '4px dotted #777777', display: 'inline-block', lineHeight: 'normal', verticalAlign: 'middle', marginLeft: '20px', cursor: 'pointer', boxSizing: 'content-box'});
- appendMyElement(saver['saverPanel'], saver['buttonOptions']);
-
- /* Panel Save Button */
- saver['buttonSave'] = createMyElement('div', {title: '{Save: click to download the selected video format}'}, 'click', function() {
- saveMyVideo();
- });
- 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: '1px', marginLeft: '20px', cursor: 'pointer', boxSizing: 'content-box'});
- appendMyElement(saver['saverPanel'], saver['buttonSave']);
-
- /* Panel Save Button Link */
- saver['buttonSaveLink'] = createMyElement('div', {title: '{Save: right click & save as to download the selected video format}'});
- styleMyElement(saver['buttonSaveLink'], {display: 'inline-block', color: '#777777', fontSize: '14px', fontWeight: 'bold', lineHeight: 'normal', verticalAlign: 'middle', marginLeft: '5px', marginBottom: '2px', boxSizing: 'content-box'});
- appendMyElement(saver['saverPanel'], saver['buttonSaveLink']);
-
- /* Disable Features */
- if (saver['videoDefinitions'].length < 2) feature['definition'] = false;
- if (saver['videoContainers'].length < 2) feature['container'] = false;
-
- /* Select The Video */
- if (feature['definition'] || feature['container'] || feature['openpagelink']) {
- if (!option['definition'] || saver['videoDefinitions'].indexOf(option['definition']) == -1) option['definition'] = saver['videoSave'].replace(/Definition.*/, 'Definition');
- if (!option['container'] || saver['videoContainers'].indexOf(option['container']) == -1) option['container'] = saver['videoSave'].replace(/.*\s/, '');
- selectMyVideo();
- }
-
- /* Save The Video On Autosave */
- if (option['autosave']) saveMyVideo();
-
- /* Panel Options */
- saver['optionsContent'] = createMyElement('div');
- styleMyElement(saver['optionsContent'], {display: 'none', fontSize: '14px', fontWeight: 'bold', padding: '10px', textAlign: 'center', boxSizing: 'content-box'});
- appendMyElement(saver['saverPanel'], saver['optionsContent']);
-
- /* Options Object => option: [label, options, new line, change video] */
- var options = {
- 'definition': ['Definition', saver['videoDefinitions'], true, true],
- 'container': ['Container', saver['videoContainers'], false, true],
- 'openpagelink': ['Open Page Link', ['On', 'Off'], true, true],
- 'autosave': ['Autosave', ['On', 'Off'], true, true],
- 'showsavelink': ['Show Save Link', ['On', 'Off'], false, true],
- 'savedash': ['Save DASH (Video With Audio)', ['On', 'Off'], true, false]
- };
-
- /* Options */
- var optionsBox, optionBox, optionLabel, optionMenu, optionMenuItem;
- for (var o in options) {
- if (feature[o] === false) continue;
- if (options[o][2]) {
- optionsBox = createMyElement('div');
- styleMyElement(optionsBox, {display: 'block', padding: '5px 0px 5px 0px'});
- appendMyElement(saver['optionsContent'], optionsBox);
- }
- optionBox = createMyElement('div');
- styleMyElement(optionBox, {display: 'inline-block'});
- optionLabel = createMyElement('div', {textContent: options[o][0]});
- styleMyElement(optionLabel, {display: 'inline-block', color: '#777777', marginRight: '10px', verticalAlign: 'middle'});
- optionMenu = createMyElement('select', {id: 'savetube-option-' + o}, 'change', function() {
- var id = this.id.replace('savetube-option-', '');
- if (this.value == 'On' || this.value == 'Off') {
- option[id] = (this.value == 'On') ? true : false;
- }
- else {
- option[id] = this.value;
- }
- setMyOptions(id, option[id]);
- if (options[id][3]) {
- if (saver['isShowingLink']) {
- cleanMyElement(saver['buttonSaveLink'], false);
- saver['isShowingLink'] = false;
- }
- selectMyVideo();
- if (option['autosave']) {
- saveMyVideo();
- }
- }
- });
- 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'});
- appendMyElement(optionBox, optionLabel);
- appendMyElement(optionBox, optionMenu);
- appendMyElement(optionsBox, optionBox);
- for (var i = 0; i < options[o][1].length; i++) {
- optionMenuItem = createMyElement('option', {value: options[o][1][i], textContent: options[o][1][i]});
- styleMyElement(optionMenuItem, {fontSize: '14px', fontWeight: 'bold', cursor: 'pointer'});
- appendMyElement(optionMenu, optionMenuItem);
- }
- if (optionMenu.value == 'On' || optionMenu.value == 'Off') {
- if (option[o]) optionMenu.value = 'On';
- else optionMenu.value = 'Off';
- }
- else {
- optionMenu.value = option[o];
- }
- }
-
- /* Hide */
- if (option['hidden']) {
- toggleMySaver('hide');
- }
- }
-
- function toggleMySaver(toggle) {
- if (toggle == 'hide') {
- styleMyElement(saver['saverPanel'], {right: '-' + (saver['saverPanel'].offsetWidth - 40) + 'px', backgroundColor: 'transparent', borderColor: '#777777'});
- styleMyElement(saver['buttonHide'], {borderTop: '8px solid transparent', borderBottom: '8px solid transparent', borderLeft: '0px solid #777777', borderRight: '15px solid #777777'});
- }
- else {
- if (option['hidden']) {
- styleMyElement(saver['saverPanel'], {right: '25px', backgroundColor: '#FFFFFF', borderColor: '#CCCCCC', transition: 'right 2s, background-color 5s, border-color 5s'});
- styleMyElement(saver['buttonHide'], {borderTop: '8px solid transparent', borderBottom: '8px solid transparent', borderLeft: '15px solid #777777', borderRight: '0px solid #777777'});
- option['hidden'] = false;
- }
- else {
- styleMyElement(saver['saverPanel'], {right: '-' + (saver['saverPanel'].offsetWidth - 40) + 'px', backgroundColor: 'transparent', borderColor: '#777777', transition: 'right 2s, background-color 5s, border-color 5s'});
- styleMyElement(saver['buttonHide'], {borderTop: '8px solid transparent', borderBottom: '8px solid transparent', borderLeft: '0px solid #777777', borderRight: '15px solid #777777'});
- option['hidden'] = true;
- }
- setMyOptions('hidden', option['hidden']);
- }
- }
-
- function setMyOptions(key, value) {
- key = page.site + '_' + userscript.toLowerCase() + '_' + key;
- try {
- localStorage.setItem(key, value);
- if (localStorage.getItem(key) == value) return;
- else throw false;
- }
- catch(e) {
- var date = new Date();
- date.setTime(date.getTime() + (356*24*60*60*1000));
- var expires = '; expires=' + date.toGMTString();
- page.doc.cookie = key + '=' + value + expires + '; path=/';
- }
- }
-
- function getMyOptions() {
- for (var opt in option) {
- var key = page.site + '_' + userscript.toLowerCase() + '_' + opt;
- try {
- if (localStorage.getItem(key)) {
- option[opt] = localStorage.getItem(key);
- continue;
- }
- else throw false;
- }
- catch(e) {
- var cookies = page.doc.cookie.split(';');
- for (var i=0; i < cookies.length; i++) {
- var cookie = cookies[i];
- while (cookie.charAt(0) == ' ') cookie = cookie.substring(1, cookie.length);
- option[opt] = (cookie.indexOf(key) == 0) ? cookie.substring(key.length + 1, cookie.length) : option[opt];
- }
- }
- }
- var boolOptions = ['openpagelink', 'autosave', 'showsavelink', 'savedash', 'hidden'];
- for (var i = 0; i < boolOptions.length; i++) {
- option[boolOptions[i]] = (option[boolOptions[i]] === true || option[boolOptions[i]] == 'true') ? true : false;
- }
- }
-
- function selectMyVideo() {
- if (option['openpagelink']) {
- saver['videoSave'] = 'Page Link';
- }
- else {
- var vdoCont = (option['container'] != 'Any') ? [option['container']] : saver['videoContainers'];
- var vdoDef = saver['videoDefinitions'];
- var vdoList = {};
- for (var vC = 0; vC < vdoCont.length; vC++) {
- if (vdoCont[vC] != 'Any') {
- for (var vD = 0; vD < vdoDef.length; vD++) {
- var format = vdoDef[vD] + ' ' + vdoCont[vC];
- if (!vdoList[vdoDef[vD]]) {
- for (var vL in saver['videoList']) {
- if (vL == format) {
- vdoList[vdoDef[vD]] = vL;
- break;
- }
- }
- }
- }
- }
- }
- var vdoDef2 = [];
- var keepDef = false;
- for (var vD = 0; vD < vdoDef.length; vD++) {
- if (vdoDef[vD] == option['definition'] && keepDef == false) keepDef = true;
- if (keepDef == true) vdoDef2.push(vdoDef[vD])
- }
- for (var vD = 0; vD < vdoDef2.length; vD++) {
- if (vdoList[vdoDef2[vD]]) {
- saver['videoSave'] = vdoList[vdoDef2[vD]];
- break;
- }
- }
- }
- saver['videoMenu'].value = saver['videoSave'];
- }
-
- function saveMyVideo() {
- var vdoURL = saver['videoList'][saver['videoSave']];
- var vdoDef = ' (' + saver['videoSave'].split(' ').slice(0, -1).join('').match(/[A-Z]/g).join('') + ')';
- var vdoExt = saver['videoSave'].split(' ').slice(-1).join('');
- var vdoTle = (saver['videoTitle']) ? saver['videoTitle'] + vdoDef : page.url.replace(/https?:\/\//, '').replace(/[^0-9a-zA-Z]/g, '-') + vdoDef;
- if (saver['videoSave'] == 'Page Link' || vdoURL == 'DASH' || (vdoExt == 'M3U8' && !option['showsavelink'])) {
- var vdoVdo, vdoAdo;
- if (saver['videoSave'] == 'Page Link' || vdoExt == 'M3U8') {
- vdoVdo = vdoURL;
- vdoAdo = '';
- vdoDef = '';
- }
- else {
- if (saver['videoSave'].indexOf('MP4') != -1) {
- vdoVdo = saver['videoList'][saver['videoSave'].replace('MP4', 'Video MP4')];
- vdoAdo = saver['videoList']['Medium Bitrate Audio MP4'] || saver['videoList'][saver['videoSave'].replace('MP4', 'Audio MP4')];
- }
- else {
- vdoVdo = saver['videoList'][saver['videoSave'].replace('WebM', 'Video WebM')];
- vdoAdo = saver['videoList']['High Bitrate Audio WebM'] || saver['videoList']['Medium Bitrate Audio WebM'] || saver['videoList']['Medium Bitrate Audio MP4'];
- }
- }
- page.win.location.href = 'savetube:' + vdoTle + 'SEPARATOR' + vdoVdo + 'SEPARATOR' + vdoAdo;
- }
- else {
- if (page.site == 'youtube' && saver['videoSave'] == 'High Definition MP4') {
- vdoURL = vdoURL + '&title=' + vdoTle;
- }
- var vdoLnk = createMyElement('a', {href: vdoURL, target: '_blank', textContent: '[Link]'});
- styleMyElement(vdoLnk, {color: '#777777', textDecoration: 'underline'});
- if (option['showsavelink'] || vdoExt == 'M3U8') {
- if (!saver['isShowingLink']) {
- appendMyElement(saver['buttonSaveLink'], vdoLnk);
- saver['isShowingLink'] = true;
- if (page.site == 'youtube' && saver['videoSave'] == 'High Definition MP4') {
- page.win.location.href = vdoURL;
- }
- }
- }
- else {
- if (!saver['isSaving']) {
- if (page.site == 'youtube' && saver['videoSave'] == 'High Definition MP4') {
- page.win.location.href = vdoURL;
- }
- else {
- if (page.win.URL && page.win.URL.createObjectURL) {
- saver['isSaving'] = true;
- styleMyElement(saver['buttonSave'], {borderBottomWidth: '1px', cursor: 'none'});
- var vdoLnkBlob, vdoBlob, vdoBlobLnk;
- vdoLnkBlob = createMyElement('a');
- styleMyElement(vdoLnkBlob, {display: 'none'});
- appendMyElement(page.body, vdoLnkBlob);
- var XHRequest = new XMLHttpRequest();
- XHRequest.open('GET', vdoURL);
- XHRequest.responseType = 'arraybuffer';
- XHRequest.onload = function() {
- if (this.status === 200 && this.response) {
- vdoBlob = new Blob([this.response], {type: mediatypes[vdoExt]});
- vdoBlobLnk = page.win.URL.createObjectURL(vdoBlob);
- modifyMyElement(vdoLnkBlob, {href: vdoBlobLnk, target: '_blank', download: vdoTle + '.' + vdoExt.toLowerCase()});
- vdoLnkBlob.click();
- page.win.URL.revokeObjectURL(vdoBlobLnk);
- removeMyElement(page.body, vdoLnkBlob);
- saver['isSaving'] = false;
- styleMyElement(saver['buttonSave'], {borderBottomWidth: '0px', cursor: 'pointer'});
- }
- else {
- saver['isSaving'] = false;
- styleMyElement(saver['buttonSave'], {borderBottomWidth: '0px', cursor: 'pointer'});
- if (!saver['isShowingLink']) {
- appendMyElement(saver['buttonSaveLink'], vdoLnk);
- saver['isShowingLink'] = true;
- }
- }
- }
- XHRequest.onerror = function() {
- saver['isSaving'] = false;
- styleMyElement(saver['buttonSave'], {borderBottomWidth: '0px', cursor: 'pointer'});
- if (!saver['isShowingLink']) {
- appendMyElement(saver['buttonSaveLink'], vdoLnk);
- saver['isShowingLink'] = true;
- }
- }
- XHRequest.send();
- }
- else {
- appendMyElement(saver['buttonSaveLink'], vdoLnk);
- saver['isShowingLink'] = true;
- }
- }
- }
- }
- }
- }
-
- function showMyMessage(cause, content) {
- if (cause == '!content') {
- modifyMyElement(saver['saverMessage'], {innerHTML: 'Couldn\'t get the videos content. Please report it <a href="' + contact + '" style="color:#00892C">here</a>.'});
- }
- else if (cause == '!videos') {
- modifyMyElement(saver['saverMessage'], {innerHTML: 'Couldn\'t get any video. Please report it <a href="' + contact + '" style="color:#00892C">here</a>.'});
- }
- else if (cause == '!support') {
- modifyMyElement(saver['saverMessage'], {innerHTML: 'This video uses the RTMP protocol which is not supported.'});
- }
- else if (cause == 'embed') {
- modifyMyElement(saver['saverMessage'], {innerHTML: 'This is an embedded video. You can get it <a href="' + content + '" style="color:#00892C">here</a>.'});
- }
- else if (cause == 'other') {
- modifyMyElement(saver['saverMessage'], {innerHTML: content});
- }
- }
-
-
- // ==========Websites========== //
-
- function SaveTube() {
-
- // =====YouTube===== //
-
- if (page.url.indexOf('youtube.com/watch') != -1) {
-
- /* Video Availability */
- if (getMyContent(page.url, /"playabilityStatus":\{"status":"(ERROR|UNPLAYABLE|LIVE_STREAM_OFFLINE|LOGIN_REQUIRED)"/)) return;
-
- /* Get Video ID */
- var ytVideoId = parseMyContent(page.url, /(?:\?|&)v=(.*?)(&|$)/);
-
- /* Create Saver */
- var ytVideoTitle, ytVideoAuthor;
- var ytVideoList = {};
- var ytDefaultVideo = 'Low Definition MP4';
- function ytCreateSaver(data) {
- if (data) {
- saver = data;
- }
- else {
- saver = {
- 'videoList': ytVideoList,
- 'videoDefinitions': ['Ultra High Definition', 'Quad High Definition', 'Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
- 'videoContainers': ['MP4', 'WebM', 'M3U8', 'Any'],
- 'videoSave': ytDefaultVideo,
- 'videoTitle': ytVideoTitle
- };
- }
- createMySaver();
- }
-
- /* Script */
- var ytScriptUrl;
- function ytGetScriptUrl() {
- if (!ytScriptUrl) {
- ytScriptUrl = getMyContent(page.url, /"js(?:Url)?":\s*"(.*?)"/);
- if (!ytScriptUrl) {
- ytScriptUrl = getMyContent(page.url.replace(/watch.*?v=/, 'embed/').replace(/&.*$/, ''), /"js(?:Url)?":\s*"(.*?)"/);
- }
- if (ytScriptUrl && ytScriptUrl.indexOf('//') == -1) {
- ytScriptUrl = page.win.location.protocol + '//' + page.win.location.hostname + ytScriptUrl;
- }
- if (!ytScriptUrl) {
- showMyMessage('other', 'Couldn\'t get the script link. Please report it <a href="' + contact + '" style="color:#00892C">here</a>.');
- }
- }
- }
-
- /* Parameter Unscrambler */
- var ytUnscrambleParam = {};
- function ytGetUnscrambleParamFunc() {
- ytGetScriptUrl();
- var ytMainFuncName, ytMainFuncBody, ytExtraFuncName, ytExtraFuncBody;
- /* s */
- ytMainFuncName = getMyContent(ytScriptUrl, /[\w$]+&&\([\w$]+=([\w$]+)\(decodeURIComponent/);
- if (ytMainFuncName) {
- ytMainFuncBody = getMyContent(ytScriptUrl, new RegExp('(?:^|;)' + ytMainFuncName.replace(/\$/, '\\$') + '\\s*=\\s*function\\s*' + '\\s*\\(\\w+\\)\\s*\\{(.*?\\))\\};'));
- if (ytMainFuncBody) {
- ytExtraFuncName = parseMyContent(ytMainFuncBody, /;([\w$]+)[\.|\[]/);
- if (ytExtraFuncName) {
- ytExtraFuncBody = getMyContent(ytScriptUrl, new RegExp('var\\s+' + ytExtraFuncName.replace(/\$/, '\\$') + '=\\s*\\{(.*?)\\};'));
- if (ytExtraFuncBody) {
- ytMainFuncBody = 'var ' + ytExtraFuncName + '={' + ytExtraFuncBody + '};' + ytMainFuncBody;
- ytExtraFuncBody = getMyContent(ytScriptUrl, /use strict';(var.*?[\)|\]]),/);
- if (ytExtraFuncBody) {
- ytMainFuncBody = 'try {' + ytExtraFuncBody + ';' + ytMainFuncBody + '} catch(e) {return null}';
- ytUnscrambleParam['s'] = new Function(ytMainFuncBody.replace(/.*;return\s+(\w).*/, '$1'), ytMainFuncBody);
- }
- }
- }
- }
- }
- /* n */
- ytMainFuncName = getMyContent(ytScriptUrl, /(?:^|;)([\w$]+)=function\([\w$]+\)\s*\{var\s+\w=\w\.split\(\w\.slice/);
- if (!ytMainFuncName) {
- ytMainFuncName = getMyContent(ytScriptUrl, /(?:^|;)([\w$]+)=function\([\w$]+\)\s*\{var\s+\w=\w\[\w\[\d+\]\]\(\w\[\d+\]\)/);
- }
- if (ytMainFuncName) {
- ytMainFuncBody = getMyContent(ytScriptUrl, new RegExp('(?:^|;)' + ytMainFuncName.replace(/\$/, '\\$') + '\\s*=\\s*function\\s*' + '\\s*\\(\\w+\\)\\s*\\{(.*?\\))\\};'));
- if (ytMainFuncBody) {
- ytMainFuncBody = ytMainFuncBody.replace(/(\d+)?--(\d+)/, '$1- -$2').replace(/if\(typeof.*?;/, '');
- ytExtraFuncBody = getMyContent(ytScriptUrl, /use strict';(var.*?[\)|\]]),/);
- if (ytExtraFuncBody) {
- ytMainFuncBody = 'try {' + ytExtraFuncBody + ';' + ytMainFuncBody + '} catch(e) {return null}';
- ytUnscrambleParam['n'] = new Function(ytMainFuncBody.replace(/.*\+(\w)\}return.*/, '$1'), ytMainFuncBody);
- }
- }
- }
- }
-
- /* Get Videos Content */
- var ytVideosContent = {};
- var ytVideosContentHLS;
- var ytVideoInfoUrl = page.win.location.protocol + '//' + page.win.location.hostname + '/youtubei/v1/player?prettyPrint=false';
- var ytVideoInfoClients = {
- 'MWEB': {
- 'clientName': 'MWEB',
- 'clientVersion': '2.20250311.03.00',
- 'userAgent': 'Mozilla/5.0 (iPad; CPU OS 16_7_10 like Mac OS X) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.6 Mobile/15E148 Safari/604.1,gzip(gfe)'
- },
- 'WEB_SAFARI': {
- 'clientName': 'WEB',
- 'clientVersion': '2.20250312.04.00',
- 'userAgent': 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.5 Safari/605.1.15,gzip(gfe)'
- },
- 'IOS': {
- 'clientName': 'IOS',
- 'clientVersion': '20.10.4',
- 'userAgent': 'com.google.ios.youtube/20.10.4 (iPhone16,2; U; CPU iOS 18_3_2 like Mac OS X;)'
- },
- 'TV': {
- 'clientName': 'TVHTML5',
- 'clientVersion': '7.20250312.16.00',
- 'userAgent': 'Mozilla/5.0 (ChromiumStylePlatform) Cobalt/Version'
- },
- 'TV_EMBEDDED': {
- 'clientName': 'TVHTML5_SIMPLY_EMBEDDED_PLAYER',
- 'clientVersion': '2.0'
- }
- };
- var ytVideoInfoData = {'videoId': ytVideoId, 'context': {'client': null}, 'playbackContext': {'contentPlaybackContext': {'html5Preference': 'HTML5_PREF_WANTS'}}};
- var ytVideoInfoExtra = {'signatureTimestamp': 20068, 'visitorData': ''};
- var ytVideoInfoHeaders;
- function ytGetVideos(api, client, embed) {
- if (api) {
- ytVideoInfoData['context']['client'] = ytVideoInfoClients[client];
- if (embed) {
- ytVideoInfoData['context']['client']['clientScreen'] = 'EMBED';
- ytVideoInfoData['context']['thirdParty'] = {};
- ytVideoInfoData['context']['thirdParty']['embedUrl'] = 'https://www.youtube.com';
- }
- ytGetScriptUrl();
- ytVideoInfoExtra['signatureTimestamp'] = getMyContent(ytScriptUrl, /signatureTimestamp:(\d+)/);
- if (ytVideoInfoExtra['signatureTimestamp']) {
- ytVideoInfoData['playbackContext']['contentPlaybackContext']['signatureTimestamp'] = ytVideoInfoExtra['signatureTimestamp'];
- }
- ytVideoInfoExtra['visitorData'] = getMyContent(page.url, /"visitorData":\s*"(.*?)"/);
- if (ytVideoInfoClients[client]['userAgent']) {
- if (!ytVideoInfoHeaders) ytVideoInfoHeaders = {};
- ytVideoInfoHeaders['User-Agent'] = ytVideoInfoClients[client]['userAgent'];
- }
- if (ytVideoInfoExtra['visitorData']) {
- if (!ytVideoInfoHeaders) ytVideoInfoHeaders = {};
- ytVideoInfoHeaders['X-Goog-Visitor-Id'] = ytVideoInfoExtra['visitorData'];
- }
- ytVideosContent = getMyContent(ytVideoInfoUrl, null, ytVideoInfoData, ytVideoInfoHeaders && ytVideoInfoHeaders);
- }
- else {
- ytVideosContent = getMyContent(page.url, /ytInitialPlayerResponse\s*=\s*({.*?});/);
- }
- try {
- ytVideosContent = JSON.parse(ytVideosContent);
- if (!ytVideosContentHLS) ytVideosContentHLS = ytVideosContent['streamingData']['hlsManifestUrl'];
- if (ytVideosContent['videoDetails'] && ytVideosContent['videoDetails']['videoId'] != ytVideoId) {
- ytVideosContent = {};
- }
- if (ytVideosContent['videoDetails']) {
- ytVideoTitle = ytVideosContent['videoDetails']['title'];
- ytVideoAuthor = ytVideosContent['videoDetails']['author'];
- ytVideoTitle = ytVideoTitle + ' by ' + ytVideoAuthor;
- ytVideoTitle = cleanMyContent(ytVideoTitle, false, true);
- if (ytVideosContent['captions'] && ytVideosContent['captions']['playerCaptionsTracklistRenderer']) {
- ytSubtitlesContent = ytVideosContent['captions']['playerCaptionsTracklistRenderer']['captionTracks'];
- }
- }
- }
- catch(e) {
- ytVideosContent = {};
- }
- ytVideosContent = (ytVideosContent['streamingData']) ? ytVideosContent['streamingData'] : {};
- }
-
- /* Get Videos */
- ytGetVideos(true, 'TV', false);
- if (!ytVideosContent['formats']) {
- ytGetVideos(true, 'MWEB', false);
- }
- if (!ytVideosContent['formats']) {
- ytGetVideos(true, 'TV_EMBEDDED', true);
- }
- if (ytVideosContent['formats']) {
- var ytVideoFormats = {
- '18': 'Low Definition MP4',
- '22': 'High Definition MP4',
- '43': 'Low Definition WebM',
- '133': 'Very Low Definition Video MP4',
- '134': 'Low Definition Video MP4',
- '135': 'Standard Definition Video MP4',
- '136': 'High Definition Video MP4',
- '137': 'Full High Definition Video MP4',
- '140': 'Medium Bitrate Audio MP4',
- '242': 'Very Low Definition Video WebM',
- '243': 'Low Definition Video WebM',
- '244': 'Standard Definition Video WebM',
- '247': 'High Definition Video WebM',
- '248': 'Full High Definition Video WebM',
- '249': 'Low Bitrate Audio WebM',
- '250': 'Medium Bitrate Audio WebM',
- '251': 'High Bitrate Audio WebM',
- '264': 'Quad High Definition Video MP4',
- '271': 'Quad High Definition Video WebM',
- '272': 'Ultra High Definition Video WebM',
- '298': 'High Definition Video MP4',
- '299': 'Full High Definition Video MP4',
- '302': 'High Definition Video WebM',
- '303': 'Full High Definition Video WebM',
- '308': 'Quad High Definition Video WebM',
- '313': 'Ultra High Definition Video WebM',
- '315': 'Ultra High Definition Video WebM',
- '333': 'Standard Definition Video WebM',
- '334': 'High Definition Video WebM',
- '335': 'Full High Definition Video WebM',
- '337': 'Ultra High Definition Video WebM'
- };
- var ytVideoFound = false;
- var ytVideos = (ytVideosContent['adaptiveFormats']) ? ytVideosContent['formats'].concat(ytVideosContent['adaptiveFormats']) : ytVideosContent['formats']
- var ytVideoParse, ytVideoCodeParse, ytVideoCode, myVideoCode, ytVideo, ytSParam, ytSParamName, ytNParam;
- if (ytVideos[0]['signatureCipher'] || ytVideos[0]['cipher'] || (ytVideos[0]['url'] && parseMyContent(ytVideos[0]['url'], /(?:&|&)n=(.*?)(&|&|$)/))) {
- ytGetUnscrambleParamFunc();
- }
- for (var i = 0; i < ytVideos.length; i++) {
- if (ytVideos[i]['signatureCipher'] || ytVideos[i]['cipher']) {
- ytVideo = ytVideos[i]['signatureCipher'] || ytVideos[i]['cipher'];
- ytVideo = cleanMyContent(ytVideo, true);
- ytVideoParse = ytVideo.match(/(.*)(url=.*$)/);
- if (ytVideoParse) {
- ytVideo = ytVideoParse[2] + '&' + ytVideoParse[1];
- ytVideo = ytVideo.replace(/url=/, '').replace(/&$/, '');
- }
- ytSParam = parseMyContent(ytVideo, /&s=(.*?)(&|$)/);
- if (ytSParam && ytUnscrambleParam['s']) {
- ytSParam = ytUnscrambleParam['s'](ytSParam);
- if (ytSParam) {
- ytSParamName = parseMyContent(ytVideo, /&sp=(.*?)(&|$)/);
- ytSParamName = (ytSParamName) ? ytSParamName : ((/&lsig=/.test(ytVideo)) ? 'sig' : 'signature');
- ytVideo = ytVideo.replace(/&s=.*?(&|$)/, '&' + ytSParamName + '=' + ytSParam + '$1');
- }
- else ytVideo = '';
- }
- else ytVideo = '';
- }
- else {
- ytVideo = ytVideos[i]['url'];
- ytVideo = cleanMyContent(ytVideo, true);
- if (/&sig=/.test(ytVideo) && !/&lsig=/.test(ytVideo)) {
- ytVideo = ytVideo.replace(/&sig=/, '&signature=');
- }
- }
- ytVideoCode = ytVideos[i]['itag'];
- if (!ytVideoCode) continue;
- myVideoCode = ytVideoFormats[ytVideoCode];
- if (!myVideoCode) continue;
- if (myVideoCode.indexOf('Video') != -1) {
- if (ytVideo.indexOf('source=yt_otf') != -1) continue;
- }
- ytVideo = cleanMyContent(ytVideo, true);
- ytNParam = parseMyContent(ytVideo, /&n=(.*?)(&|$)/);
- if (ytNParam && ytUnscrambleParam['n']) {
- ytNParam = ytUnscrambleParam['n'](ytNParam);
- if (ytNParam) {
- ytVideo = ytVideo.replace(/&n=.*?(&|$)/, '&n=' + ytNParam + '$1');
- }
- }
- if (ytVideo.indexOf('ratebypass') == -1) ytVideo += '&ratebypass=yes';
- if (ytVideo && ytVideo.indexOf('http') == 0) {
- if (!ytVideoFound) ytVideoFound = true;
- ytVideoList[myVideoCode] = ytVideo;
- }
- }
- if (ytVideoFound) {
- /* DASH */
- if (ytVideoList['Medium Bitrate Audio MP4'] || ytVideoList['Medium Bitrate Audio WebM']) {
- for (var myVideoCode in ytVideoList) {
- if (myVideoCode.indexOf('Video') != -1) {
- if (!ytVideoList[myVideoCode.replace(' Video', '')]) {
- ytVideoList[myVideoCode.replace(' Video', '')] = 'DASH';
- }
- }
- }
- }
- /* HLS */
- if (!ytVideosContentHLS) {
- ytGetVideos(true, 'IOS', false);
- }
- if (ytVideosContentHLS) {
- ytVideoList["Multi Definition M3U8"] = ytVideosContentHLS;
- }
- ytVideosContentHLS = '';
- ytGetVideos(true, 'WEB_SAFARI', false);
- if (ytVideosContentHLS) {
- var ytHLSFormats = {
- '92': 'Very Low Definition M3U8',
- '93': 'Low Definition M3U8',
- '94': 'Standard Definition M3U8',
- '95': 'High Definition M3U8',
- '96': 'Full High Definition M3U8'
- };
- var ytHLSVideos, ytHLSVideo, ytVideoCode, myVideoCode;
- ytHLSVideos = getMyContent(ytVideosContentHLS, /(http.*?m3u8)/g);
- if (ytHLSVideos) {
- for (var i = 0; i < ytHLSVideos.length; i++) {
- ytHLSVideo = ytHLSVideos[i];
- ytVideoCode = parseMyContent(ytHLSVideo, /\/itag\/(\d{1,3})\//);
- if (ytVideoCode) {
- myVideoCode = ytHLSFormats[ytVideoCode];
- if (myVideoCode) {
- ytVideoList[myVideoCode] = ytHLSVideo;
- }
- }
- }
- }
- }
- ytCreateSaver();
- }
- else {
- ytCreateSaver({'warnMess': '!videos'});
- }
- }
- else {
- /* HLS */
- if (!ytVideosContentHLS) {
- ytGetVideos(true, 'IOS', false);
- }
- if (ytVideosContentHLS) {
- ytVideoList["Multi Definition M3U8"] = ytVideosContentHLS;
- ytDefaultVideo = 'Multi Definition M3U8';
- ytVideosContentHLS = '';
- ytGetVideos(true, 'WEB_SAFARI', false);
- if (ytVideosContentHLS) {
- var ytHLSFormats = {
- '92': 'Very Low Definition M3U8',
- '93': 'Low Definition M3U8',
- '94': 'Standard Definition M3U8',
- '95': 'High Definition M3U8',
- '96': 'Full High Definition M3U8'
- };
- var ytHLSVideos, ytHLSVideo, ytVideoCode, myVideoCode;
- ytHLSVideos = getMyContent(ytVideosContentHLS, /(http.*?m3u8)/g);
- if (ytHLSVideos) {
- for (var i = 0; i < ytHLSVideos.length; i++) {
- ytHLSVideo = ytHLSVideos[i];
- ytVideoCode = parseMyContent(ytHLSVideo, /\/itag\/(\d{1,3})\//);
- if (ytVideoCode) {
- myVideoCode = ytHLSFormats[ytVideoCode];
- if (myVideoCode) {
- ytVideoList[myVideoCode] = ytHLSVideo;
- }
- }
- }
- }
- }
- ytCreateSaver();
- }
- else {
- ytCreateSaver({'warnMess': '!content'});
- }
- }
-
- }
-
- // =====DailyMotion===== //
-
- else if (page.url.indexOf('dailymotion.com/video') != -1) {
-
- /* Video Source */
- var dmMetadataUrl = page.url.replace(/\/video\//, "/player/metadata/video/");
-
- /* Video Availability */
- if (getMyContent(dmMetadataUrl, /"error":\{"title":"(.*?)"/)) return;
- if (getMyContent(dmMetadataUrl, /"error_title":"(.*?)"/)) return;
-
- /* Get Video Title */
- var dmVideoTitle = getMyContent(dmMetadataUrl, /"title":"((\\"|[^"])*?)"/);
- if (dmVideoTitle) {
- var dmVideoAuthor = getMyContent(dmMetadataUrl, /"screenname":"((\\"|[^"])*?)"/);
- if (dmVideoAuthor) dmVideoTitle = dmVideoTitle + ' by ' + dmVideoAuthor;
- dmVideoTitle = cleanMyContent(dmVideoTitle, false, true);
- }
-
- /* Get Videos Content */
- var dmVideosContent = getMyContent(dmMetadataUrl, /"qualities":\{(.*?)\]\},/);
-
- /* Get Videos */
- if (dmVideosContent) {
- var dmVideoFormats = {'auto': 'Low Definition MP4', '240': 'Very Low Definition MP4', '380': 'Low Definition MP4',
- '480': 'Standard Definition MP4', '720': 'High Definition MP4', '1080': 'Full High Definition MP4'};
- var dmVideoList = {};
- var dmVideoFound = false;
- var myVideoCode, dmVideo;
- for (var dmVideoCode in dmVideoFormats) {
- dmVideo = parseMyContent(dmVideosContent, new RegExp('"' + dmVideoCode + '".*?"type":"video.*?mp4","url":"(.*?)"'));
- if (dmVideo) {
- if (!dmVideoFound) dmVideoFound = true;
- dmVideo = cleanMyContent(dmVideo, true);
- myVideoCode = dmVideoFormats[dmVideoCode];
- if (!dmVideoList[myVideoCode]) dmVideoList[myVideoCode] = dmVideo;
- }
- }
- if (!dmVideoFound) {
- var dmHLSManifest = parseMyContent(dmVideosContent, /"type":"application.*?mpegURL","url":"(.*?)"/);
- if (dmHLSManifest) {
- dmVideoFound = true;
- dmHLSManifest = cleanMyContent(dmHLSManifest, true);
- dmVideoList["Multi Definition M3U8"] = dmHLSManifest;
- var dmHLSFormats = {
- 'h264_aac_2': 'Low Definition M3U8',
- 'h264_aac_hq_2': 'Standard Definition M3U8',
- 'h264_aac_hd_2': 'High Definition M3U8',
- };
- var dmHLSVideos = getMyContent(dmHLSManifest, /(http.*?m3u8)/g);
- var dmHLSVideo;
- if (dmHLSVideos) {
- for (var i = 0; i < dmHLSVideos.length; i++) {
- dmHLSVideo = dmHLSVideos[i];
- dmVideoCode = parseMyContent(dmHLSVideo, /mp4_(h264_aac_.*).m3u8/);
- if (dmVideoCode) {
- myVideoCode = dmHLSFormats[dmVideoCode];
- if (myVideoCode) {
- dmVideoList[myVideoCode] = dmHLSVideo;
- }
- }
- }
- }
- }
- }
-
- if (dmVideoFound) {
- /* Create Saver */
- var dmDefaultVideo = 'Low Definition MP4';
- if (!dmVideoList[dmDefaultVideo]) dmDefaultVideo = 'Low Definition M3U8';
- saver = {
- 'videoList': dmVideoList,
- 'videoDefinitions': ['Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
- 'videoContainers': ['MP4'],
- 'videoSave': dmDefaultVideo,
- 'videoTitle': dmVideoTitle
- };
- createMySaver();
- }
- else {
- saver = {'warnMess': '!videos'};
- createMySaver();
- }
- }
- else {
- saver = {'warnMess': '!content'};
- createMySaver();
- }
-
- }
-
- // =====Vimeo===== //
-
- else if (page.url.indexOf('vimeo.com/') != -1) {
-
- /* Page Type */
- var viPageType = getMyContent(page.url, /meta\s+property="og:type"\s+content="(.*?)"/);
- if (!viPageType || viPageType.indexOf('video') == -1) return;
-
- /* Get Video Title */
- var viVideoTitle;
- if (viPageType.indexOf('video') != -1) {
- viVideoTitle = getMyContent(page.url, /meta\s+property="og:title"\s+content="(.*?)"/);
- }
- else {
- viVideoTitle = getMyContent(page.url, /"title":"((\\"|[^"])*?)"/);
- }
- if (viVideoTitle) {
- viVideoTitle = viVideoTitle.replace(/\s*on\s*Vimeo$/, '');
- var viVideoAuthor = getMyContent(page.url, /"display_name":"((\\"|[^"])*?)"/);
- if (viVideoAuthor) viVideoTitle = viVideoTitle + ' by ' + viVideoAuthor;
- viVideoTitle = cleanMyContent(viVideoTitle, false, true);
- }
-
- /* Get Content Source */
- var viVideoSource = getMyContent(page.url, /config_url":"(.*?)"/);
- if (viVideoSource) viVideoSource = cleanMyContent(viVideoSource, false);
- else {
- viVideoSource = getMyContent(page.url, /data-config-url="(.*?)"/);
- if (viVideoSource) viVideoSource = viVideoSource.replace(/&/g, '&');
- else viVideoSource = getMyContent(page.url, /embedUrl":"(.*?)"/);
- }
-
- /* Get Videos Content */
- var viVideosContent;
- if (viVideoSource) {
- viVideosContent = getMyContent(viVideoSource);
- try {
- viVideosContent = JSON.parse(viVideosContent);
- if (viVideosContent['request'] && viVideosContent['request']['files']) {
- viVideosContent = viVideosContent['request']['files'];
- }
- else {
- viVideosContent = '';
- }
- }
- catch(e) {
- viVideosContent = '';
- }
- if (!viVideosContent) {
- viVideosContent = getMyContent(viVideoSource, null, null, {'withCredentials':true});
- try {
- viVideosContent = JSON.parse(viVideosContent);
- if (viVideosContent['request'] && viVideosContent['request']['files']) {
- viVideosContent = viVideosContent['request']['files'];
- }
- else {
- viVideosContent = '';
- }
- }
- catch(e) {
- viVideosContent = '';
- }
- }
- if (!viVideosContent) {
- var viAPIToken = getMyContent('https://vimeo.com/_next/jwt', /"token":"((\\"|[^"])*?)"/, null, {'X-Requested-With':'XMLHttpRequest'});
- var viAPIUrl = '';
- if (page.url.replace(/\/$/, '').split('/').length > 4) {
- viAPIUrl = 'https://api.vimeo.com/videos/' + page.url.split('/')[3] + ':' + page.url.split('/')[4] + '?fields=config_url';
- }
- else {
- viAPIUrl = 'https://api.vimeo.com/videos/' + page.url.split('/')[3] + '?fields=config_url';
- }
- if (viAPIToken) {
- viVideoSource = getMyContent(viAPIUrl, /config_url":\s*"(.*?)"/, null, {'Authorization':'jwt ' + viAPIToken});
- if (viVideoSource) {
- viVideoSource = cleanMyContent(viVideoSource, false);
- viVideosContent = getMyContent(viVideoSource);
- try {
- viVideosContent = JSON.parse(viVideosContent);
- if (viVideosContent['request'] && viVideosContent['request']['files']) {
- viVideosContent = viVideosContent['request']['files'];
- }
- else {
- viVideosContent = '';
- }
- }
- catch(e) {
- viVideosContent = '';
- }
- if (!viVideosContent) {
- viVideosContent = getMyContent(viVideoSource, null, null, {'withCredentials':true});
- try {
- viVideosContent = JSON.parse(viVideosContent);
- if (viVideosContent['request'] && viVideosContent['request']['files']) {
- viVideosContent = viVideosContent['request']['files'];
- }
- else {
- viVideosContent = '';
- }
- }
- catch(e) {
- viVideosContent = '';
- }
- }
- }
- }
- }
- }
-
- /* Get Videos */
- if (viVideosContent) {
- var viVideoFormats = {'1440p': 'Quad High Definition MP4', '1080p': 'Full High Definition MP4', '720p': 'High Definition MP4', '540p': 'Standard Definition MP4',
- '414': 'Standard Definition MP4', '480p': 'Standard Definition MP4', '360p': 'Low Definition MP4', '270p': 'Very Low Definition MP4', '240p': 'Very Low Definition MP4'};
- var viVideoList = {};
- var viVideoFound = false;
- var viVideo, myVideoCode;
- if (viVideosContent['progressive']) {
- var viVideos = viVideosContent['progressive'];
- for (var i = 0; i < viVideos.length; i++) {
- for (var viVideoCode in viVideoFormats) {
- if (viVideos[i]['quality'] == viVideoCode || viVideos[i]['height'] == viVideoCode) {
- viVideo = viVideos[i]['url'];
- if (viVideo) {
- if (!viVideoFound) viVideoFound = true;
- myVideoCode = viVideoFormats[viVideoCode];
- viVideoList[myVideoCode] = viVideo;
- }
- }
- }
- }
- }
- if (viVideosContent['hls']) {
- if (viVideosContent['hls']['cdns']) {
- if (viVideosContent['hls']['cdns']['akfire_interconnect_quic']) {
- viVideoList["Multi Definition M3U8"] = viVideosContent['hls']['cdns']['akfire_interconnect_quic']['url'];
- if (!viVideoFound) viVideoFound = true;
- }
- else if (viVideosContent['hls']['cdns']['fastly_skyfire']) {
- viVideoList["Multi Definition M3U8"] = viVideosContent['hls']['cdns']['fastly_skyfire']['url'];
- if (!viVideoFound) viVideoFound = true;
- }
- }
- }
-
- if (viVideoFound) {
- /* Create Saver */
- var viDefaultVideo = 'Low Definition MP4';
- if (!viVideoList[viDefaultVideo]) {
- viDefaultVideo = 'Multi Definition M3U8';
- }
- saver = {
- 'videoList': viVideoList,
- 'videoDefinitions': ['Quad High Definition', 'Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
- 'videoContainers': ['MP4'],
- 'videoSave': viDefaultVideo,
- 'videoTitle': viVideoTitle
- };
- createMySaver();
- }
- else {
- saver = {'warnMess': '!videos'};
- createMySaver();
- }
- }
- else {
- saver = {'warnMess': '!content'};
- createMySaver();
- }
-
- }
-
- // =====IMDB===== //
-
- else if (page.url.indexOf('imdb.com') != -1) {
-
- /* Redirect To Video Page */
- if (page.url.indexOf('/video/') == -1 && page.url.indexOf('/videoplayer/') == -1) {
- page.doc.addEventListener('click', function(e) {
- var p = e.target.parentNode;
- while (p) {
- if (p.tagName === 'A' && p.href.indexOf('/video/imdb') != -1) {
- page.win.location.href = p.href.replace(/imdb\/inline.*/, '');
- }
- p = p.parentNode;
- }
- }, false);
- return;
- }
-
- /* Get Video Title */
- var imdbVideoTitle = getMyContent(page.url, /meta\s+property="og:title"\s+content="(.*?)"/);
- if (imdbVideoTitle) imdbVideoTitle = cleanMyContent(imdbVideoTitle, false, true);
-
- /* Get Videos Content */
- var imdbVideosContent = getMyContent(page.url, /"playbackURLs":(\[.*?\])/);
- try {
- imdbVideosContent = JSON.parse(imdbVideosContent);
- }
- catch(e) {
- imdbVideosContent = {};
- }
-
- /* Get Videos */
- var imdbVideoList = {};
- if (imdbVideosContent) {
- var imdbVideoFormats = {'1080p': 'Full High Definition MP4', '720p': 'High Definition MP4', '480p': 'Standard Definition MP4',
- '360p': 'Low Definition MP4', 'SD': 'Low Definition MP4', '240p': 'Very Low Definition MP4', 'AUTO': 'Multi Definition M3U8'};
- var imdbVideoFound = false;
- var myVideoCode, imdbVideo;
- for (var imdbVideoCode in imdbVideoFormats) {
- for (var i = 0; i < imdbVideosContent.length; i++) {
- if (imdbVideosContent[i]["displayName"]["value"] == imdbVideoCode) {
- imdbVideo = imdbVideosContent[i]["url"];
- }
- if (imdbVideo) {
- imdbVideo = cleanMyContent(imdbVideo, false);
- if (!imdbVideoFound) imdbVideoFound = true;
- myVideoCode = imdbVideoFormats[imdbVideoCode];
- if (!imdbVideoList[myVideoCode]) imdbVideoList[myVideoCode] = imdbVideo;
- }
- }
- }
-
- if (imdbVideoFound) {
- /* Create Saver */
- var imdbDefaultVideo = 'Low Definition MP4';
- saver = {
- 'videoList': imdbVideoList,
- 'videoDefinitions': ['Full High Definition', 'High Definition', 'Standard Definition', 'Low Definition', 'Very Low Definition'],
- 'videoContainers': ['MP4', 'M3U8', 'Any'],
- 'videoSave': imdbDefaultVideo,
- 'videoTitle': imdbVideoTitle
- };
- createMySaver();
- }
- else {
- saver = {'warnMess': '!videos'};
- createMySaver();
- }
- }
- else {
- saver = {'warnMess': '!content'};
- createMySaver();
- }
-
- }
-
- }
-
-
- // ==========Run========== //
-
- getMyOptions();
- SaveTube();
-
- page.win.setInterval(function() {
- if (page.url != page.win.location.href.replace(page.win.location.hash, '')) {
- if (saver['saverPanel'] && saver['saverPanel'].parentNode) {
- removeMyElement(saver['saverPanel'].parentNode, saver['saverPanel']);
- }
- page.doc = page.win.document;
- page.body = page.doc.body;
- page.url = page.win.location.href.replace(page.win.location.hash, '');
- SaveTube();
- }
- }, 500);
-
- })();