// ==UserScript==
// @name Boton Descarga MP3 con Inicio y Final para videos de Youtube
// @author ELDiDi
// @version 1.3
// @description Al cargar un video de Youtube se agrega un Boton para descargar el MP3 de manera directa haciendo uso del servicio (http://youtubeinmp3.com/api/). Ademas puedes indicar el Inicio y Final del audio para evitar partes del video no deseadas como dialogos o intros.
// @match https://www.youtube.com/*
// @require http://ajax.googleapis.com/ajax/libs/jquery/1.7.2/jquery.min.js
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @run-at document-end
// @icon 
// @namespace https://greasyfork.org/users/73478
// ==/UserScript==
/*
--- 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;
}
// Fuerza la ejecucion del script para resolver el problema de tener que refrescar pagina para hacerlo funcionar
waitForKeyElements("#watch7-views-info", ObtenerInfoDelVideo);
// Comprueba el link por primera vez, obtiene la duracion del video, titulo enlace de descarga.
function ObtenerInfoDelVideo () {
// Obtiene URL del video
var VideoURL = document.URL.split("&")[0];
// AJAX Load link
var url = '//youtubeinmp3.com/fetch/?format=json&video='+VideoURL+"&hq=1";
var method = 'GET';
var xmlhttp = new XMLHttpRequest();
if (!("withCredentials" in xmlhttp) && typeof XDomainRequest != "undefined") {
// XDomainRequest for IE.
xmlhttp = new XDomainRequest();
xmlhttp.open(method, url);
} else {
// xmlhttp for Chrome/Firefox/Opera/Safari.
xmlhttp.open(method, url, true);
}
// Old style
if (!xmlhttp) {
xmlhttp = new XMLHttpRequest();
xmlhttp.open(method, url, true);
}
//Respuesta de la solicitud por AJAX
xmlhttp.onload = function () {
if (xmlhttp.readyState==4 && xmlhttp.status==200) {
var response = xmlhttp.responseText;
// Si la respuesta contiene la palabra titulo signigica que el servicio puede convertir el video, caso contrario muestra mensaje.
if(response.indexOf("title") == -1){
InsertDiv('Lamentablemente este video no puede ser convertido a MP3');
}else{
var JSONObj = JSON.parse(response);
var VideoTitle = JSONObj.title;
var VideoLength = JSONObj.length-1;
var VideoDownloadLink = JSONObj.link;
//Se declaran variables Globales
window.VideoURL = VideoURL; //Global
window.VideoTitle = VideoTitle;
window.VideoLength = VideoLength;
window.VideoDownloadLink = VideoDownloadLink;
InsertDiv();
}
}
};
xmlhttp.onerror = function () {
InsertDiv('Ocurrio un error al obtener datos del video. Por favor reintente la operacion.');
};
xmlhttp.send();
} // END FUNCTION ObtenerLink
function InsertDiv(Error){
// Si se recibe Error, muestra el mensaje, caso contrario muestra Campos Inicio-Final y boton Descargar
var DivNode = document.createElement('div'); //crea el elemento DIV
DivNode.id = 'id_div';
if (Error){
DivNode.style = 'padding:10px;';
DivNode.innerHTML = '<strong>'+Error+'</strong>';
DivNode.innerHTML += '<br>Puedes intentar hacerlo manualmente desde <a href="http://www.youtubeinmp3.com/download/?video='+ VideoURL +'" target="_blank">aqui</a>'; // y le mete el contenido
}else{
//Se agrega STYLE a la cabezera de la pagina
var head, style;
head = document.getElementsByTagName('head')[0];
if (head){
style = document.createElement('style');
style.type = 'text/css';
style.innerHTML = '#InicioTexto:hover, #FinalTexto:hover{text-decoration: underline;}';
style.innerHTML += '#InicioTexto,#FinalTexto{cursor:pointer;}';
style.innerHTML += '#startInput, #endInput{padding:3px;border-width:1px;border-color:#cccccc;border-style:solid;font-size:12px;text-align:center;}';
style.innerHTML += '#DivForWarnings {padding:5px;border-width:0px;font-size:12px;color:red;text-align:center;}';
head.appendChild(style);
}
DivNode.style = 'overflow: hidden;';
DivNode.style = 'display:inline-block';
DivNode.style = 'height:auto';
//var htmlDuracion = 'Duracion: '+VideoLength+' seg. ';
var htmlDuracion = '';
var htmlInput1 = '<span id="InicioTexto" title="Haga click aqui para obtener la posicion actual del video.">Inicio:</span> <input id="startInput" onClick="this.select();" type="text" size="4"> ';
var htmlInput2 = '<span id="FinalTexto" title="Haga click aqui para obtener la posicion actual del video.">Final:</span> <input id="endInput" onClick="this.select();" type="text" size="4">';
var iframeSource = '//www.youtubeinmp3.com/es/widget/button/?video='+VideoURL+'';
var iframe = '<iframe id="IframeBotonDescargar" style="width:150px;height:30px;border:0;padding-top: 5px; margin-left: 10px; margin-bottom: -7px;;" scrolling="no" src="'+iframeSource+'"></iframe>';
var DivForWarnings = '<br><div id="DivForWarnings" style="display:none1"> </div>';
DivNode.innerHTML = htmlDuracion + htmlInput1 + htmlInput2 + iframe + DivForWarnings;
}
// Add to page
var parentElement=document.getElementById('watch7-views-info');
if (parentElement) {
var DivDOM = document.getElementById('id_div');
if (DivDOM) {
console.log('El DOM ya estaba en la pagina.');
DivDOM.parentElement.removeChild(DivDOM);
console.log('El DOM fue removido.');
}
parentElement.appendChild(DivNode);
console.log('El DOM ha sido incrustado.');
//Se establecen funciones para los elementos Inicio-Final
document.getElementById('InicioTexto').addEventListener('click', ObtenerPocisionInicio, false);
document.getElementById('FinalTexto').addEventListener('click', ObtenerPocisionFinal, false);
var InicioInput = document.getElementById('startInput');
var FinalInput = document.getElementById('endInput');
InicioInput.addEventListener("focus", DisableDownloadBtn, false);
InicioInput.addEventListener('focusout', OnChangeInput, false);
InicioInput.addEventListener("keyup", FormatearTiempo, false);
FinalInput.addEventListener("focus", DisableDownloadBtn, false);
FinalInput.addEventListener('focusout', OnChangeInput, false);
FinalInput.addEventListener("keyup", FormatearTiempo, false);
InicioInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(0);
FinalInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(VideoLength);
}else{
alert('ERROR: No se encuentra el lugar adecuado para agregar el boton de descarga. Youtube debe haber cambiado algo en el sitio.');
return;
}
} //end function InsertDiv
function ObtenerPocisionInicio() {
var Reproductor = document.getElementById("movie_player");
var TiempoObtenido = Reproductor.getCurrentTime();
var TiempoObtenido_ConvertidoConPuntos = ConvertirSegundosSinPuntosAMiuntosConDosPuntos( Math.floor(TiempoObtenido) );
InicioInput = document.getElementById("startInput");
InicioInput.value = TiempoObtenido_ConvertidoConPuntos;
OnChangeInput();
}
function ObtenerPocisionFinal(){
var Reproductor = document.getElementById("movie_player");
var TiempoObtenido = Reproductor.getCurrentTime();
var TiempoObtenido_ConvertidoConPuntos = ConvertirSegundosSinPuntosAMiuntosConDosPuntos( Math.floor(TiempoObtenido) );
FinalInput = document.getElementById("endInput");
FinalInput.value = TiempoObtenido_ConvertidoConPuntos;
OnChangeInput();
}
function OnChangeInput(){
var InicioInput = document.getElementById('startInput');
var FinalInput = document.getElementById('endInput');
var TiempoInicio = ConvertirASegundosSinPuntos(InicioInput.value);
var TiempoFinal = ConvertirASegundosSinPuntos(FinalInput.value);
if (isNaN(TiempoInicio)){
InicioInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(0);
}else if (isNaN(TiempoFinal)){
FinalInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(VideoLength);
}else if (TiempoInicio == TiempoFinal){
//alert("El valor de Inicio no puede ser igual al de el Final.");
MostrarWarning('El valor de Inicio no puede ser igual al del Final.');
InicioInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(0);
FinalInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(VideoLength);
}else if (TiempoInicio < 0 || TiempoInicio > VideoLength){
//alert("El valor de Inicio es incorrecto.");
MostrarWarning('El valor de Inicio es incorrecto.');
InicioInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(0);
}else{
if (TiempoFinal < TiempoInicio || TiempoFinal > VideoLength){
//alert("El tiempo Final no puede ser menor al tiempo Inicial.");
MostrarWarning('El tiempo Final no puede ser menor al tiempo Inicial.');
//FinalInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(VideoLength);
}else{
GenerarBotonDescargar(VideoURL, TiempoInicio, TiempoFinal);
}
}
//alert("OnChangeInput\nVideoUrl: "+VideoURL+"\nInicioValue: "+ InicioValue +"\nFinalValue: "+FinalValue +"\nInicio: "+Inicio+"\nFinal: "+Final);
}
function MostrarWarning(Mensaje){
var DivForWarningsDOM = document.getElementById('DivForWarnings');
DivForWarningsDOM.innerHTML = ' ';
DivForWarningsDOM.innerHTML = Mensaje;
InicioInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(0);
FinalInput.value = ConvertirSegundosSinPuntosAMiuntosConDosPuntos(VideoLength);
setTimeout(function(){DivForWarningsDOM.innerHTML=' ';}, 3000);
}
function GenerarBotonDescargar(VideoURLParameter, InicioParameter, FinalParameter){
//alert("Se va a generar el boton con los siguientes parametros:\nVideoURLParameter: "+VideoURLParameter+"\nInicioParameter: "+InicioParameter+"\nFinalParameter: "+FinalParameter);
if (VideoURLParameter == null){ var VideoURLParameter = VideoURL;}
if (InicioParameter == null){ var InicioParameter = ConvertirASegundosSinPuntos(document.getElementById('startInput').value);}
if (FinalParameter == null){ var FinalParameter = ConvertirASegundosSinPuntos(document.getElementById('endInput').value);}
//var iframeSource = '//www.youtubeinmp3.com/widget/button/?video='+VideoURLParameter+'&start='+InicioParameter+'&end='+FinalParameter+'&title='+VideoTitle+'_'+InicioParameter+'_'+FinalParameter;
var iframeSource = '//www.youtubeinmp3.com/widget/button/?video='+VideoURLParameter+'&start='+InicioParameter+'&end='+FinalParameter+'';
var BotonGenerado = '<iframe id="IframeBotonDescargar" style="width:150px;height:30px;border:0;overflow:hidden;" scrolling="no" src="'+iframeSource+'">';
var iframeDOM = document.getElementById('IframeBotonDescargar');
iframeDOM.src = iframeSource;
}
function DisableDownloadBtn(){
var Iframe = document.getElementById('IframeBotonDescargar');
//Iframe.src = 'about:blank';
var html_code= '<!DOCTYPE html><html><body style="background-color:#D8D8D8;"></body></html>';
Iframe.src = "data:text/html;charset=utf-8," + escape(html_code);
}
function FormatearTiempo(e) {
//alert(VideoLength);
if (VideoLength>=3600){ // Si los segundos son igual o mayor que 3600 (60 minutos o mas) FORMATO SERA 1:00:00
var r = /([0-9]{1})([0-9]{2})([0-9]{2})/i, str = e.target.value.replace(/[^0-9]/ig, "");
while (r.test(str)) {
str = str.replace(r, '$1' + ':' + '$2' + ':' + '$3');
}
e.target.value = str.slice(0, 7);
}else if (VideoLength>=600){ // Si los segundos son igual o mayor o que 600 (10 minutos o mas) FORMATO SERA 10:00
var r = /([0-9]{2})([0-9]{2})/i, str = e.target.value.replace(/[^0-9]/ig, "");
while (r.test(str)) {
str = str.replace(r, '$1' + ':' + '$2');
}
e.target.value = str.slice(0, 5);
}else{ // Si los segundos son menores que 600 (9:59 minutos o menos) FORMATO 0:00
var r = /([0-9]{1})([0-9]{2})/i,
str = e.target.value.replace(/[^0-9]/ig, "");
while (r.test(str)) {
str = str.replace(r, '$1' + ':' + '$2');
}
e.target.value = str.slice(0, 4);
}
};
function ConvertirASegundosSinPuntos(MinutosConDosPuntosParameter){ // ejemplo: 3:47 transformado en 227
var str = MinutosConDosPuntosParameter;
var p = str.split(':'),
s = 0, m = 1, h = 2;
while (p.length > 0) {
s += m * parseInt(p.pop(), 10);
m *= 60;
}
//alert(MinutosConDosPuntosParameter + " equivalen a "+ s +" segundos.");
return s;
}
function ConvertirSegundosSinPuntosAMiuntosConDosPuntos(SegundosSinPuntosParameter){ // ejemplo: 227 transformado en 3:47
var Horas = Math.floor(SegundosSinPuntosParameter / 3600);
var Minutos = Math.floor(( SegundosSinPuntosParameter - Horas * 3600) / 60);
var Segundos = (SegundosSinPuntosParameter) - (Horas * 3600) - (Minutos * 60);
if(VideoLength >= 3600){ // 3600 segundos o mas - 60 minutos o mas - Formato 1:00:00
(Minutos.toString().length < 2) ? Minutos = "0" + Minutos : Minutos = Minutos;
(Segundos.toString().length < 2) ? Segundos = "0" + Segundos : Segundos = Segundos;
DuracionConDosPuntos = Horas +":"+ Minutos +":"+ Segundos;
}else if (VideoLength >= 600){ // 600 segundos o mas - 10 minutos o mas - Formato 10:00
(Minutos.toString().length < 2) ? Minutos = "0" + Minutos : Minutos = Minutos;
(Segundos.toString().length < 2) ? Segundos = "0" + Segundos : Segundos = Segundos;
DuracionConDosPuntos = Minutos +":"+ Segundos;
}else{ // 599 o menos segundos - menos de 10 minutos - Formato 0:00
(Minutos.toString().length < 2) ? Minutos = Minutos : Minutos = Minutos;
(Segundos.toString().length < 2) ? Segundos = "0"+Segundos : Segundos = Segundos ;
DuracionConDosPuntos = Minutos +":"+ Segundos;
}
return DuracionConDosPuntos;
}