Skip to content

Instantly share code, notes, and snippets.

@dwbfox
Created January 30, 2012 02:12
Show Gist options
  • Save dwbfox/1701972 to your computer and use it in GitHub Desktop.
Save dwbfox/1701972 to your computer and use it in GitHub Desktop.
JS YouTube Download Userscript
// ==UserScript==
// @id www.youtube.com-2925af97-1b59-4d38-bc09-e2fff53f9d6a@defb5eb4-0168-463d-913b-8b4fbfa18502
// @name YouTube + MP3 Downloader
// @version 1.3.2.1
// @namespace defb5eb4-0168-463d-913b-8b4fbfa18502
// @author atmega
// @description Download YouTube videos with a single click of a button!
// @include http://www.youtube.com/watch?v=*
// @include http://www.youtube.com/watch?feature=*
// @run-at document-end
// @icon 
// ==/UserScript==
(function () {
timer1 = new Date().getTime();
// Video Settings
var VIDEO_TITLE = document.getElementById('eow-title').getAttribute('title');
var VIDEO_ID = window.location.href.match(/v=[A-Za-z0-9-_]{11}/)[0].substring(2);
// Script settings
var SETTINGS = {
"SCRIPT_VERSION": "1.3.2.1",
"URL": "http://userscripts.org/scripts/source/123836.user.js",
"SCRIPT_ID": 123836,
"DOWNLOAD_DISABLE_TIME": 3,
"NOTIFY_WAIT_TIME": 5,
"QUALITY": {
"13": "3GP",
"17": "3GP (HQ)",
"5": "240p (FLV)",
"34": "360p (FLV)",
"35": "480p (FLV)",
"18": "360p (MP4)",
"22": "720p HD(MP4)",
"37": "1080p HD (MP4)",
"43": "360p (WebM)",
"44": "480p (WebM)",
"45": "720p HD(WebM)",
"46": "1080p HD (WebM)",
"100": "360p 3D(WebM)",
"101": "480p 3D(WebM)",
"102": "720p 3D(WebM)",
"82": "360p 3D(MP4)",
"83": "240p 3D(MP4)",
"84": "720p 3D(MP4)",
"85": "520p 3D(MP4)"
}
};
// Create the main interface div box along with the download button
createDownloadBox();
// Get an array of all available download links
links = getDownloadLinks();
// Populate the combox box with available links
populateList(links);
// Update button is clicked
document.getElementById('download-checkUpdate').addEventListener('click', function (e) {
e.preventDefault();
notify('Checking for updates...this might take a few seconds.');
var rVer = checkForUpdates(SETTINGS.SCRIPT_ID, function (rver) {
if (confirm('Local version: ' + SETTINGS.SCRIPT_VERSION + '\nRemote version: ' + rver + '\n\nWould you like to update this script?')) {
window.location.href = SETTINGS.URL;
}
});
});
// Report bug button is cliced
document.getElementById('download-reportBug').addEventListener('click', function (e) {
e.preventDefault();
notify('Feature coming soon!');
});
// Download button is clicked
document.getElementById('download-button').addEventListener('click', function (e) {
var cmdButton = this;
cmdButton.setAttribute('disabled', 'disabled');
window.location = document.getElementById('qualitySelect').value;
setTimeout(function () {
cmdButton.removeAttribute('disabled');
}, SETTINGS.DOWNLOAD_DISABLE_TIME * 1000);
});
timer2 = new Date().getTime();
diff = (timer2 - timer1) / 1000;
notify('YouTube + MP3: Script loaded in <strong>' + diff + '</strong> seconds.');
function createDownloadBox() {
var headline = document.getElementById('watch-actions');
// Create the main download widget box
var box = document.createElement('div');
var boxCSS = 'padding:15px';
box.setAttribute('style', boxCSS);
box.setAttribute('id', 'download-box');
headline.appendChild(box);
// Attach the script logo
var logoLink = document.createElement('a');
var logoImg = document.createElement('img');
logoImg.setAttribute('src', 'http://i41.tinypic.com/4h71g1.png');
logoImg.setAttribute('class', 'yt-uix-tooltip-reverse yt-uix-tooltip');
logoImg.setAttribute('data-tooltip-text', 'YouTube + MP3 - Check for updates');
logoLink.href = '#';
logoLink.setAttribute('id', 'download-checkUpdate');
logoLink.setAttribute('style', 'position:relative;top:6px;');
logoLink.appendChild(logoImg);
box.appendChild(logoLink);
// Report bug logo
var logoLink = document.createElement('a');
var logoImg = document.createElement('img');
logoImg.setAttribute('src', 'http://i40.tinypic.com/2edmtzc.png');
logoImg.setAttribute('class', 'yt-uix-tooltip-reverse yt-uix-tooltip');
logoImg.setAttribute('data-tooltip-text', 'YouTube + MP3 - Report a bug');
logoLink.href = '#';
logoLink.setAttribute('id', 'download-reportBug');
logoLink.setAttribute('style', 'position:relative;top:6px;');
logoLink.appendChild(logoImg);
box.appendChild(logoLink);
// Create the combo box
var selector = document.createElement('select');
selector.setAttribute('id', 'qualitySelect');
box.appendChild(document.createTextNode('Available Formats '));
box.appendChild(selector);
// Create the download button
var button = document.createElement('button');
button.setAttribute('class', 'start yt-uix-tooltip-reverse yt-uix-button yt-uix-button-default');
button.setAttribute('id', 'download-button');
button.setAttribute('style', 'position:relative;bottom:2px;left:10px;');
button.appendChild(document.createTextNode('Download video'));
box.appendChild(button);
// Create notification box
var notify = document.createElement('div');
notify.setAttribute('id', 'download-notify');
notify.setAttribute('style', 'display:none;text-align:center;padding-top:6px;margin-top:10px;width:auto;height:24px;background:#B1FFBD;border:1px solid #8BFF7D;');
document.getElementById('download-box').appendChild(notify);
return box;
}
function getDownloadLinks() {
var PAGE_SOURCE = document.body.innerHTML;
var downloadLinks = [];
var matches = PAGE_SOURCE.match(/m_map": "url=(.*?)", "/g)[0].split('url=');
// Iterate through each valid download link and clean it up
for (i = 0; i <= matches.length - 1; i++) {
// Make sure it's a valid download link
if (matches[i].indexOf('lscache') !== -1) {
downloadLinks.push(unescape(matches[i]).replace(/(\\u0026quality.+|\\u0026type.+)/, ''));
}
}
return downloadLinks;
}
function notify(m) {
var nbox = document.getElementById('download-notify');
nbox.innerHTML = m;
nbox.style.display = "block";
// Hide the notification box after 'x' seconds.
setTimeout(function () {
nbox.style.display = "none";
}, SETTINGS.NOTIFY_WAIT_TIME * 1000);
}
function populateList(list) {
console.log('Populating combo box with ' + list.length + ' links.');
var qSelect = document.getElementById('qualitySelect');
// Append MP3 download link
var option = document.createElement('option');
option.setAttribute('value', 'http://www.youtube-mp3.org/#v=' + VIDEO_ID);
qSelect.appendChild(option);
option.appendChild(document.createTextNode('MP3'));
// iterate through the download link list and append it to the select combo box
for (i = 0; i <= list.length - 1; i++) {
// Attach the video's title to the download link
var downloadLink = list[i] + "&title=" + VIDEO_TITLE;
// Get the itag query string which indicates video format and quality
var format = downloadLink.match(/itag=\d{1,3}/)[0].substr(5);
var quality = SETTINGS.QUALITY[format];
if (typeof quality === 'undefined') {
console.log("Ran into unknown quality: " + format + ". Skipping...");
continue;
}
//console.log('Video format: ' + format);
option = document.createElement('option');
option.setAttribute('value', downloadLink);
qSelect.appendChild(option);
option.appendChild(document.createTextNode(quality));
//console.log('Link ' + (i + 1) + ' proccessed.');
}
}
function checkForUpdates(scriptID, success) {
console.log('checkForUpdates: Now checking for updates...');
// Browser does not support GM_xmlhttprequest
if (typeof GM_xmlhttpRequest === 'undefined') {
notify('Your browser does not support this feature.');
return;
}
// Request
GM_xmlhttpRequest({
method: "GET",
url: "http://userscripts.org/scripts/review/" + scriptID,
onload: function (res) {
//get remote version
try {
var remoteVersion = res.responseText.match(/@version\s+(\d|\.)+/)[0].match(/(\d|\.)+/)[0]
console.info('Successfully received remote version ' + remoteVersion);
success(remoteVersion);
} catch (e) {
console.warn('Unable to check for updates: ' + e.message);
}
}
});
}
})();
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment