// ==UserScript==
// @name Arte+7 Downloader
// @namespace GuGuss
// @description Download videos or get stream link of ARTE programs in the selected language.
// @include https://*.arte.tv/*
// @version 2.7
// @updateURL https://github.com/GuGuss/ARTE-7-Downloader/raw/master/arte-downloader.user.js
// @grant GM_xmlhttpRequest
// @icon http://www.arte.tv/favicon.ico
// ==/UserScript==
/*
Support:
- new 360 HTML5 player: http://info.arte.tv/fr/wasala-arriver
- old 360 flash player: http://future.arte.tv/fr/5-metres-une-plongee-360deg-sur-votre-ordinateur
*/
/* --- GLOBAL VARIABLES --- */
var scriptVersion = "2.6";
var player = [];
var nbVideoPlayers = 0;
var is_playlist = false;
var playerJson;
var nbVideos;
var nbHTTP;
var nbRTMP;
var nbHLS;
var languages;
var qualities;
var videoPlayerElements;
var videoPlayerClass = {
'standard': 'arte_vp_url',
'oembed': 'arte_vp_url_oembed',
'general': 'video-container',
'live-oembed': 'arte_vp_live-url-oembed',
'generic': 'data-url',
'teaser': 'data-teaser-url'
};
var videoPlayerClassEmbedded = {
'story': 'embed.embed--delay iframe',
'embedded': 'media_embed iframe'
};
/* --- FUNCTIONS: utilities --- */
function getURLParameter(url, name) {
return decodeURIComponent((new RegExp('[?|&]' + name + '=' + '([^&;]+?)(&|#|;|$)').exec(url) || [, ""])[1].replace(/\+/g, '%20')) || null;
}
function insertAfter(newNode, referenceNode) {
referenceNode.parentNode.insertBefore(newNode, referenceNode.nextSibling);
}
function stringStartsWith(string, prefix) {
return string.slice(0, prefix.length) === prefix;
}
function hasClass(element, cls) {
return (' ' + element.className + ' ').indexOf(' ' + cls + ' ') > -1;
}
// Get a parent node of the chosen type and class
function getParent(nodeReference, nodeName, classString) {
var parent = node;
var nbNodeIteration = 0;
var nbNodeIterationMax = 10;
// any node
if (nodeName === '') {
console.log("> Looking for a parent node with class '" + classString + "'");
while (parent.nodeName !== "BODY" &&
nbNodeIteration < nbNodeIterationMax &&
hasClass(parent, classString) === false) {
nbNodeIteration++;
parent = parent.parentNode;
}
}
// with defined node type
else {
console.log("> Looking for a <" + nodeName + " class='" + classString + "'> parent node");
while (parent.nodeName !== "BODY" &&
nbNodeIteration < nbNodeIterationMax &&
(parent.nodeName !== nodeName.toUpperCase() || hasClass(parent, classString) === false)) {
nbNodeIteration++;
parent = parent.parentNode;
}
}
return parent;
}
/* --- FUNCTIONS: analysis --- */
function addLanguage(videoElementIndex, language, wording) {
if (!languages[videoElementIndex].hasOwnProperty(language)) {
languages[videoElementIndex][language] = wording;
}
}
function addQuality(videoElementIndex, quality, wording) {
if (!qualities[videoElementIndex].hasOwnProperty(quality)) {
qualities[videoElementIndex][quality] = wording;
}
}
function preParsePlayerJson(videoElementIndex) {
if (playerJson[videoElementIndex]) {
var videos = Object.keys(playerJson[videoElementIndex]["videoJsonPlayer"]["VSR"]);
nbVideos[videoElementIndex] = videos.length;
// Loop through all videos URLs.
for (var key in videos) {
var video = playerJson[videoElementIndex]["videoJsonPlayer"]["VSR"][videos[key]];
// Check if video format or media type
if (video["videoFormat"] === "HBBTV" || video["mediaType"] === "mp4") {
nbHTTP[videoElementIndex]++;
} else if (video["videoFormat"] === "RMP4" || video["mediaType"] === "rtmp") {
nbRTMP[videoElementIndex]++;
} else if (video["videoFormat"] === "M3U8" || video["mediaType"] === "hls") {
nbHLS[videoElementIndex]++;
}
addLanguage(videoElementIndex, video["versionCode"], video["versionLibelle"]);
addQuality(videoElementIndex, (
video["VQU"] !== undefined ? video["VQU"] : video["quality"]),
video["height"] ? video["height"] + "p@" + video["bitrate"] + "bps" : video["quality"]);
}
// Remove Apple HLS if HTTP available
if (nbHTTP[videoElementIndex] > 0) {
delete qualities[videoElementIndex]["XS"];
delete qualities[videoElementIndex]["XQ"];
}
// Reorder qualities
var sortedKeys = Object.keys(qualities[videoElementIndex]).sort(
function(a, b) {
// array of sorted keys
return qualities[videoElementIndex][b].split('@')[1].split('b')[0] * 1 - qualities[videoElementIndex][a].split('@')[1].split('b')[0] * 1;
}
);
// Create new object to rearrange qualities according to new key order
var temp = new Object;
for (var i = 0; i < sortedKeys.length; i++) {
temp[sortedKeys[i]] = qualities[videoElementIndex][sortedKeys[i]];
}
qualities[videoElementIndex] = temp; // replace with new ordered object
// Display preparse info
console.log("\n====== player #" + videoElementIndex + " ======\n> " +
nbVideos[videoElementIndex] + " formats: " + nbHTTP[videoElementIndex] + " HTTP videos | " + nbRTMP[videoElementIndex] + " RTMP streams | " + nbHLS[videoElementIndex] + " HLS streams.");
var languagesFound = "";
for (l in languages[videoElementIndex]) {
languagesFound += "\n - " + languages[videoElementIndex][l];
}
console.log("> Languages:" + languagesFound);
}
}
function parsePlayerJson(playerJsonUrl, videoElement, videoElementIndex) {
console.log(" - #" + videoElementIndex + " player JSON: " + playerJsonUrl);
GM_xmlhttpRequest({
method: "GET",
url: playerJsonUrl,
onload: function(response) {
playerJson[videoElementIndex] = JSON.parse(response.responseText);
preParsePlayerJson(videoElementIndex);
decoratePlayer(videoElement, videoElementIndex);
}
});
}
function getVideoName(videoElementIndex) {
var name = playerJson[videoElementIndex]['videoJsonPlayer']['VTI'];
if (name === null) {
name = playerJson[videoElementIndex]['videoJsonPlayer']['VST']['VNA'];
if (name === null) {
return "undefined";
}
}
name = name.split('_').join(' ');
return name.charAt(0).toUpperCase() + name.slice(1);
}
function getVideoUrl(videoElementIndex, quality, language) {
// Get videos object
var videos = Object.keys(playerJson[videoElementIndex]["videoJsonPlayer"]["VSR"]);
// Check if there are HTTP videos
if (nbHTTP[videoElementIndex] > 0) {
// Loop through all videos URLs.
for (var key in videos) {
var video = playerJson[videoElementIndex]["videoJsonPlayer"]["VSR"][videos[key]];
// Check language, format, quality
if (video["versionCode"] === language &&
(video["videoFormat"] === "HBBTV" || video["mediaType"] === "mp4") &&
(video["VQU"] === quality || video["quality"] === quality)) {
console.log("> " + quality + " MP4 in " + language + ": " + video["url"]);
return video["url"];
}
}
}
// Search RTMP streams
if (nbRTMP[videoElementIndex] > 0) {
for (var key in videos) {
var video = playerJson[videoElementIndex]["videoJsonPlayer"]["VSR"][videos[key]];
if ((video["versionCode"] === language || language === undefined) &&
(video["VQU"] === quality || video["quality"] === quality) &&
(video["videoFormat"] === "RMP4" || video["mediaType"] === "rtmp")
) {
var url = video["streamer"] + video["url"];
console.log("> " + quality + " RTMP stream: " + url);
return url;
}
}
}
// Search HLS streams
if (nbHLS[videoElementIndex] > 0) {
for (var key in videos) {
var video = playerJson[videoElementIndex]["videoJsonPlayer"]["VSR"][videos[key]];
if (
(video["videoFormat"] === "M3U8" || video["mediaType"] === "hls") &&
(video["VQU"] === quality || video["quality"] === quality) &&
video["versionCode"] === language
) {
console.log("> HLS stream: " + video["url"]);
return video["url"];
}
}
}
console.log("> Video not found.")
return '';
}
function findPlayerJson(videoElement, videoElementIndex) {
// Get player URL to find its associated json
var playerUrl = null;
var jsonUrl = null;
for (key in videoPlayerClass) {
playerUrl = videoElement.getAttribute(videoPlayerClass[key]);
if (playerUrl !== null) {
break;
}
}
// oembed
if (playerUrl !== null && (key === "oembed" || (key === "live-oembed"))) {
GM_xmlhttpRequest({
method: "GET",
url: playerUrl,
onload: function(response) {
jsonUrl = unescape(response.responseText.split("json_url=")[1].split('"')[0]);
if (jsonUrl !== undefined) {
parsePlayerJson(jsonUrl, videoElement, videoElementIndex);
}
}
});
}
// http://www.arte.tv/arte_vp/
else if (stringStartsWith(location.href, "http://www.arte.tv/arte_vp/")) {
playerUrl = unescape(location.href.split("json_url=")[1]);
parsePlayerJson(playerUrl, videoElement, videoElementIndex);
}
// iframe embedded media
else if (playerUrl === null) {
playerUrl = unescape(videoElement.getAttribute('src'));
jsonUrl = playerUrl.split('json_url=')[1];
if (jsonUrl !== undefined) {
parsePlayerJson(jsonUrl, videoElement, videoElementIndex);
} else {
console.log("> Searching a 360 video in: " + playerUrl);
GM_xmlhttpRequest({
method: "GET",
url: playerUrl,
onload: function(response) {
var doc = response.responseText;
var videoName, videoURL;
// new 360 HTML5 player
if (playerUrl.indexOf("arte360") > -1) {
var playerJS = playerUrl.split("?root")[0] + "jsmin/output.min.js?" + doc.split("window.appVersion = \"")[1].split("\"")[0];
console.log("> 360 player JS: " + playerJS);
var root = getURLParameter(playerUrl, "root")
videoName = getURLParameter(playerUrl, "video")
videoURL = root + "/video/download/4K/" + videoName + "_4K.mp4";
console.log("> Video URL: " + videoURL);
var subtitlesURL = root + "/subtitles/" + videoName + "_" + getURLParameter(playerUrl, "lang") + ".srt";
console.log("> Subtitles URL: " + subtitlesURL);
decoratePlayer360(videoElement, videoURL, videoName, subtitlesURL);
}
// old 360 flash player
else if (playerUrl.indexOf("360FlashPlayers") > -1) {
console.log("> old player");
var xml = doc.split('xml:"')[1].split('"')[0];
GM_xmlhttpRequest({
method: "GET",
url: playerUrl + xml,
onload: function(response) {
xml = response.responseText;
// Get video URL
videoName = xml.split('videourl="%SWFPATH%/')[1].split('"')[0];
videoURL = playerUrl + videoName;
console.log(videoURL);
decoratePlayer360(videoElement, videoURL, videoName);
}
});
}
}
});
}
}
// Check if player URL is the json
else if (playerUrl.substring(playerUrl.length - 6, playerUrl.length - 1) === ".json") {
parsePlayerJson(playerUrl, videoElement, videoElementIndex);
}
// Look for player URL inside the player json
else {
GM_xmlhttpRequest({
method: "GET",
url: playerUrl,
onload: function(response) {
var json = JSON.parse(response.responseText);
playerUrl = json["videoJsonPlayer"]["videoPlayerUrl"];
if (playerUrl === undefined) { // not found ? Look for playlist file inside the livestream player
console.log("Video player URL not available. Fetching livestream player URL");
playerUrl = videoElement.getAttribute(videoPlayerClass['live']);
}
parsePlayerJson(playerUrl, videoElement, videoElementIndex);
}
});
}
}
function findPlayers() {
// Check playlist
playlistJson = /playlist_url=([^&]+)/.exec(window.location.href);
if (playlistJson != null) {
playlistJson=unescape(playlistJson[1]);
console.log("> Found playlist json: " + playlistJson)
console.log()
videoPlayerElements = parent.document.querySelectorAll("div.arte-playerfs.arte-playerfs--show");
GM_xmlhttpRequest({
method: "GET",
url: playlistJson,
onload: function(response) {
jsonUrl = JSON.parse(response.responseText);
//check whether exists a valid entry
if(typeof jsonUrl["videos"]!=="undefined" && typeof jsonUrl["videos"][0]!=="undefined") {
jsonUrl = jsonUrl["videos"][0]["jsonUrl"];
jsonUrl = jsonUrl.replace(/\\/g, ''); // remove backslashes from the URL
parsePlayerJson(jsonUrl, videoPlayerElements[0], 0);
}
}
});
nbVideoPlayers = videoPlayerElements.length;
return true;
} else {
// Check regular tags
for (tag in videoPlayerClass) {
videoPlayerElements = document.querySelectorAll("div[" + videoPlayerClass[tag] + "]");
if (videoPlayerElements.length > 0) {
console.log(videoPlayerClass[tag])
break;
}
}
// Check embedded tags
if (videoPlayerElements.length === 0) {
for (tag in videoPlayerClassEmbedded) {
videoPlayerElements = document.querySelectorAll("div." + videoPlayerClassEmbedded[tag]);
if (videoPlayerElements.length > 0) {
console.log(tag)
break;
}
}
}
// Check iframe
if (videoPlayerElements.length === 0) {
videoPlayerElements = document.querySelectorAll("iframe[arte-video]");
// Check 360 (no attributes yet)
if (videoPlayerElements.length === 0) {
videoPlayerElements = document.querySelectorAll("iframe");
}
}
// Check arte_vp with no parent frame
if (videoPlayerElements.length === 0 && stringStartsWith(unescape(top.location), "http://www.arte.tv/arte_vp/")) {
videoPlayerElements = document.querySelectorAll("body");
}
nbVideoPlayers = videoPlayerElements.length;
return false;
}
}
/* --- FUNCTIONS: decorating --- */
function createButtonDownload(videoElementIndex, language) {
var button = document.createElement('a');
var videoUrl;
for (q in qualities[videoElementIndex]) {
videoUrl = getVideoUrl(videoElementIndex, q, language);
if (videoUrl !== '') {
quality = q;
break;
}
}
// Check if video exists
if (videoUrl === null) {
console.log("Could not find video feed");
return null;
}
// Check RTMP stream
if (nbRTMP[videoElementIndex] > 0 && videoUrl.substring(0, 7) === "rtmp://") { // check first because it ends with .mp4 like HTTP
button.innerHTML = "Open VLC > CTRL+R > Network > Copy this link > Convert/Save video. ";
}
// Check HTTP
else if (nbHTTP[videoElementIndex] > 0 && videoUrl.substring(videoUrl.length - 4, videoUrl.length) === ".mp4") {
button.innerHTML = "Download video ";
}
// Check HLS stream : should not happen
else if (nbHLS[videoElementIndex] > 0 && videoUrl.substring(videoUrl.length - 5, videoUrl.length === ".m3u8")) {
button.innerHTML = "Open VLC > CTRL+R > Network > Copy this link > Convert/Save video. ";
}
// Unknown URL format : should not happen
else {
console.log('Unknown URL format');
return null;
}
button.setAttribute('id', 'btnDownload' + videoElementIndex); // to refer later in select changes
button.setAttribute('href', videoUrl);
button.setAttribute('target', '_blank');
button.setAttribute('download', getVideoName(videoElementIndex) + ".mp4");
button.setAttribute('class', 'btn btn-default');
button.setAttribute('style', 'line-height: 17px; margin-left:10px; text-align: center; padding-top: 9px; padding-bottom: 9px; padding-left: 12px; padding-right: 12px; color:rgb(40, 40, 40); background-color: rgb(230, 230, 230); font-family: ProximaNova,Arial,Helvetica,sans-serif; font-size: 13px; font-weight: 400;');
return button;
}
function createButtonMetadata(videoElementIndex) {
var title = getVideoName(videoElementIndex);
var subtitle = playerJson[videoElementIndex]['videoJsonPlayer']['VSU'];
var description_short = playerJson[videoElementIndex]['videoJsonPlayer']['V7T'];
var description = playerJson[videoElementIndex]['videoJsonPlayer']['VDE'];
var tags = playerJson[videoElementIndex]['videoJsonPlayer']['VTA'];
// Continue if at least one field is filled
if (title !== undefined || description_short !== undefined || subtitle !== undefined || description !== undefined || tags !== undefined) {
var button = document.createElement('a');
button.setAttribute('class', 'btn btn-default');
button.setAttribute('style', 'line-height: 17px; margin-left:10px; text-align: center; padding: 10px; color:rgb(40, 40, 40); background-color: rgb(230, 230, 230); font-family: ProximaNova,Arial,Helvetica,sans-serif; font-size: 13px;');
button.innerHTML = "Download description ";
var metadata = "[Title]\n" + title + "\n\n[Subtitle]\n" + subtitle + "\n\n[Description-short]\n" + description_short + "\n\n[Description]\n" + description + "\n\n[Tags]\n" + tags;
var encodedData = window.btoa(unescape(encodeURIComponent(metadata)));
button.setAttribute('href', 'data:application/octet-stream;charset=utf-8;base64,' + encodedData);
button.setAttribute('download', getVideoName(videoElementIndex) + '.txt');
return button;
} else {
return null;
}
}
function getComboboxSelectedValue(combobox) {
var cb = document.getElementById(combobox);
if (cb == null) {
cb = parent.document.getElementById(combobox);
}
return cb[cb.selectedIndex].value;
}
function createLanguageComboBox(videoElementIndex) {
var languageComboBox = document.createElement('select');
languageComboBox.setAttribute('id', 'cbLanguage' + videoElementIndex);
// Associate onchange event with function (bypass for GM)
languageComboBox.onchange = function() {
var selectedLanguage = languageComboBox.options[languageComboBox.selectedIndex].value;
console.log("\n> Language changed to " + selectedLanguage);
var btn = document.getElementById('btnDownload' + videoElementIndex);
var selectedQuality = getComboboxSelectedValue('cbQuality' + videoElementIndex);
var url = getVideoUrl(videoElementIndex, selectedQuality, selectedLanguage);
if (url !== '') {
btn.style.visibility = "visible";
btn.setAttribute('href', url);
} else {
btn.style.visibility = "hidden";
}
};
// Fill with available languages
for (l in languages[videoElementIndex]) {
if (languages[videoElementIndex][l] !== 0) {
languageComboBox.innerHTML += "";
}
}
languageComboBox.setAttribute('class', 'btn btn-default');
languageComboBox.setAttribute('style', (languageComboBox.innerHTML === "" ? "visibility:hidden;" : "max-width: 140px; padding: 6px; color:rgb(40, 40, 40); background-color: rgb(230, 230, 230); font-family: ProximaNova,Arial,Helvetica,sans-serif; font-size: 13px; font-weight: 400;"));
return languageComboBox;
}
function createQualityComboBox(videoElementIndex) {
var qualityComboBox = document.createElement('select');
qualityComboBox.setAttribute('id', 'cbQuality' + videoElementIndex);
// Associate onchange event with function (bypass for GM)
qualityComboBox.onchange = function() {
var selectedQuality = qualityComboBox.options[qualityComboBox.selectedIndex].value;
console.log("\n> Quality changed to " + selectedQuality);
var btn = document.getElementById('btnDownload' + videoElementIndex);
if (btn == null) {
btn = parent.document.getElementById('btnDownload' + videoElementIndex);
}
var selectedLanguage = getComboboxSelectedValue('cbLanguage' + videoElementIndex);
console.log(selectedLanguage);
var url = getVideoUrl(videoElementIndex, selectedQuality, selectedLanguage);
if (url !== '') {
btn.style.visibility = "visible";
btn.setAttribute('href', url);
} else {
console.log("Video not found for these settings!")
btn.style.visibility = "hidden";
}
};
// Fill with available qualities
for (q in qualities[videoElementIndex]) {
if (qualities[videoElementIndex][q] !== 0) {
qualityComboBox.innerHTML += "";
}
}
qualityComboBox.setAttribute('class', 'btn btn-default');
qualityComboBox.setAttribute('style', 'width:140px; padding: 6px; margin-left:10px; color:rgb(40, 40, 40); background-color: rgb(230, 230, 230); font-family: ProximaNova,Arial,Helvetica,sans-serif; font-size: 13px; font-weight: 400;');
return qualityComboBox;
}
function createCreditsElement() {
var credits = document.createElement('div');
credits.setAttribute('style', 'text-align: center; line-height: 20px; font-size: 11.2px; color: rgb(255, 255, 255); font-family: ProximaNova, Arial, Helvetica, sans-serif; padding: 5px; background-image:url("")');
credits.innerHTML = 'Arte Downloader v.' + scriptVersion + ' built by and for the community with love' +
'
Contribute Here.';
return credits;
}
function decoratePlayer360(videoElement, videoURL, videoName) {
var container = document.createElement('div');
insertAfter(container, videoElement);
container.setAttribute('class', 'ArteDownloader-v' + scriptVersion)
container.setAttribute('style', 'background-image:url(""); padding: 10px;');
var button = document.createElement('a');
button.innerHTML = "Download " + videoName + " ";
button.setAttribute('id', 'btnArteDownloader');
button.setAttribute('href', videoURL);
button.setAttribute('target', '_blank');
button.setAttribute('download', videoName);
button.setAttribute('class', 'btn btn-default');
button.setAttribute('style', 'margin-left:10px; text-align: center; padding-top: 9px; padding-bottom: 9px; padding-left: 12px; padding-right: 12px; color:rgb(40, 40, 40); background-color: rgb(230, 230, 230); font-family: ProximaNova,Arial,Helvetica,sans-serif; font-size: 13px; font-weight: 400;');
container.appendChild(button);
}
function decoratePlayer(videoElement, videoElementIndex) {
var parent;
var bRoyalSlider = false;
var container = document.createElement('div');
if (is_playlist) {
console.log("> Decorating playlist player")
videoElement.insertBefore(container, videoElement.firstChild)
} else {
// Look for the parent to decorate
parent = videoElement.parentNode;
if (videoElement.id == "jwPlayerContainer") {
console.log("> Decorating playlist player");
}
// iframe player
else if (videoElement.nodeName === "IFRAME") {
console.log("> Decorating iFrame player");
// Arte touslesinternets
if (stringStartsWith(window.location.href, "http://touslesinternets.arte")) {
parent.insertBefore(container, videoElement);
} else {
// Arte Tracks
if (stringStartsWith(window.location.href, "http://tracks.arte")) {
parent = getParent(videoElement, '', "video");
}
insertAfter(container, parent);
}
}
// http://www.arte.tv/arte_vp
else if (stringStartsWith(unescape(top.location), "http://www.arte.tv/arte_vp/")) {
console.log("> Decorating arte_vp");
var child = document.getElementById("arte_vp_player_container");
child.parentNode.insertBefore(container, child);
parent = container;
}
// overlayed player for Arte Cinema or media embedded
else if (stringStartsWith(location.href, "http://cinema.arte") ||
(parent.parentNode.getAttribute('id') === "embed_widget")) {
console.log("> Decorating overlayed Cinema player");
parent = parent.parentNode.parentNode;
parent.appendChild(container);
}
// royal slider player
else if (stringStartsWith(videoElement.getAttribute('class'), 'rsContent')) {
console.log("> Decorating RoyalSlider player");
bRoyalSlider = true;
// Get the parent with SliderTeaserView type
while (parent.getAttribute('data-teaser-type') !== "SliderTeaserView" && parent.getAttribute('class') !== 'dnd-drop-wrapper') {
parent = parent.parentNode;
}
insertAfter(container, parent);
}
// regular player
else {
console.log("> Decorating regular player");
if (stringStartsWith(location.href, "http://concert.arte")) {
var playerSection = document.querySelector('section#section-player');
if (playerSection !== null) {
insertAfter(container, playerSection);
} else {
parent = parent.parentNode;
parent.appendChild(container);
}
} else {
parent = parent.parentNode;
parent.appendChild(container);
}
}
}
container.setAttribute('class', 'ArteDownloader-v' + scriptVersion)
container.setAttribute('style', 'background-image:url(""); padding: 10px;');
// Create index indicator if Royal Slider
if (bRoyalSlider === true) {
var indexElement = document.createElement('span');
indexElement.innerHTML = "Video " + (videoElementIndex + 1) + " / " + nbVideoPlayers;
indexElement.setAttribute('style', 'margin:10px; text-align: center; color:rgb(255, 255, 255); font-family: ProximaNova,Arial,Helvetica,sans-serif; font-size: 13px;');
container.appendChild(indexElement);
}
// Create video name + description span
var videoNameSpan = document.createElement('span');
var subtitle = playerJson[videoElementIndex]['videoJsonPlayer']['VSU'];
videoNameSpan.innerHTML = "" + getVideoName(videoElementIndex) + (subtitle !== undefined ? " - " + subtitle : "") + "
";
videoNameSpan.setAttribute('style', 'margin:10px; text-align: center; color:rgb(255, 255, 255); font-family: ProximaNova,Arial,Helvetica,sans-serif; font-size: 13px;');
container.appendChild(videoNameSpan);
// Create language combobox
var languageComboBox = createLanguageComboBox(videoElementIndex)
container.appendChild(languageComboBox);
// Check if there are languages available to select
var selectedLanguage;
if (languageComboBox.options.length > 0) {
selectedLanguage = languageComboBox.options[languageComboBox.selectedIndex].value;
}
// Create quality combobox
container.appendChild(createQualityComboBox(videoElementIndex));
// Create download button
var btnDownload = createButtonDownload(videoElementIndex, selectedLanguage);
if (btnDownload !== null) {
container.appendChild(btnDownload);
}
// Create metadata button
var btnMetadata = createButtonMetadata(videoElementIndex);
if (btnMetadata !== null) {
container.appendChild(btnMetadata);
}
// Create credits element if not RoyalSlider or if last player from RoyalSlider
if (bRoyalSlider === false || videoElementIndex === nbVideoPlayers - 1) { // glitch with javascript XHR requests concurrency
var credits = createCreditsElement();
parent.appendChild(credits);
}
}
is_playlist = findPlayers();
console.log("> Found " + nbVideoPlayers + " video player(s):");
if (nbVideoPlayers > 0) {
// Init global vars
playerJson = [nbVideoPlayers];
nbVideos = [nbVideoPlayers];
nbHTTP = [nbVideoPlayers];
nbRTMP = [nbVideoPlayers];
nbHLS = [nbVideoPlayers];
languages = [nbVideoPlayers];
qualities = [nbVideoPlayers];
for (i = 0; i < nbVideoPlayers; i++) {
playerJson[i] = 0;
nbVideos[i] = 0;
nbHTTP[i] = 0;
nbRTMP[i] = 0;
nbHLS[i] = 0;
languages[i] = new Object;
qualities[i] = new Object;
}
if (!is_playlist) {
// Analyse each video player, then decorate them
for (var i = 0; i < nbVideoPlayers; i++) {
findPlayerJson(videoPlayerElements[i], i);
}
}
}