Greasy Fork is available in English.
Download audio from Disney+
// ==UserScript== // @name Disney+ Audio Downloader // @name:fr Disney+ Audio Downloader // @namespace https://greasyfork.org/users/572942-stegner // @homepage https://greasyfork.org/scripts/405994-disney-audio-downloader // @description Download audio from Disney+ // @description:fr Télécharger l'audio de Disney+ // @version 1.9 // @author stegner // @license MIT; https://opensource.org/licenses/MIT // @match https://www.disneyplus.com/* // @grant none // @run-at document-start // ==/UserScript== (function(open, send) { 'use strict'; var debug = (location.hash=="#debug"); debuglog("Script loaded : Disney+ Audio Downloader"); function init(){ debuglog("Document state : "+document.readyState); if (document.readyState == "complete" || document.readyState == "loaded"){ start(); debuglog("Already loaded"); } else { if (window.addEventListener) { window.addEventListener("load", start, false); debuglog("Onload method : addEventListener"); } else if (window.attachEvent) { window.attachEvent("onload", start); debuglog("Onload method : attachEvent"); } else { window.onload = start; debuglog("Onload method : onload"); } } document.listen=true; } function start(){ debuglog("start"); if (typeof document.initaudio !== "undefined") { document.initaudio(); } if (typeof document.initsub !== "undefined") { document.initsub(); } listensend(); document.handleinterval = setInterval(buttonhandle,100); } if(!document.listen){ init(); } document.initaudio = function(){ debuglog("initaudio"); document.audios = []; document.content = new Uint8Array(); document.baseurl=""; document.m3u8found=false; document.wait=false; document.downloading=false; document.filename=""; document.episode=""; document.audioid=null; // Add download icon document.styleSheets[0].addRule('#audioTrackPicker > div:before','content:"";color:#fff;padding-right:35px;padding-top:2px;background:url() no-repeat right;width:20px;height:18px;position:absolute;top:6px;right:10px;opacity:0.6;cursor:pointer;'); document.styleSheets[0].addRule('#audioTrackPicker > div:hover:before','opacity:1;'); } // Catch M3U8 files function listensend(){ debuglog("listensend"); var newOpen = function(...args) { if(!document.m3u8found && args.length>=2){ if(args[1].indexOf(".m3u8")>0 && document.url!=args[1]) { // m3u8 url debuglog("m3u8 found : "+args[1]); document.url = args[1]; document.langs = []; document.baseurl=document.url.substring(0,document.url.lastIndexOf('/')+1); document.m3u8found=true; getpagecontent(m3u8loaded,document.url); } } open.call(this,...args); } var newSend = function(...args) { if(args[0] && args[0].match && args[0].match(/globalization/)){ this.addEventListener('readystatechange', function(e) { try { document.globalization = JSON.parse(e.target.response).data.globalization; } catch(e) {} }, false); } send.call(this,...args); } if(typeof unsafeWindow !== "undefined"){ debuglog("Window state : unsafe"); var define = Object.defineProperty; define(unsafeWindow.XMLHttpRequest.prototype, "open", {value: exportFunction(newOpen, window)}); define(unsafeWindow.XMLHttpRequest.prototype, "send", {value: exportFunction(newSend, window)}); } else { debuglog("Window state : safe"); XMLHttpRequest.prototype.open = newOpen; XMLHttpRequest.prototype.send = newSend; } } function m3u8loaded(response) { debuglog("m3u8loaded"); if (typeof document.m3u8sub !== "undefined") { document.m3u8sub(response); } if (typeof document.m3u8audio !== "undefined") { document.m3u8audio(response); } } document.m3u8audio = function(response){ var lines = response.split('#'); var found = false; if(lines[2].indexOf("EXT-X-INDEPENDENT-SEGMENTS")==0){ // Audio tracks list var quality=null; lines.forEach(function(line) { if(line.indexOf('TYPE=AUDIO')>0) { var lang = linetoarray(line); lang.LOCALIZED = document.globalization.audio.find(t => t.language == lang.LANGUAGE); // audio infos if(line.indexOf('GROUP-ID="eac-3"')>0 && (quality==null||quality=="eac-3")){ quality="eac-3"; document.audios.push(lang); debuglog("Audio found : "+document.audios[document.audios.length-1].NAME); } else if(line.indexOf('GROUP-ID="aac-128k"')>0 && (quality==null||quality=="aac-128k")){ quality="aac-128k"; document.audios.push(lang); debuglog("Audio found : "+document.audios[document.audios.length-1].NAME); } else if(line.indexOf('GROUP-ID="aac-64k"')>0 && (quality==null||quality=="aac-64k")){ quality="aac-64k"; document.audios.push(lang); debuglog("Audio found : "+document.audios[document.audios.length-1].NAME); } } }); } else if(response.indexOf(".mp4a")>0) { downloadmp4a(response); } } function downloadmp4a(m3u8data){ debuglog("downloadmp4a"); var lines = m3u8data.split(/\r?\n/g); var mapfound=false; var percent; var i=0; document.downloadInterval = setInterval(function () { var line = lines[i]; var url=null; if(line!=null){ var uri = document.audios[document.audioid].URI; if(line.indexOf("map.mp4a")>0 && !mapfound){ // Get mp4a map debuglog("Download map"); url = document.baseurl+uri.substring(0,uri.lastIndexOf("/")+1)+line.substring(line.indexOf('"')+1,line.lastIndexOf('"')); mapfound=true; } else if(line.indexOf("_000.mp4a")>0 && line.indexOf("MAIN")>0){ // Get mp4a data url = document.baseurl+uri.substring(0,uri.lastIndexOf("/")+1)+line; } if(url!=null && !document.downloading){ // Download file getpagecontent(mp4aloaded,url,true); document.downloading=true; i++; } else if(url==null){ // Skip line i++; } if(percent!=Math.round((i/lines.length)*100)){ percent=Math.round((i/lines.length)*100); document.styleSheets[0].addRule('#audioTrackPicker > div:nth-child('+(document.audioid+1)+'):before','content:"'+percent+'%";'); } } else { // Download finished clearInterval(document.downloadInterval); document.styleSheets[0].addRule('#audioTrackPicker > div:nth-child('+(document.audioid+1)+'):before','content:"";'); exportblob(document.content, 'video/mp4'); document.content=new Uint8Array(); document.wait=false; } },10); } function mp4aloaded(response) { debuglog("mp4aloaded"); document.downloading=false; document.content=appendbuffer(document.content,response); } function linetoarray(line) { var r###lt = []; var values = line.split(','); values.forEach(function(value) { var data = value.replace(/\r\n|\r|\n/g,'').split('='); if(data.length>1) { var key = data[0]; var content = data[1].replace(/"/g,''); r###lt[key]=content; } }); return r###lt; } function buttonhandle() { var buttons = document.getElementsByClassName("control-icon-btn"); if(buttons.length>0) { if (typeof document.clickhandl###b !== "undefined") { document.clickhandl###b(); } if (typeof document.clickhandleaudio !== "undefined") { document.clickhandleaudio(); } document.filename = document.getElementsByClassName("title-field")[0]?.innerText; if(document.getElementsByClassName("subtitle-field").length>0) { document.episode = document.getElementsByClassName("subtitle-field")[0]?.innerText } } if(document.oldlocation!=window.location.href&&document.oldlocation!=null) { // location changed document.m3u8found=false; document.audios = []; document.langs = []; } document.oldlocation=window.location.href; } document.clickhandleaudio = function() { var picker = document.getElementsByClassName("options-picker audio-track-picker"); if(picker && picker[0]) { picker[0].childNodes.forEach(function(child) { var element = child.childNodes[0]; var lang = element.childNodes[1].innerHTML; if(child.onclick==null) { child.onclick = selectaudio; } }); } } function selectaudio(e) { debuglog("selectaudio"); var width = this.offsetWidth; // Check click position if(e.layerX>=width-30&&e.layerX<=width-10&&e.layerY>=5&&e.layerY<=25){ // Download audio download(this.childNodes[0].childNodes[1].innerHTML); // Cancel selection return false; } } function download(langname) { if(!document.wait){ debuglog("Download audio : "+langname); var count=0; document.audios.forEach(function(audio) { if(audio.LOCALIZED && Object.values(audio.LOCALIZED.renditions).includes(langname)) { document.audioid=count; document.ad=(audio.NAME.indexOf("[Audio Description]")>0); getpagecontent(m3u8loaded,document.baseurl+audio.URI); document.wait=true; } count++; }); if(count==0){ alert("An error has occurred, please reload the page."); } } } function getpagecontent(callback,url,binary) { debuglog("Downloading : "+url); var http=new XMLHttpRequest(); http.open("GET", url, true); if(binary){ http.responseType = "arraybuffer"; } http.onloadend = function() { if(http.readyState == 4 && http.status == 200) { if(binary){ callback(http.response); } else { callback(http.responseText); } } else if (http.status === 404) { debuglog("Not found"); callback(""); } else { debuglog("Unknown error, retrying"); setTimeout(function () { getpagecontent(callback,url,binary); },100); } } http.send(); } function appendbuffer(buffer1, buffer2) { var tmp = new Uint8Array(buffer1.byteLength + buffer2.byteLength); tmp.set(new Uint8Array(buffer1), 0); tmp.set(new Uint8Array(buffer2), buffer1.byteLength); return tmp; }; // Save file as arraybuffer function exportblob(data, mimeType) { debuglog("exportblob"); var blob, url; var output = document.filename; if(document.episode!="") { output+= " - "+document.episode.replace(':',''); } output += "."+document.audios[document.audioid].LANGUAGE; if(document.ad){ output +=".ad"; } output += ".mp4"; blob = new Blob([data], { type: mimeType }); url = window.URL.createObjectURL(blob); downloadurl(url, output); setTimeout(function() { return window.URL.revokeObjectURL(url); }, 1000); }; function downloadurl(data, fileName) { debuglog("Save audio"); var a; a = document.createElement('a'); a.href = data; a.download = fileName; document.body.appendChild(a); a.style = 'display: none'; a.click(); a.remove(); }; function debuglog(message){ if(debug){ console.log("%c [debug] "+message, 'background: #222; color: #bada55'); } } })(XMLHttpRequest.prototype.open, XMLHttpRequest.prototype.send);