NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
/** YouTube link resolving Originally written by angelsl With contributions from Manish Burman http://mburman.com With contributions from LouCypher https://github.com/LouCypher YTGrab is distributed under the GNU LGPL v3 or later and comes with no warranty. Full preamble at https://github.com/angelsl/misc-Scripts/blob/master/Greasemonkey/LICENSE.md#ytgrab //===========DS===========// This is a DefSoul MOD for use with hive. All non hive related code is credited to angelsl and contributers above. (My code will have //===========DS===========// above it) angelsl's scripts can be found here > https://github.com/angelsl/misc-Scripts //===========DS===========\\ // ==UserScript== // @name Hive - YouTube to Hive / Local Download // @namespace https://openuserjs.org/users/DefSoul/scripts // @description Inserts a download button on YouTube video pages and sends to hive -Major fixes // @version 2.3 > playlists now get their own folder // @run-at document-end // @include http*://www.youtube.com/* // @include http*://api.hive.im/api/* // @include https://touch.hive.im/account/* // @exclude http*://*.google.com/* // @exclude http*://*.facebook.com/* // @exclude http*://facebook.com/* // @exclude about:blank // @exclude http*://*.stripe.com/* // @require https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js // @resource toastrCss http://cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/css/toastr.min.css // @require http://cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/js/toastr.min.js // @require http://esprima.org/esprima.js // @require https://code.novetica.org/projects/LJVE/repos/esprima/browse/test/3rdparty/escodegen.browser.js?at=878e4471793e72ce990096cb4c92863d89e69696&raw // @grant GM_xmlhttpRequest // @grant GM_getValue // @grant GM_setValue // @grant GM_log // @grant GM_getResourceText // @grant GM_addStyle // @grant unsafeWindow // ==/UserScript== */ //===========DS===========// var nameB = "YouTube to Hive / Local Download: Test "; GM_log(nameB + location.href); var folderName = "# YouTube #"; // CASE SENSITIVE var ytFolderId; var uploadFolderId; var auth; auth = GM_getValue("auth"); var link; GM_setValue("ready", "false"); //GM_deleteValue("auth"); var ru; var uploadToHive; var uploadPng = ""; var downloadPng = ""; function log(str){console.log('%c ' + str, 'background: #000000; color: #FFFFFF');} // CUSTOM LOG var newCSS = GM_getResourceText ("toastrCss"); GM_addStyle(newCSS); toastr.options = { "closeButton": false, "debug": false, "newestOnTop": false, "progressBar": false, "positionClass": "toast-bottom-right", "preventDuplicates": true, "onclick": null, "showDuration": "300", "hideDuration": "1000", "timeOut": "12000", "extendedTimeOut": "1000", "showEasing": "swing", "hideEasing": "linear", "showMethod": "fadeIn", "hideMethod": "fadeOut" }; $(document).on("click", "#hiveSwitch", function(){ if ($("#hiveSwitch").attr("src") === uploadPng){ uploadToHive = false; $("#hiveSwitch").attr("src", downloadPng); $("#hiveSwitch").attr("title", "Local download activated."); document.getElementById('btnDownload').innerHTML = 'Download'; $("#hiveSwitch").css("right", "9px"); if ($("#watch-action-panels").css("display") == "none") document.getElementById("btnDownload").click(); } else{ uploadToHive = true; $("#hiveSwitch").attr("src", uploadPng); $("#hiveSwitch").attr("title", "Upload to Hive activated."); document.getElementById('btnDownload').innerHTML = ' Upload '; $("#hiveSwitch").css("right", "0px"); if ($("#watch-action-panels").css("display") == "none") document.getElementById("btnDownload").click(); } }); //===========DS===========\\ if (typeof unsafeWindow === 'undefined' || typeof unsafeWindow.ytplayer === 'undefined') { var p = document.createElement('p'); p.setAttribute('onclick', 'return window;'); unsafeWindow = p.onclick(); } function main(decipher) { var dashmpd = unsafeWindow.ytplayer.config.args.dashmpd, mpbsrgx = /\/s\/([\w\.]+)/, mpbs; if (typeof dashmpd !== 'undefined') { mpbs = mpbsrgx.exec(dashmpd); if(mpbs) dashmpd = dashmpd.replace(mpbsrgx, "/signature/"+decipher(mpbs[1])); GM_xmlhttpRequest({method: "GET", url: dashmpd, onload: function (t) { main2(t.responseText, decipher); }}); } else main2(false, decipher); } function main2(dashmpd, decipher) { "use strict"; var uriencToMap = function (s) { var n = {}, a = s.split("&"), idy, c; for (idy = 0; idy < a.length; idy++) { c = a[idy].split("="); n[c[0]] = decodeURIComponent(c[1]); } return n; }, uwyca = unsafeWindow.ytplayer.config.args, title = uwyca.title.replace(/[\/\\\:\*\?\"<\>\|]/g, ""), fmtrgx = /^[\-\w+]+\/(?:x-)?([\-\w+]+)/, fmt_map = {}, idx, idz, n, a, qual, fmt, fmt_list, map, uefmss, dashlist, ul, q, div, type, itag, maporder, fpsa, fpsb, fpsw = false; fmt_list = uwyca.fmt_list.split(","); for (idx = 0; idx < fmt_list.length; idx++) { a = fmt_list[idx].split("/"); fmt_map[a[0]] = a[1].split("x")[1] + "p"; } map = {}; uefmss = uwyca.url_encoded_fmt_stream_map.split(","); for (idx = 0; idx < uefmss.length; idx++) { n = uriencToMap(uefmss[idx]); qual = fmt_map[n.itag]; if (!(qual in map)) { map[qual] = []; } fmt = fmtrgx.exec(n.type); map[qual].push($("<a>" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.url + ((n.url.indexOf("signature=") !== -1) ? "" : ("&signature=" + (n.sig || decipher(n.s)))) + "&title=" + title).attr("title", "Format ID: " + n.itag + " | Quality: " + n.quality + " | Mime: " + n.type)); } dashlist = uwyca.adaptive_fmts; if (typeof dashlist !== 'undefined') { dashlist = dashlist.split(","); for (idx = 0; idx < dashlist.length; idx++) { n = uriencToMap(dashlist[idx]); qual = n.type.indexOf("audio/") === 0 ? "Audio" : (("size" in n) ? (n.size.split('x')[1] + 'p' + n.fps) : (n.itag in fmt_map) ? (fmt_map[n.itag]) : ("Unknown")); if (!(qual in map)) { map[qual] = []; } fmt = fmtrgx.exec(n.type); if (parseInt(n.fps) == 1) fpsw = 1; map[qual].push($("<a>DASH" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.url + ((n.url.indexOf("signature=") !== -1) ? "" : ("&signature=" + (n.sig || decipher(n.s)))) + "&title=" + title).attr("title", "Format ID: " + n.itag + " | Bitrate: " + n.bitrate + " | Mime: " + n.type + " | Res: " + n.size + " | FPS: " + n.fps)); } } if (dashmpd !== false) { dashmpd = $($.parseXML(dashmpd)); dashmpd.find("AdaptationSet").each(function() { q = $(this); type = q.attr("mimeType"); q.children("Representation").each(function() { n = $(this); itag = n.attr("id"); qual = type.indexOf("audio/") === 0 ? "Audio" : (n.attr("height") + 'p' + n.attr("frameRate")); if (!(qual in map)) { map[qual] = []; } fmt = fmtrgx.exec(type); if (parseInt(n.attr("frameRate")) == 1) fpsw = 1; map[qual].push($("<a>MPD" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.children("BaseURL").text() + "&title=" + title).attr("title", "Format ID: " + itag + " | Bitrate: " + n.attr("bandwidth") + " | Mime: " + type + (type.indexOf("audio/") === 0 ? " | Sample Rate: " + n.attr("audioSamplingRate") : " | Res: " + n.attr("width") + 'x' + n.attr("height") + " | FPS: " + n.attr("frameRate")))); }); }); } maporder = Object.keys(map); maporder.sort(function(a,b) { if((a == "Audio" && b == "Unknown") || (b == "Audio" && a != "Unknown")) return -1; if ((b == "Audio" && a == "Unknown") || (a == "Audio" && b != "Unknown")) return 1; fpsa = a.split('p')[1] || 0; fpsb = b.split('p')[1] || 0; if (fpsa != fpsb) return parseInt(fpsb)-parseInt(fpsa); return parseInt(b)-parseInt(a); }); ul = $("<ul class=\"watch-extras-section\" />"); for (n = 0; n < maporder.length; ++n) { q = maporder[n]; if (map[q].length < 1) { continue; } div = $("<div class=\"content\" />").append(map[q][0]); for (idz = 1; idz < map[q].length; idz++) { div.append(" ").append(map[q][idz]); } ul.append($("<li><h4 class=\"title\" style=\"font-weight: bold; color: #333333;\">" + q + "</h4></li>").append(div)); } $("#action-panel-share").after($("<div id=\"action-panel-sldownload\" class=\"action-panel-content hid\" data-panel-loaded=\"true\" />").append(ul)); $("#watch8-secondary-actions").find("> div").eq(1).after($('<button class="yt-uix-button yt-uix-button-size-default yt-uix-button-opacity action-panel-trigger yt-uix-button-opacity yt-uix-tooltip" style="text-align: center;" type="button" onclick=";return false;" title="" id="btnDownload" data-trigger-for="action-panel-sldownload" data-button-toggle="true"><span class="yt-uix-button-content">Upload</span></button>')).size(); //===========DS===========// //$("#hiveSwitch").css("display", "block"); $("#watch8-secondary-actions").find("> div").eq(1).after($('<img title="Upload to Hive activated." src="' + uploadPng + '" type="button" onclick=";return false;" title="" id="hiveSwitch" style="right: 0px; bottom: 41px; z-index: 9999999; cursor: pointer; position: absolute; display: block; height: 50px; width: 50px; padding-left: 5px;" data-button-toggle="true"><span class=""></span></button>')).size(); //===========DS===========\\ if (fpsw) ul.after($("<p style='color: green;'>At this time Hive only accepts Mp4 & Flv video files, the other formats are for local downloading.</p>")); } function run() { if (typeof unsafeWindow.ytplayer !== 'undefined') { GM_xmlhttpRequest({method: "GET", url: unsafeWindow.ytplayer.config.assets.js.replace(/^\/\//, "https://"), onload: function (t) { main((function (u) { "use strict"; var sres = /function ([a-zA-Z$0-9]+)\(a\)\{a=a\.split\(""\);([a-zA-Z0-9]*)\.?.*?return a\.join\(""\)\};/g.exec(u); if (!sres) { return function (v) { return v; }; } return eval("(function(s){" + (sres[2] !== "" ? (new RegExp("var " + sres[2] + "={.+?}};", "g").exec(u)[0]) : "") + sres[0] + "return " + sres[1] + "(s);})"); }(t.responseText))); }}); } } //DS// function run2(val) { log("run2 running"); GM_xmlhttpRequest({ method: "GET", url: val, onload: function(t){ json = JSON.parse(t.responseText); //for(var key in json) { // var value = json[key]; // log(value); //} }}); } setInterval(function(){ if ($(".playlist-actions").length && !$("#PlaylistToHive").length){ $(".playlist-actions").append('<button id="PlaylistToHive" class="yt-uix-button yt-uix-button-size-default yt-uix-button-default yt-uix-button-has-icon no-icon-markup yt-uix-playlistlike yt-uix-tooltip" type="button" onclick=";return false;" aria-label="To Hive" title="To Hive" data-like-tooltip="Save to Playlists" data-unlike-tooltip="Remove" data-like-label="Save" data-unlike-label="Saved" data-tooltip-text="To Hive" aria-labelledby="yt-uix-tooltip95-arialabel" data-tooltip-hide-timer="235"><span class="yt-uix-button-content">To Hive</span></button>'); } }, 1000); //DS\\ waitForKeyElements("#watch8-secondary-actions", run); //===========DS===========/ function createFolderInParent(uploadFolderName, parent){ GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST "method": "post", "url": "https://api.hive.im/api/hive/create/", "data": "filename=" + uploadFolderName + "&parent=" + parent + "&locked=false", "headers": { 'Content-Type': 'application/x-www-form-urlencoded;', 'Authorization': auth, 'Client-Type': 'Browser', 'Client-Version': '0.1', 'Referer': 'https://touch.hive.im/', 'Origin': 'https://touch.hive.im/' }, "onload": function(data){ var r = data.responseText; var json = JSON.parse(r); uploadFolderId = json.data.id; log("createFolderInParent <" + uploadFolderName + "> " + json.data.id); return json.data.id; } }); } function createFolder(uploadFolderName){ GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST "method": "get", "url": "https://api.hive.im/api/hive/get/", "headers": { 'Content-Type': 'application/x-www-form-urlencoded;', 'Authorization': auth, 'Client-Type': 'Browser', 'Client-Version': '0.1', 'Referer': 'https://touch.hive.im/myfiles/videos', 'Origin': 'https://touch.hive.im/' }, "onload": function(data){ var r = data.responseText; var json = JSON.parse(r); for (var i = 0; i < json.data.length; i++){ var id; if (json.data[i].title === "Videos"){ // FINDS INITIAL VIDEOS FOLDER ID //log("we got a video ova here", "green"); parentId = json.data[i].parentId; id = json.data[i].id; GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST "method": "post", "url": "https://api.hive.im/api/hive/get-children/", "data": "&parentId=" + id + "&limit=1000", "headers": { 'Content-Type': 'application/x-www-form-urlencoded;', 'Authorization': auth, 'Client-Type': 'Browser', 'Client-Version': '0.1', 'Referer': 'https://touch.hive.im/', 'Origin': 'https://touch.hive.im/' }, "onload": function(data){ var r = data.responseText; var json = JSON.parse(r); var hasFolderIndex; Object.keys(json.data).forEach(function(key) { //log(json.data[key].title, "blue"); hasFolderIndex += json.data[key].title; if (json.data[key].title === uploadFolderName){ ytFolderId = json.data[key].id; log("<" + uploadFolderName + "> Already exists. " + ytFolderId, "green"); //return json.data[key].id; } }); if (hasFolderIndex.indexOf(uploadFolderName) == -1){ // SEARCHES VIDEOS FOLDER TO SEE IF uploadFolderName EXISTS log("does not contain: " + uploadFolderName, "red"); GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST "method": "post", "url": "https://api.hive.im/api/hive/create/", "data": "filename=" + uploadFolderName + "&parent=" + id + "&locked=false", "headers": { 'Content-Type': 'application/x-www-form-urlencoded;', 'Authorization': auth, 'Client-Type': 'Browser', 'Client-Version': '0.1', 'Referer': 'https://touch.hive.im/', 'Origin': 'https://touch.hive.im/' }, "onload": function(data){ var r = data.responseText; var json = JSON.parse(r); ytFolderId = json.data.id; log("Create folder <" + uploadFolderName + "> " + json.data.id); return json.data.id; } }); } else{ //log("does contain: " + uploadFolderName, "green"); } } }); //log(parentId + "\n" + currentId); } //log(item, "blue"); } //log(r, "blue"); } }); } function cdReq(href, nameT, folderId){ log("cdReq start: " + href); GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST "method": "post", "url": "https://api.hive.im/api/transfer/add/", "data": "remoteUrl=" + window.btoa(href) + "&parentId=" + folderId, //"data": "remoteUrl=" + window.btoa(href), "headers": { 'Content-Type': 'application/x-www-form-urlencoded;', 'Authorization': GM_getValue("auth"), 'Client-Type': 'Browser', 'Client-Version': '0.1', 'Referer': 'https://touch.hive.im/', 'Origin': 'https://touch.hive.im/' }, "onload": function(data){ var r = data.responseText; var json = JSON.parse(r); if (json.status === "success"){ toastr.success(nameT, "Status: " + json.data.status); log("========= " + nameT + " success =========", "green"); log("Job ID: " + json.data.jobId, "blue"); log("Data Status: " + json.data.status, "blue"); log("Folder Id: " + folderId, "blue"); log("", "red"); } else{ if (json.message === "quotaExceeded"){ toastr.warning(nameT, "Quota Exceeded"); } else if (json.message === "securityViolation"){ toastr.error(nameT, "Security Violation"); } log("========= " + nameT + " error =========", "green"); log("Message: " + json.message, "blue"); log("", "red"); } //log("cdReq >" + data.responseText); //transferItemsList(); // GO GET ITEMS IN CURRENT TRANSFER LIST } }); } $(document).on("click", "#PlaylistToHive", function(e){ // MAIN CLICK EVENT e.preventDefault(); toastr.warning("Extracting links.", "Please don't navigate from page!"); var tiles = document.getElementsByClassName("yt-uix-tile"); var titlesClass = document.getElementsByClassName("pl-video-title-link"); var titles = []; var vids = []; // CONTAINS ALL COMPLETE URLS OF ALL ITEMS IN PLAYLIST var mp4s = []; var playlistTitle = document.getElementsByClassName("pl-header-title"); playlistTitle = $(playlistTitle[0]).html(); playlistTitle = playlistTitle.replace(/(\r\n|\n|\r)/gm,""); playlistTitle = playlistTitle.trim(); playlistTitle = playlistTitle.replace(/[`~!@#$%^&*()_|+\=÷¿?;:'",.<>\{\}\[\]\\\/]/gi, ' '); log(playlistTitle, "blue"); createFolderInParent(playlistTitle, ytFolderId); //return; for (var i = 0; i < tiles.length; i++){ var r = $(titlesClass[i]).html(); r = r.replace(/(\r\n|\n|\r)/gm,""); r = r.trim(); r = r.replace(/[`~!@#$%^&*()_|+\=÷¿?;:'",.<>\{\}\[\]\\\/]/gi, '%20'); r = r.replace(/ /g, "%20"); //r = "&title=" + r; //log(r); titles.push(r); // CREATES ARRAY OF VIDEO TITLES vids.push("https://www.youtube.com/watch?v=" + $(tiles[i]).attr("data-video-id")); // CREATES ARRAY OF VIDEO URLS } var jjj = 0; for (var jI = 0; jI < vids.length; jI++){ var toastTitle; extract(vids[jI]).done(function (result) { for (var j = 0; j < result.formats.length; j++){ if (result.formats[j].ext === "mp4" && typeof result.formats[j].format_note == "undefined"){ toastTitle = titles[jjj]; toastTitle = toastTitle.replace(/%20/g, " "); mp4s = []; mp4s.push(result.formats[j].url + "&title=" + titles[jjj]); } } //log("MP4S 1: >>" + mp4s[0], "blue"); // HIGHEST QUALITY MP4 cdReq(mp4s[0], toastTitle, uploadFolderId); jjj++; setTimeout(function(){ if (jjj === jI){ toastr.info("Finished!"); } }, 5000); }); } }); if (window.top === window.self) { //=========MAIN WINDOW=========// if (document.location.href.indexOf("touch.hive.im") !== -1){ return; } createFolder(folderName); if (!$("#iframeHive").length || typeof auth == "undefined"){ var iframe = document.createElement('iframe'); iframe.id = "iframeHive"; iframe.src = "https://touch.hive.im/account/?1"; iframe.style = "height: 0px; width: 0px; display: none; overflow:hidden"; document.body.appendChild(iframe); $("#iframeHive").attr("style", "height: 0px; width: 0px; display: none; overflow:hidden"); //$("#iframeHive").attr("style", "height: 600px; width: 600px; display: block; overflow:hidden"); log("iframe created! " + nameB); } var onceB = 0; setInterval(function(){ //log("AA: " + auth); if (onceB === 0 && typeof auth !== "undefined"){ GM_setValue("ready", "true"); GM_setValue("auth", auth); $("#iframeHive").remove(); //log("TRUE: " + auth); } if (onceB === 0 && GM_getValue("ready") == "true"){ onceB = 1; auth = GM_getValue("auth"); log("A: " + auth); $("#iframeHive").remove(); //init(); } }, 250); $(document).on("click", "a", function(evt){ // MAIN CLICK EVENT if ($(this).attr('href').indexOf('googlevideo') !== -1){ if (uploadToHive === false) return; log($("#hiveSwitch").attr("src")); evt.preventDefault(); ru = $(this).attr('href'); //log("pre: " + ru); ru = ru.replace(/ /g, "%20"); //log("post: " + ru); var vidTitle = $("#eow-title").attr("title"); cdReq(ru, vidTitle, ytFolderId); } }); } else { //=========IFRAME WINDOW=========// try{ auth = unsafeWindow.account.token; } catch(err){} var once = 0; setInterval(function(){ // EVENT FOR WHEN PAGE IS LOADED // RUNS ONCE if (once === 0 && $("#username").text().indexOf("My Account") !== -1){ once = 1; log("ready"); auth = unsafeWindow.account.token; GM_setValue("auth", unsafeWindow.account.token); GM_setValue("ready", "true"); } else if (once === 1 && auth == "undefined"){ GM_setValue("ready", "false"); try{ auth = unsafeWindow.account.token; } catch(err){} } }, 200); } //===========DS===========\\ // START YT-LINKS CODE // var YT_FORMATS = { '5': {'ext': 'flv', 'width': 400, 'height': 240}, '6': {'ext': 'flv', 'width': 450, 'height': 270}, '13': {'ext': '3gp'}, '17': {'ext': '3gp', 'width': 176, 'height': 144}, '18': {'ext': 'mp4', 'width': 640, 'height': 360}, '22': {'ext': 'mp4', 'width': 1280, 'height': 720}, '34': {'ext': 'flv', 'width': 640, 'height': 360}, '35': {'ext': 'flv', 'width': 854, 'height': 480}, '36': {'ext': '3gp', 'width': 320, 'height': 240}, '37': {'ext': 'mp4', 'width': 1920, 'height': 1080}, '38': {'ext': 'mp4', 'width': 4096, 'height': 3072}, '43': {'ext': 'webm', 'width': 640, 'height': 360}, '44': {'ext': 'webm', 'width': 854, 'height': 480}, '45': {'ext': 'webm', 'width': 1280, 'height': 720}, '46': {'ext': 'webm', 'width': 1920, 'height': 1080}, '59': {'ext': 'mp4', 'width': 854, 'height': 480}, '78': {'ext': 'mp4', 'width': 854, 'height': 480}, // 3d videos '82': {'ext': 'mp4', 'height': 360, 'format_note': '3D', 'preference': -20}, '83': {'ext': 'mp4', 'height': 480, 'format_note': '3D', 'preference': -20}, '84': {'ext': 'mp4', 'height': 720, 'format_note': '3D', 'preference': -20}, '85': {'ext': 'mp4', 'height': 1080, 'format_note': '3D', 'preference': -20}, '100': {'ext': 'webm', 'height': 360, 'format_note': '3D', 'preference': -20}, '101': {'ext': 'webm', 'height': 480, 'format_note': '3D', 'preference': -20}, '102': {'ext': 'webm', 'height': 720, 'format_note': '3D', 'preference': -20}, // Apple HTTP Live Streaming '92': {'ext': 'mp4', 'height': 240, 'format_note': 'HLS', 'preference': -10}, '93': {'ext': 'mp4', 'height': 360, 'format_note': 'HLS', 'preference': -10}, '94': {'ext': 'mp4', 'height': 480, 'format_note': 'HLS', 'preference': -10}, '95': {'ext': 'mp4', 'height': 720, 'format_note': 'HLS', 'preference': -10}, '96': {'ext': 'mp4', 'height': 1080, 'format_note': 'HLS', 'preference': -10}, '132': {'ext': 'mp4', 'height': 240, 'format_note': 'HLS', 'preference': -10}, '151': {'ext': 'mp4', 'height': 72, 'format_note': 'HLS', 'preference': -10}, // DASH mp4 video '133': {'ext': 'mp4', 'height': 240, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '134': {'ext': 'mp4', 'height': 360, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '135': {'ext': 'mp4', 'height': 480, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '136': {'ext': 'mp4', 'height': 720, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '137': {'ext': 'mp4', 'height': 1080, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '138': {'ext': 'mp4', 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, // Height can vary (https://github.com/rg3/youtube-dl/issues/4559) '160': {'ext': 'mp4', 'height': 144, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '264': {'ext': 'mp4', 'height': 1440, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '298': { 'ext': 'mp4', 'height': 720, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'fps': 60, 'vcodec': 'h264' }, '299': { 'ext': 'mp4', 'height': 1080, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'fps': 60, 'vcodec': 'h264' }, '266': { 'ext': 'mp4', 'height': 2160, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'vcodec': 'h264' }, // Dash mp4 audio '139': { 'ext': 'm4a', 'format_note': 'DASH audio', 'acodec': 'aac', 'vcodec': 'none', 'abr': 48, 'preference': -50, 'container': 'm4a_dash' }, '140': { 'ext': 'm4a', 'format_note': 'DASH audio', 'acodec': 'aac', 'vcodec': 'none', 'abr': 128, 'preference': -50, 'container': 'm4a_dash' }, '141': { 'ext': 'm4a', 'format_note': 'DASH audio', 'acodec': 'aac', 'vcodec': 'none', 'abr': 256, 'preference': -50, 'container': 'm4a_dash' }, // Dash webm '167': { 'ext': 'webm', 'height': 360, 'width': 640, 'format_note': 'DASH video', 'acodec': 'none', 'container': 'webm', 'vcodec': 'VP8', 'preference': -40 }, '168': { 'ext': 'webm', 'height': 480, 'width': 854, 'format_note': 'DASH video', 'acodec': 'none', 'container': 'webm', 'vcodec': 'VP8', 'preference': -40 }, '169': { 'ext': 'webm', 'height': 720, 'width': 1280, 'format_note': 'DASH video', 'acodec': 'none', 'container': 'webm', 'vcodec': 'VP8', 'preference': -40 }, '170': { 'ext': 'webm', 'height': 1080, 'width': 1920, 'format_note': 'DASH video', 'acodec': 'none', 'container': 'webm', 'vcodec': 'VP8', 'preference': -40 }, '218': { 'ext': 'webm', 'height': 480, 'width': 854, 'format_note': 'DASH video', 'acodec': 'none', 'container': 'webm', 'vcodec': 'VP8', 'preference': -40 }, '219': { 'ext': 'webm', 'height': 480, 'width': 854, 'format_note': 'DASH video', 'acodec': 'none', 'container': 'webm', 'vcodec': 'VP8', 'preference': -40 }, '278': { 'ext': 'webm', 'height': 144, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'container': 'webm', 'vcodec': 'VP9' }, '242': {'ext': 'webm', 'height': 240, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '243': {'ext': 'webm', 'height': 360, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '244': {'ext': 'webm', 'height': 480, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '245': {'ext': 'webm', 'height': 480, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '246': {'ext': 'webm', 'height': 480, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '247': {'ext': 'webm', 'height': 720, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '248': {'ext': 'webm', 'height': 1080, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '271': {'ext': 'webm', 'height': 1440, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '272': {'ext': 'webm', 'height': 2160, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40}, '302': { 'ext': 'webm', 'height': 720, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'fps': 60, 'vcodec': 'VP9' }, '303': { 'ext': 'webm', 'height': 1080, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'fps': 60, 'vcodec': 'VP9' }, '308': { 'ext': 'webm', 'height': 1440, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'fps': 60, 'vcodec': 'VP9' }, '313': { 'ext': 'webm', 'height': 2160, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'vcodec': 'VP9' }, '315': { 'ext': 'webm', 'height': 2160, 'format_note': 'DASH video', 'acodec': 'none', 'preference': -40, 'fps': 60, 'vcodec': 'VP9' }, // Dash webm audio '171': {'ext': 'webm', 'vcodec': 'none', 'format_note': 'DASH audio', 'abr': 128, 'preference': -50}, '172': {'ext': 'webm', 'vcodec': 'none', 'format_note': 'DASH audio', 'abr': 256, 'preference': -50}, // Dash webm audio with opus inside '249': { 'ext': 'webm', 'vcodec': 'none', 'format_note': 'DASH audio', 'acodec': 'opus', 'abr': 50, 'preference': -50 }, '250': { 'ext': 'webm', 'vcodec': 'none', 'format_note': 'DASH audio', 'acodec': 'opus', 'abr': 70, 'preference': -50 }, '251': { 'ext': 'webm', 'vcodec': 'none', 'format_note': 'DASH audio', 'acodec': 'opus', 'abr': 160, 'preference': -50 }, // RTMP (unnamed) '_rtmp': {'protocol': 'rtmp'}, } // QueryString - begin // This is public domain code written in 2011 by Jan Wolter and distributed // for free at http://unixpapa.com/js/querystring.html // // Query String Parser // // qs= new QueryString() // qs= new QueryString(string) // // Create a query string object based on the given query string. If // no string is given, we use the one from the current page by default. // // qs.value(key) // // Return a value for the named key. If the key was not defined, // it will return undefined. If the key was multiply defined it will // return the last value set. If it was defined without a value, it // will return an empty string. // // qs.values(key) // // Return an array of values for the named key. If the key was not // defined, an empty array will be returned. If the key was multiply // defined, the values will be given in the order they appeared on // in the query string. // // qs.keys() // // Return an array of unique keys in the query string. The order will // not necessarily be the same as in the original query, and repeated // keys will only be listed once. // // QueryString.decode(string) // // This static method is an error tolerant version of the builtin // function decodeURIComponent(), modified to also change pluses into // spaces, so that it is suitable for query string decoding. You // shouldn't usually need to call this yourself as the value(), // values(), and keys() methods already decode everything they return. // // Note: W3C recommends that ; be accepted as an alternative to & for // separating query string fields. To support that, simply insert a semicolon // immediately after each ampersand in the regular expression in the first // function below. function QueryString(qs) { this.dict = {}; // If no query string was passed in use the one from the current page if (!qs) qs = location.search; // Delete leading question mark, if there is one if (qs.charAt(0) == '?') qs = qs.substring(1); // Parse it var re = /([^=&]+)(=([^&]*))?/g; while (match = re.exec(qs)) { var key = decodeURIComponent(match[1].replace(/\+/g, ' ')); var value = match[3] ? QueryString.decode(match[3]) : ''; if (this.dict[key]) this.dict[key].push(value); else this.dict[key] = [value]; } } QueryString.decode = function (s) { s = s.replace(/\+/g, ' '); s = s.replace(/%([EF][0-9A-F])%([89AB][0-9A-F])%([89AB][0-9A-F])/gi, function (code, hex1, hex2, hex3) { var n1 = parseInt(hex1, 16) - 0xE0; var n2 = parseInt(hex2, 16) - 0x80; if (n1 == 0 && n2 < 32) return code; var n3 = parseInt(hex3, 16) - 0x80; var n = (n1 << 12) + (n2 << 6) + n3; if (n > 0xFFFF) return code; return String.fromCharCode(n); }); s = s.replace(/%([CD][0-9A-F])%([89AB][0-9A-F])/gi, function (code, hex1, hex2) { var n1 = parseInt(hex1, 16) - 0xC0; if (n1 < 2) return code; var n2 = parseInt(hex2, 16) - 0x80; return String.fromCharCode((n1 << 6) + n2); }); s = s.replace(/%([0-7][0-9A-F])/gi, function (code, hex) { return String.fromCharCode(parseInt(hex, 16)); }); return s; }; QueryString.prototype.value = function (key) { var a = this.dict[key]; return a ? a[a.length - 1] : undefined; }; QueryString.prototype.values = function (key) { var a = this.dict[key]; return a ? a : []; }; QueryString.prototype.keys = function () { var a = []; for (var key in this.dict) a.push(key); return a; }; // QueryString - end var Queue = function () { var previous = new $.Deferred().resolve(); return function (fn, fail) { return previous = previous.then(fn, fail || fn); }; }; var queue = Queue(); // lower case for idiomatic use var LAST_PLAYER_URL = null; var LAST_FUNC = null; function log(s) { try { console.log(s); } catch (ignore) { } } function download(url) { log('Downloading webpage ' + url); var deferred = $.Deferred(); //var userAgent = navigator.userAgent; var userAgent = 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/43.0.2357.81 Safari/537.36'; $.get('http://query.yahooapis.com/v1/public/yql', { q: 'select * from xClient where url="' + url + '" and ua="' + userAgent + '"', format: 'json', env: 'store://datatables.org/alltableswithkeys', callback: '' }).done(function (data) { try { deferred.resolve(data.query.results.resources.content); } catch (e) { log(e); deferred.resolve(null); } }); return deferred.promise(); } function extractId(url) { var r = /^http(s?):\/\/www\.youtube\.com\/watch\?v=(.+)/.exec(url); return r !== null ? r[2] : null; } function searchRegex(regex, string, defaultValue) { var r = regex.exec(string); if (r !== null) { return r[1]; } else { return defaultValue; } } function parseQS(s) { var qs = new QueryString(s); var obj = {}; var keys = qs.keys(); var size = keys.length; for (var i = 0; i < size; i++) { var k = keys[i]; obj[k] = qs.values(k); } return obj; } function decryptSignature(s, playerUrl) { var deferred = $.Deferred(); if (playerUrl === null) { log('Cannot decrypt signature without player_url'); deferred.resolve(null); } if (playerUrl.indexOf('//') === 0) { playerUrl = 'https:' + playerUrl; } if (LAST_PLAYER_URL === playerUrl && LAST_FUNC !== null) { var func = LAST_FUNC; var signature = func(s); deferred.resolve(signature); } else { download(playerUrl).done(function (jscode) { var func = null; if (LAST_PLAYER_URL === playerUrl && LAST_FUNC !== null) { func = LAST_FUNC; } else { var r = /\.sig\|\|([a-zA-Z0-9$]+)\(/.exec(jscode); if (r === null) { log("Couldn't find the signature code with regex"); } var funcname = r[1]; function shortcut(jscode) { var p = jscode.split('function ' + funcname + '('); if (p.length !== 2) { return null; } var i1 = p[0].lastIndexOf('};var '); if (i1 === -1) { return null; } var p1 = p[0].substr(i1 + 2); var i2 = p[1].indexOf('};'); if (i2 === -1) { return null; } var p2 = p[1].substr(0, i2 + 2); return p1 + 'function ' + funcname + '(' + p2; } var temp = shortcut(jscode); if (temp !== null) { jscode = temp; } var ast = esprima.parse(jscode); function traverse(object, level, visitor) { var key, child; if (visitor.call(null, object) === false) { return; } if (level > 8) { return; } for (key in object) { if (object.hasOwnProperty(key)) { child = object[key]; if (typeof child === 'object' && child !== null) { traverse(child, level + 1, visitor); } } } } traverse(ast, 0, function (node) { if (node.type === 'FunctionDeclaration' && node.id.name == funcname) { func = eval('(' + escodegen.generate(node) + ')'); } if (node.type === 'VariableDeclarator') { try { eval(escodegen.generate(node)); } catch (ignore) { } } }); LAST_PLAYER_URL = playerUrl; LAST_FUNC = func; } var signature = func(s); deferred.resolve(signature); }); } return deferred.promise(); } function parseDashManifest(video_id, dash_manifest_url, player_url, age_gate) { var deferred = $.Deferred(); var r = /\/s\/([a-fA-F0-9\.]+)/.exec(dash_manifest_url); if (r !== null) { var s = r[1]; var formats = []; decryptSignature(s, player_url).done(function (dec_s) { dash_manifest_url = dash_manifest_url.replace(new RegExp('/s/' + s), '/signature/' + dec_s); download(dash_manifest_url).done(function (dash_doc) { dash_doc = $.parseXML(dash_doc); $(dash_doc).find('AdaptationSet').each(function (index, elemSet) { var mimeType = $(elemSet).attr('mimeType'); $(elemSet).find('Representation').each(function (index, elemRep) { var url_el = $(elemRep).find('BaseURL'); if (mimeType.indexOf('audio/') === 0 || mimeType.indexOf('video/') === 0) { var format_id = $(elemRep).attr('id'); var video_url = $(url_el).text(); var filesize = parseInt($(url_el).attr('yt:contentLength')); var f = { format_id: format_id, url: video_url, widt: parseInt($(elemRep).attr('width')), height: parseInt($(elemRep).attr('height')), filesize: filesize } formats.push(f); } }); }); deferred.resolve({ dash_manifest_url: dash_manifest_url, formats: formats }); }); }); } else { deferred.resolve(null); } return deferred.promise(); } function getSubtitles(videoId) { var deferred = $.Deferred(); download('https://video.google.com/timedtext?hl=en&type=list&v=' + videoId).done(function (subsDoc) { subsDoc = $.parseXML(subsDoc); var subLangList = {}; $(subsDoc).find('track').each(function (index, track) { var lang = $(track).attr('lang_code'); if (subLangList.hasOwnProperty(lang)) { return; } var subFormats = []; ['sbv', 'vtt', 'srt'].forEach(function (ext) { var params = $.param({ lang: lang, v: videoId, fmt: ext, name: $(track).attr('name'), }); subFormats.push({ 'url': 'https://www.youtube.com/api/timedtext?' + params, 'ext': ext, }); }); subLangList[lang] = subFormats; }); deferred.resolve(subLangList); }).fail(function () { deferred.resolve(null); }); return deferred.promise(); } function extractSupport(video_id, video_webpage, age_gate, embed_webpage, video_info) { var deferred = $.Deferred(); function fail(s) { log(s); deferred.resolve(null); return deferred.promise(); } if (!video_info.hasOwnProperty('token')) { if (video_info.hasOwnProperty('reason')) { return fail('YouTube said: ' + video_info['reason'][0]); } else { return fail('"token" parameter not in video info for unknown reason'); } } var view_count = 0; if (video_info.hasOwnProperty('view_count')) { view_count = parseInt(video_info['view_count'][0]); } // Check for "rental" videos if (video_info.hasOwnProperty('ypc_video_rental_bar_text') && !video_info.hasOwnProperty('author')) { return fail('"rental" videos not supported'); } //Start extracting information //self.report_information_extraction(video_id) // uploader if (!video_info.hasOwnProperty('author')) { return fail('Unable to extract uploader name'); } var video_uploader = decodeURIComponent(video_info['author'][0]); // uploader_id var video_uploader_id = null; var mobj = /<link itemprop="url" href="http:\/\/www.youtube.com\/(?:user|channel)\/([^"]+)">/.exec(video_webpage); if (mobj !== null) { video_uploader_id = mobj[1]; } else { //return fail('unable to extract uploader nickname'); log('unable to extract uploader nickname'); } // title var video_title = '_'; if (video_info.hasOwnProperty('title')) { video_title = video_info['title'][0]; } else { return fail('Unable to extract video title'); } // upload date var upload_date = null; mobj = /id="eow-date.*?>(.*?)<\/span>/.exec(video_webpage); if (mobj === null) { mobj = /id="watch-uploader-info".*?>.*?(?:Published|Uploaded|Streamed live) on (.*?)<\/strong>/.exec(video_webpage); } if (mobj !== null) { upload_date = new Date(mobj[1]); //upload_date = ' '.join(re.sub(r'[/,-]', r' ', mobj.group(1)).split()) //upload_date = unified_strdate(upload_date) } // TODO: categories // description // TODO: // subtitles var videoSubtitles = null; queue(function () { return getSubtitles(video_id).done(function (subs) { videoSubtitles = subs; }); }); // TODO: //automatic_captions = self.extract_automatic_captions(video_id, video_webpage) var video_duration = null; if (!video_info.hasOwnProperty('length_seconds')) { return fail('unable to extract video duration'); } else { video_duration = parseInt(decodeURIComponent(video_info['length_seconds'][0])); } // TODO: // annotations //video_annotations = None //if self._downloader.params.get('writeannotations', False): // video_annotations = self._extract_annotations(video_id) var formats = []; if (video_info.hasOwnProperty('conn') && video_info['conn'][0].startswith('rtmp')) { return fail('RTMP not supported'); } else if (video_info.hasOwnProperty('url_encoded_fmt_stream_map') || video_info.hasOwnProperty('adaptive_fmts')) { var encodedUrlMap = ''; if (video_info.hasOwnProperty('url_encoded_fmt_stream_map')) { encodedUrlMap = encodedUrlMap + ',' + video_info['url_encoded_fmt_stream_map'][0]; } if (video_info.hasOwnProperty('adaptive_fmts')) { encodedUrlMap = encodedUrlMap + ',' + video_info['adaptive_fmts'][0]; } if (encodedUrlMap.indexOf('rtmpe%3Dyes') !== -1) { return fail('rtmpe downloads are not supported'); } var arr = encodedUrlMap.split(','); var size = arr.length; for (var i = 0; i < size; i++) { if (arr[i].length == 0) { continue; } var urlData = parseQS(arr[i]); if (!urlData.hasOwnProperty('itag') || !urlData.hasOwnProperty('url')) { continue; } var formatId = urlData['itag'][0]; var url = urlData['url'][0]; if (url.indexOf('ratebypass') === -1) { url += '&ratebypass=yes'; } if (urlData.hasOwnProperty('sig')) { url += '&signature=' + urlData['sig'][0]; formats.push({ format_id: formatId, url: url }); } else if (urlData.hasOwnProperty('s')) { var encrypted_sig = urlData['s'][0]; var ASSETS_RE = /"assets":.+?"js":\s*("[^"]+")/; var jsplayer_url_json = searchRegex(ASSETS_RE, age_gate ? embed_webpage : video_webpage); // TODO: /* if not jsplayer_url_json and not age_gate: # We need the embed website after all if embed_webpage is None: embed_url = proto + '://www.youtube.com/embed/%s' % video_id embed_webpage = self._download( embed_url, video_id, 'Downloading embed webpage') jsplayer_url_json = self._searchRegex( ASSETS_RE, embed_webpage, 'JS player URL') */ var playerUrl = JSON.parse(jsplayer_url_json); (function (encrypted_sig, playerUrl, formatId, url) { queue(function () { return decryptSignature(encrypted_sig, playerUrl).done(function (signature) { url += '&signature=' + signature; formats.push({ format_id: formatId, url: url }); }); }); })(encrypted_sig, playerUrl, formatId, url); } else if (url.indexOf('signature') !== -1) { // already decrypted formats.push({ format_id: formatId, url: url }); } } } else if (video_info.hasOwnProperty('hlsvp')) { return fail('HLS not supported'); } else { return fail('no conn, hlsvp or url_encoded_fmt_stream_map information found in video info'); } var dashManifestUrl = null; function buildResult(formats) { var size = formats.length; for (var i = 0; i < size; i++) { var fmt = formats[i]; var master = YT_FORMATS[fmt['format_id']]; $.extend(fmt, master); } return { 'id': video_id, 'uploader': video_uploader, 'uploader_id': video_uploader_id, 'upload_date': upload_date, 'title': video_title, 'thumbnail': 'https://i.ytimg.com/vi/' + video_id + '/hqdefault.jpg', //'description': video_description, //'categories': video_categories, subtitles: videoSubtitles, //'automatic_captions': automatic_captions, 'duration': video_duration, 'age_limit': age_gate ? 18 : 0, //'annotations': video_annotations, 'webpage_url': 'https://www.youtube.com/watch?v=' + video_id, 'view_count': view_count, //'average_rating': float_or_none(video_info.get('avg_rating', [None])[0]), 'formats': formats, dash_manifest_url: dashManifestUrl } } // Look for the DASH manifest if (video_info.hasOwnProperty('dashmpd')) { dashManifestUrl = video_info['dashmpd'][0]; queue(function () { return parseDashManifest(video_id, dashManifestUrl, playerUrl, age_gate).done(function (dash) { if (dash != null) { dashManifestUrl = dash.dash_manifest_url; } deferred.resolve(buildResult(dash && dash.formats ? formats.concat(dash.formats) : formats)); }); }); } return deferred.promise(); } function extract(url) { var deferred = $.Deferred(); var video_id = extractId(url); // Get video webpage url = 'https://www.youtube.com/watch?v=' + video_id + '&gl=US&hl=en&has_verified=1&bpctr=9999999999'; download(url).done(function (video_webpage) { var age_gate = false; if (/player-age-gate-content">/i.test(video_webpage)) { age_gate = true; // We simulate the access to the video from www.youtube.com/v/{video_id} // this can be viewed without login into Youtube url = 'https://www.youtube.com/embed/' + video_id; download(url).done(function (embed_webpage) { var sts = searchRegex(/"sts"\s*:\s*(\d+)/, embed_webpage); var videoInfoUrl = 'https://www.youtube.com/get_video_info?video_id=' + video_id + '&eurl=' + encodeURIComponent('https://youtube.googleapis.com/v/' + video_id) + '&sts=' + sts; download(videoInfoUrl).done(function (video_info_webpage) { var video_info = parseQS(video_info_webpage); extractSupport(video_id, video_webpage, age_gate, embed_webpage, video_info).done(function (result) { deferred.resolve(result); }); }); }); } else { age_gate = false; var videoInfoUrl = 'https://www.youtube.com/get_video_info?&video_id=' + video_id + '&el=detailpage&ps=default&eurl=&gl=US&hl=en' download(videoInfoUrl).done(function (video_info_webpage) { var videoInfo = parseQS(video_info_webpage); extractSupport(video_id, video_webpage, age_gate, '', videoInfo).done(function (result) { deferred.resolve(result); }); }).fail(function () { deferred.resolve(null); }); } }); return deferred.promise(); } function search(q) { var deferred = $.Deferred(); var url = 'http://www.youtube.com/results?search_query=' + encodeURIComponent(q) + '&hl=en'; download(url).done(function (html) { var re = /<h3 class="yt-lockup-title"><a href="\/watch\?v=(.*?)".*? title="(.*?)".*? Duration: (.*?)\.<\/span>.*?by <a href="\/user\/(.*?)".*?<li>([\d,]*) views<\/li>/ig; var m = null; var results = []; while (m = re.exec(html)) { var videoId = m[1]; var r = { id: videoId, url: 'https://www.youtube.com/watch?v=' + videoId, title: m[2], duration: m[3], user: m[4], views: parseInt(m[5].replace(/,/g, ''), 10), thumbnail: 'https://i.ytimg.com/vi/' + videoId + '/mqdefault.jpg', }; results.push(r); } deferred.resolve(results); }); return deferred.promise(); } function ytAutocompleteSource(request, response) { // setup global object if (typeof google === 'undefined') { google = { sbox: { p50: function (data) { data = data[1]; var result = []; var size = data.length; for (var i = 0; i < size; i++) { result.push(data[i][0]); } google.sbox.response(result); }, response: function (data) { log(data) } } } } // set global response google.sbox.response = response; $.ajax({ url: 'https://clients1.google.com/complete/search?client=youtube&hl=en&gl=us&gs_rn=23&gs_ri=youtube&ds=yt&cp=2&gs_id=8', dataType: 'script', data: { q: request.term, callback: 'google.sbox.p50' }, success: function (data) { // ignore } }); } // END YT-LINKS CODE // // START WAITFORKEYELEMENTS CODE // /*--- waitForKeyElements(): A utility function, for Greasemonkey scripts, that detects and handles AJAXed content. Usage example: waitForKeyElements ( "div.comments" , commentCallbackFunction ); //--- Page-specific function to do what we want when the node is found. function commentCallbackFunction (jNode) { jNode.text ("This comment changed by waitForKeyElements()."); } IMPORTANT: This function requires your script to have loaded jQuery. */ function waitForKeyElements ( selectorTxt, /* Required: The jQuery selector string that specifies the desired element(s). */ actionFunction, /* Required: The code to run when elements are found. It is passed a jNode to the matched element. */ bWaitOnce, /* Optional: If false, will continue to scan for new elements even after the first match is found. */ iframeSelector /* Optional: If set, identifies the iframe to search. */ ) { var targetNodes, btargetsFound; if (typeof iframeSelector == "undefined") targetNodes = $(selectorTxt); else targetNodes = $(iframeSelector).contents () .find (selectorTxt); if (targetNodes && targetNodes.length > 0) { btargetsFound = true; /*--- Found target node(s). Go through each and act if they are new. */ targetNodes.each ( function () { var jThis = $(this); var alreadyFound = jThis.data ('alreadyFound') || false; if (!alreadyFound) { //--- Call the payload function. var cancelFound = actionFunction (jThis); if (cancelFound) btargetsFound = false; else jThis.data ('alreadyFound', true); } } ); } else { btargetsFound = false; } //--- Get the timer-control variable for this selector. var controlObj = waitForKeyElements.controlObj || {}; var controlKey = selectorTxt.replace (/[^\w]/g, "_"); var timeControl = controlObj [controlKey]; //--- Now set or clear the timer as appropriate. if (btargetsFound && bWaitOnce && timeControl) { //--- The only condition where we need to clear the timer. clearInterval (timeControl); delete controlObj [controlKey] } else { //--- Set a timer, if needed. if ( ! timeControl) { timeControl = setInterval ( function () { waitForKeyElements ( selectorTxt, actionFunction, bWaitOnce, iframeSelector ); }, 300 ); controlObj [controlKey] = timeControl; } } waitForKeyElements.controlObj = controlObj; } // END WAITFORKEYELEMENTS CODE //