// ==UserScript==
// @name YouScrobbler
// @namespace userscripts.org
// @author http://www.lukash.de
// @description Scrobbles the currently watching YouTube video to last.fm.
// @identifier http://userscripts.org/scripts/source/119694.user.js
// @include http://*.youtube.com/*
// @include https://*.youtube.com/*
// @include http://youtube.com/*
// @include https://youtube.com/*
// @include *//*.youtube.com/tv*
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_xmlhttpRequest
// @version 1.4.8
// @noframes
// @run-at document-idle
// ==/UserScript==
/**
* You can contact me on http://www.lukash.de/youscrobbler or on http://userscripts.org/scripts/show/119694 if you have got suggestions, bugs or other questions
*/
'use strict';
const VERSION = '1.4.8';
const APIKEY = 'd2fcec004903116fe399074783ee62c7';
let lastFmAuthenticationUrl = 'http://www.last.fm/api/auth';
let authenticationSessionUrl = 'http://youscrobbler2.lukash.de/auth';
let scrobbleSongUrl = 'http://youscrobbler2.lukash.de/scrobblesong/';
let currentURL = document.URL;
let loadgif = '<div class="us_loadgif"><img alt="loading" src="" /></div>';
let BFather;
let TO1Helper = false;
let isGM;
let trackInfoFromDB = false;
/**
* --- Content ---
* 1. Initializing
* 2. General Functions
* 3. Appearance
* 4. Scrobbling and Login
* 5. Information request
* 6. Miscellaneous
* 7. Update
*/
/**
* --- 1. Initializing ---
*/
function init() {
isGM = typeof GM_getValue != 'undefined' && typeof GM_getValue('a', 'b') != 'undefined'; //is Greasemonkey or Tampermonkey
var isGM2 = 'undefined' === typeof GM_info.scriptHandler; //is Greasemonkey
if (isGM2) {
alert("-YouScrobbler Alert- \n \n YouScrobbler doesnt work with Greasemonkey. Please switch to Tampermonkey. \n\n https://addons.mozilla.org/en-US/firefox/addon/tampermonkey/ \n ");
}
if (!isLoggedIn()) {
tryGetAuthToken();
}
us_addButton();
}
function updateUrl() {
return 'http://youscrobbler.lukash.de/currentversion';
}
function us_reset() {
setTimeout(function() {
us_closebox();
}, 0);
trackInfoFromDB = false;
currentURL = document.URL;
document.getElementById('us_temp_info').setAttribute('us_video_id', getYouTubeVideoId());
document.getElementById('us_temp_info').removeAttribute('artist');
document.getElementById('us_temp_info').removeAttribute('track');
document.getElementById('us_temp_info').removeAttribute('autoscrobbleerror');
document.getElementById('us_temp_info').removeAttribute('scrobbled');
document.getElementById('us_temp_info').setAttribute('us_leftToPlay', -1);
document.getElementById('us_temp_info').removeAttribute('is_full_album');
document.getElementById('us_temp_info').removeAttribute('us_playstart_s');
// save time page was loaded aka playstart time in ctime and gay format
let time = new Date();
us_saveTempData('us_playstart_s', Math.round(time.getTime() / 1000));
us_abortScrobbling();
getTrackInfo();
us_buttonStatus();
}
/**
* --- 2. General Functions ---
*/
function initPreferences() {
if (!us_getValue('us_boxPosition')) {
us_saveValue('us_boxPosition', (screen.availWidth / 1.3) + ';' + 70);
}
if ((!us_getValue('us_color')) || (us_getValue('us_color') == 'r')) {
us_saveValue('us_color', 'red');
}
if (!us_getValue('database.id', 0)) {
us_saveValue('database.id', '');
us_saveValue('database.artist', '');
us_saveValue('database.track', '');
}
if (!us_getValue('database_additional.id', 0)) {
us_saveValue('database_additional.id', '');
us_saveValue('database_additional.albumtitle', '');
us_saveValue('database_additional.mbid', '');
}
if (!us_getValue('database.maxEntries', 0)) {
us_saveValue('database.maxEntries', 5000);
}
if (!us_getValue('scrobble_at', 0)) {
us_saveValue('scrobble_at', 75);
}
if (us_getValue('us_autoscrobble_active', 'nf') == 'nf') {
us_saveValue('us_autoscrobble_active', 1);
}
if (us_getValue('asFailNotification', 'nf') == 'nf') {
us_saveValue('asFailNotification', false);
}
if (us_getValue('scrobblingNotification', 'nf') == 'nf') {
us_saveValue('scrobblingNotification', true);
}
if (us_getValue('us_autocorrect_tracks', 'nf') == 'nf') {
us_saveValue('us_autocorrect_tracks', true);
}
}
// Creates a <type id="id">
function createIdElement(type, id) {
let el = document.createElement(type);
el.setAttribute('id', id);
return el;
}
/**
* Makes an element draggable.
*
* Based on https://gist.github.com/remarkablemark/5002d27442600510d454a5aeba370579
*
* @param {HTMLElement} element - The element that will be dragged.
* @param {HTMLElement} handle - The element that will facilitate the dragging.
* @param {Number} initialPositionX - Initial x position of the element.
* @param {Number} initialPositionY - Initial y position of the element.
* @param {Function} persistPosition - Function called after position is changed.
*/
function us_draggable(element, handle, initialPositionX, initialPositionY, persistPosition) {
let isMouseDown = false;
// initial mouse X and Y for `mousedown`
let mouseX;
let mouseY;
// element X and Y before and after move
let elementX = initialPositionX;
let elementY = initialPositionY;
// mouse button down over the handle
handle.addEventListener('mousedown', onMouseDown);
/**
* Listens to `mousedown` event.
*
* @param {Object} event - The event.
*/
function onMouseDown(event) {
mouseX = event.clientX;
mouseY = event.clientY;
isMouseDown = true;
event.preventDefault();
}
// mouse button released
document.addEventListener('mouseup', onMouseUp);
/**
* Listens to `mouseup` event.
*/
function onMouseUp() {
isMouseDown = false;
elementX = parseInt(element.style.left) || 0;
elementY = parseInt(element.style.top) || 0;
persistPosition(elementX, elementY);
}
// need to attach to the entire document
// in order to take full width and height
// this ensures the element keeps up with the mouse
document.addEventListener('mousemove', onMouseMove);
/**
* Listens to `mousemove` event.
*
* @param {Object} event - The event.
*/
function onMouseMove(event) {
if (!isMouseDown) {
return;
}
let deltaX = event.clientX - mouseX;
let deltaY = event.clientY - mouseY;
element.style.left = elementX + deltaX + 'px';
element.style.top = elementY + deltaY + 'px';
}
}
function GM_main() {
window.us_stateChanged = function(state) {
let playerNode;
if (document.getElementById('c4-player')) {
playerNode = document.getElementById('c4-player');
} else {
playerNode = document.getElementById('movie_player');
}
// get video ID
let regex = /(\?|%3F|&|%26)v=[^?&#]*/gi;
let removeRegex = /(\?|%3F|&|%26)v=/gi;
let matches = document.URL.match(regex);
let vidId;
if (matches != null) {
vidId = matches[0].replace(removeRegex, '');
} else {
matches = playerNode.getVideoUrl().match(regex);
vidId = matches[0].replace(removeRegex, '');
}
if (state == 1 && vidId != document.getElementById('us_temp_info').getAttribute('us_video_id')) {
setTimeout(function() {
document.getElementById('us_temp_info').setAttribute('us_reset_now', '1');
}, 1);
}
switch (state) {
case 1:
document.getElementById('us_temp_info').setAttribute('video_is_playing', '1');
break;
case 0:
document.getElementById('us_temp_info').setAttribute('video_end_reached', 'yes');
document.getElementById('us_temp_info').setAttribute('video_is_playing', '0');
break;
default:
document.getElementById('us_temp_info').setAttribute('video_is_playing', '0');
}
if (document.getElementById('us_temp_info').getAttribute('is_full_album') != 'yes') {
document.getElementById('us_temp_info').setAttribute('us_secs', playerNode.getDuration());
}
};
window.onYouTubePlayerReady = function() {
let playerNode;
if (document.getElementById('c4-player')) {
playerNode = document.getElementById('c4-player');
} else {
playerNode = document.getElementById('movie_player');
}
if (playerNode) {
playerNode.removeEventListener('onStateChange', 'us_stateChanged');
playerNode.addEventListener('onStateChange', 'us_stateChanged');
document.getElementById('us_temp_info').setAttribute('us_secs', playerNode.getDuration());
} else {
throw new Error('YouScrobbler: Player node not found!');
}
};
if (document.getElementById('us_temp_info').getAttribute('us_secs') == 0) {
window.onYouTubePlayerReady();
}
}
function addJS_Node(text, s_URL, funcToRun, runOnLoad) {
let D = document;
let scriptNode = D.createElement('script');
if (runOnLoad) {
scriptNode.addEventListener('load', runOnLoad);
}
scriptNode.type = 'text/javascript';
if (text) {
scriptNode.textContent = text;
}
if (s_URL) {
scriptNode.src = s_URL;
}
if (funcToRun) {
scriptNode.textContent = '(' + funcToRun.toString() + ')()';
}
let targ = D.getElementsByTagName('head')[0] || D.body || D.documentElement;
targ.appendChild(scriptNode);
}
/**
* Value Saving Method - Switcher
* uses Greasemonkey GM_ or localStorage
*/
function us_saveValue(name, value) {
if (isGM) {
GM_setValue(name, value);
} else {
localStorage.setItem(name, value);
}
}
/**
* Value Getting Method - Switcher
* uses Greasemonkey GM_ or localStorage
*/
function us_getValue(name, alternative) {
if (isGM) {
return (GM_getValue(name, alternative));
} else {
return (localStorage.getItem(name, alternative));
}
}
/**
* Temporary save data
* Saved in "us_temp_info" attributes
*/
function us_saveTempData(name, value) {
document.getElementById('us_temp_info').setAttribute(name, value);
}
/**
* Get temporary saved data
* Saved in "us_temp_info" attributes
*/
function us_getTempData(name) {
if (document.getElementById('us_temp_info').getAttribute(name)) {
let value = document.getElementById('us_temp_info').getAttribute(name);
return value;
} else {
return 0;
}
}
/**
* --- 3. Appearance ---
*/
/**
* Add the Scrobble Button to Video and Userpages
*/
function us_addButton() {
let secs = 0;
let time = new Date();
let t = Math.round(time.getTime() / 1000);
let style_el = document.createElement('style');
let head = document.getElementsByTagName('head')[0];
style_el.innerHTML = `
#us_loginbox_form table { text-align: left; table-layout: fixed; }
#us_loginbox button { background: transparent; border: none; margin: 0; padding: 0; cursor: pointer; }
.us_box { border-radius: 5px; border: 5px solid #333; background: #fff;
/* by AshKyd */
z-index:1000000; position: absolute; width: 300px; }
.us_box h3 { cursor: move; padding: 4px 8px 4px 10px; margin: 0px; border-bottom: 1px solid #AAA; background-color: #EEE; }
.us_box h4 { margin-left: 5px; margin-bottom:0px}
#us_loginbox .round_button { border-radius: 50%; width: 14px; height: 14px; padding: 3px; float: right; margin:1px 3px 0 0; background-image: linear-gradient(to bottom, #b4b4b4 0%, #9d9d9d 50%, #868686 100%); background-color: #9d9d9d; }
#us_loginbox .round_button:hover { background-image: linear-gradient(to bottom, #828282 0%, #6b6b6b 50%, #545454 100%); background-color: #3e3e3e; }
#us_loginbox .round_button svg { display: block; width: 8px; height: 8px; }
#us_box_head > ul, #us_box_head li { float:right; }
#us_box_head ul { list-style-type:none; margin: 0;}
.us_settings_grp { height:50px; vertical-align:middle; padding-right:3px;padding-left:5px}
.us_settings_grp hr { background-color: #EEE; margin: 5px 8px; height: 1px;}
.us_settings_grp_left { width:155px}
.us_settings_grp_right { width:135px}
.us_settings_grp span { vertical-align:middle}
.us_settings_grp_heading { color:#777;font-size:100%;font-weight:bold; border-bottom:1px solid #ccc; margin-bottom:4px;}
.us_settings_grp_database { cursor: help;}
#databaseMaxLength {width: 55px; }
#scrobbleStatus { font-weight: bold; }
#scrobble_at {width: 45px; }
#us_resetlogin { float: left; }
#us_loginbox_form { text-align: right; padding: 5px; }
.us_box input[type=text] { height: 16px; border: 1px solid #bbb; margin: 2px 15px 4px 2px; padding: 3px 4px; width: 170px;}
.us_box input[type=submit] { cursor:pointer; margin: 0 0 0 5px; padding: 0 4px 3px 4px; border-radius: 2px; font-size: 11px; font-weight: bold; color: white; height: 18px; border: 1px solid #3e3e3e; background-color: #6b6b6b; background-image: linear-gradient(to bottom, #828282 0%, #6b6b6b 50%, #545454 100%); }
.us_box input[type=submit]:hover { background-color: #9d9d9d; background-image: linear-gradient(to bottom, #b4b4b4 0%, #9d9d9d 50%, #868686 100%);}
.us_box input[type=submit]:active { padding: 1px 4px 2px 4px;}
.us_hidden { visibility: hidden; overflow: hidden; height: 0px; }
#us_hiddenform { margin: 0; pading-right: 10px;}
#us_hiddenform input[type=text] {margin-right:15px}
#us_loginbox #us_quickchange { position:relative; bottom:40px; width:9px; height:15px; float:right; background-image: url(); }
#us_loginbox #us_quickchange:focus { background-color: #FFF; outline:none}
.us_clickable_formdesc {}
.us_loadgif { text-align: center; padding: 10px 0; }
.us_loadgif img { border-radius: 5px; border:3px solid #91998E; }
#us_loginbox #us_more { font: normal normal bold 12pt/12pt Arial; color: #999; text-decoration: none; margin-right: 3px; }
#us_loginbox #us_more:focus { background:none; outline:none }
.us_submitbuttons { background-color: #EEE; border-top: 1px solid #AAA; padding: 5px; width: 290px; height: 18px; margin-top: 5px; }
#scrobbleStatus_parent {float: left; height: 18px; margin-left: 15px; padding-left: 5px; padding-top: 2px; color:#888}
#us_autoscrobble {vertical-align:middle;}
.us_submitbuttons_box_left {float: left;}
.us_error { background-color: #F6D8D8; border: 1px solid #f28494; padding: 5px 3px 5px 3px; width: 90%; margin: 6px auto 10px; }
.us_done { background-color: #CCFF99; border: 1px solid #99CC00; padding: 5px 3px 5px 3px; width: 90%; margin: 5px auto; }
.us_infobox { z-index:1000000; background-color: #E8E8E8; border-radius: 5px; padding: 10px; position: fixed; right: 16px; bottom: 9px; border: 1px solid #000000; font-size: 10pt; }
.us_infobox div { color: #AAAAAA; margin: 1px 5px 0 0; float: left; }
.us_infobox div img { float: right; margin: -1px -6px 1px 8px; vertical-align: middle;}
.us_infobox .sep { color:#AAA; }
.us_trackinfo { color: #47D93D; font-weight: bold; padding-right: 8px; font-size: 11pt; vertical-align: middle;}
.us_box .us_center { padding: 10px; text-align: center; }
.us_box .us_left { padding: 10px; text-align: left; }
#us_submit { float: right; margin-bottom:5px;}
us_submitbuttons_box_left {border}
#us_scrobblebutton { float:right; cursor: pointer; margin-left:16px;}
#us_start_scrobblebutton {padding-left:3px!important} /* Feather check */
#us_start_scrobblebutton_text { vertical-align: middle; background-repeat: no-repeat; background-position: left center; padding-left: calc(16px + 0.5em); display: inline-block; height: 16px; line-height: 16px; background-image: url(); }
#us_start_scrobblebutton_text.black_icon { background-image: url(); }
#fullAlbumIcon { float: left; height: 16px; width: 16px; cursor: help;}
#foundInDBIcon { float: left; height: 16px; width: 16px; cursor: help; background-image: url();}
#us_scrobble_on {font-weight:bold; color: #66CC00;}
#us_scrobble_failed {font-weight:bold; color: #D10404;}
#us_scrobble_statusbar {background-color: #66CC00; height: 2px; width: 0; opacity: 0.8; margin: 0px; padding-right: 1px; }
.us_status_failed { background-color: #CC181E; }
.us_status_hidden { display: none; }
#us_loginbox .us_status_small {color: #999; font-size:80%}
.us_box, .us_infobox {visibility: visible; opacity: 1; transition: opacity 0.5s;}
.us_box_hidden {visibility: hidden; opacity: 0; transition: visibility 0s 0.5s, opacity 0.5s;}
`;
// us_start_scrobblebutton
let button = createIdElement('span', 'us_scrobblebutton');
// Design check
if (document.getElementsByTagName('ytd-searchbox')[0]) {
BFather = document.getElementsByTagName('ytd-searchbox')[0];
button.setAttribute('class', 'yt-uix-button-group');
button.style.marginLeft = '50px';
button.style.padding = '6px 0 6px 0';
button.style.border = '1px solid var(--ytd-searchbox-legacy-border-color)';
button.style.background = 'var(--ytd-searchbox-legacy-border-color)';
button.innerHTML = `
<input id="us_temp_info" video_is_playing="1" type="hidden" us_secs="${secs}" us_playstart_s="${t}"/>
<a style="border-radius:2px; 2px; 2px; 2px;padding-right:6px;padding-left:8px!important" class="yt-uix-button yt-uix-sessionlink start yt-uix-button-default" id="us_start_scrobblebutton">
<span id="us_start_scrobblebutton_text">Scrobble</span>
</a>
<div id="us_scrobble_statusbar" class="us_status_hidden"></div>
`;
BFather.insertBefore(button, BFather.lastChild);
document.getElementById('us_scrobble_statusbar').style.position = 'relative';
document.getElementById('us_scrobble_statusbar').style.top = '6px';
} else if (document.getElementById('masthead-nav')) {
BFather = document.getElementById('masthead-nav');
button.innerHTML = `
<input id="us_temp_info" video_is_playing="1" type="hidden" us_secs="${secs}" us_playstart_s="${t}" />
<a class="start" id="us_start_scrobblebutton"><span id="us_start_scrobblebutton_text">Scrobble</span></a>
<span class="masthead-link-separator">|</span>
`; // postxanadus
BFather.insertBefore(button, BFather.firstChild);
} else if (document.getElementById('yt-masthead-content')) {
BFather = document.getElementById('yt-masthead-content');
button.setAttribute('class', 'yt-uix-button-group');
button.style.float = 'right';
button.style.marginLeft = '20px';
button.style.marginTop = '3px';
button.style.marginRight = '2px';
button.style.borderTopRightRadius = '2px';
button.style.borderBottomRightRadius = '2px';
button.innerHTML = `
<input id="us_temp_info" video_is_playing="1" type="hidden" us_secs="${secs}" us_playstart_s="${t}" />
<a style="border-radius:2px; 2px; 2px; 2px;padding-left:6px!important" class="yt-uix-button yt-uix-sessionlink start yt-uix-button-default" id="us_start_scrobblebutton">
<span id="us_start_scrobblebutton_text">Scrobble</span>
</a>
<div id="us_scrobble_statusbar" class="us_status_hidden"></div>
`;
BFather.insertBefore(button, BFather.firstChild);
} else if (document.getElementById('mh')) {
BFather = document.getElementById('mh');
button.setAttribute('class', 'ml');
button.innerHTML = `
<input id="us_temp_info" video_is_playing="1" type="hidden" us_secs="${secs}" us_playstart_s="${t}" />
<a class="start" id="us_start_scrobblebutton"><span id="us_start_scrobblebutton_text"> Scrobble</span></a>
`;
BFather.insertBefore(button, document.getElementById('se').nextSibling);
} else {
setTimeout(function() {
us_addButton();
}, 1000);
return;
}
if (us_getValue('us_color') === 'black') {
document.getElementById('us_start_scrobblebutton_text').classList.add('black_icon');
}
head.appendChild(style_el);
us_buttonStatus();
document.getElementById('us_temp_info').setAttribute('us_video_id', getYouTubeVideoId());
addJS_Node(null, null, GM_main);
setTimeout(function() {
us_ajax_scanner();
}, 1000);
checkFirstRun();
}
function us_buttonStatus() {
let secs = us_getTempData('us_secs');
document.getElementById('us_start_scrobblebutton').style.opacity = 1;
if (secs > 30) {
document.getElementById('us_scrobblebutton').addEventListener('click', us_toggleBox, true);
document.getElementById('us_scrobblebutton').title = '';
document.getElementById('us_start_scrobblebutton').style.opacity = 1;
// watched seconds till scrobbling
let time_left_to_scrobble;
if (us_getTempData('us_leftToPlay', false) == false || us_getTempData('us_leftToPlay') < 0) {
time_left_to_scrobble = parseInt(us_getTempData('us_secs') * (us_getValue('scrobble_at')) * 0.01);
us_saveTempData('us_leftToPlay', parseInt(time_left_to_scrobble));
}
tryAutoScrobble();
} else {
document.getElementById('us_start_scrobblebutton').style.opacity = 0.5;
document.getElementById('us_scrobblebutton').removeEventListener('click', us_toggleBox);
if (secs == 0) {
document.getElementById('us_scrobblebutton').title = 'There is no video to scrobble.';
} else {
document.getElementById('us_scrobblebutton').title = 'Video is too short to be scrobbled.';
}
}
if (secs == 0) {
if (!getYouTubeVideoId()) {
setTimeout(function() {
us_buttonStatus();
}, 5000);
} else {
setTimeout(function() {
us_buttonStatus();
}, 1000);
}
}
}
function us_toggleBox() {
if (!document.getElementById('us_loginbox') || document.getElementById('us_loginbox').classList.contains('us_box_hidden')) {
us_showBox(false);
} else {
us_closebox();
}
}
// Show dialog window
// Contains either login form, or scrobble form
function us_showBox(justLoggedIn) {
// check if scrobblerbox was dropped out of possible screen width and if reset
let boxPosition = us_getValue('us_boxPosition').split(';');
let boxPositionX = parseInt(boxPosition[0]);
let boxPositionY = parseInt(boxPosition[1]);
let boxPositionChanged = false;
if (boxPositionX > screen.availWidth || boxPositionX < 0) {
boxPositionX = screen.availWidth / 1.3;
boxPositionChanged = true;
}
if (boxPositionY > screen.availHeight || boxPositionY < 50) {
boxPositionY = 75;
boxPositionChanged = true;
}
if (boxPositionChanged) {
us_saveValue('us_boxPosition', boxPositionX + ';' + boxPositionY);
}
// either create loginbox or show it
if (!document.getElementById('us_loginbox')) {
let loginbox = createIdElement('div', 'us_loginbox');
loginbox.classList.add('us_box');
loginbox.style.left = boxPositionX + 'px';
loginbox.style.top = boxPositionY + 'px';
document.body.insertBefore(loginbox, document.body.firstChild);
} else if (document.getElementById('us_loginbox').classList.contains('us_box_hidden')) {
let loginbox = document.getElementById('us_loginbox');
loginbox.style.left = boxPositionX + 'px';
loginbox.style.top = boxPositionY + 'px';
loginbox.classList.remove('us_box_hidden');
}
if (!isLoggedIn()) {
let cont = `
<div id="us_loginbox_form">
<div class="us_error">You are currently not logged in!</div><br />
<span>Click Login below to authenticate your account</span><br/><br/>
<em>Note: You will leave this site and be redirected here after having logged in to Last.fm.</em><br/><br />
</div>
<div class="us_submitbuttons"><input id="us_submit" value="Authenticate" type="submit" /></div>
`;
us_boxcontent('Login to last.fm', cont);
document.getElementById('us_submit').addEventListener('click', us_authenticate);
} else {
us_scrobbleform(justLoggedIn);
}
}
/**
* inserts the scrobbleform into the window
*/
function us_scrobbleform(justLoggedIn) {
let messageText = '';
let checkedText = '';
let databaseFoundText = '';
let scrobbleStatus = '';
let feedback = getTrackInfo();
let artist = decodeURIComponent(us_getTempData('artist'));
let track = decodeURIComponent(us_getTempData('track'));
let album = decodeURIComponent(us_getTempData('album'));
if (artist == 0 && track == 0) {
artist = '';
track = '';
}
if (album == 0) {
album = '';
}
if (TO1Helper) {
let restTime;
if (us_getTempData('us_lefttoplay')) {
restTime = us_getTempData('us_lefttoplay');
} else {
restTime = us_getTempData('us_secs');
}
scrobbleStatus = `<div id="scrobbleStatus_parent"> scrobble in <span id="scrobbleStatus">${restTime}</span> sec <button type="button" id="us_abortScrobbling" title="abort scrobbling">×</button></div>`;
}
if (us_getTempData('scrobbled') == 1) {
scrobbleStatus = '<div id="scrobbleStatus_parent">scrobbled</div>';
}
if (justLoggedIn) {
messageText = '<div class="us_done">Successfully logged in</div>';
}
let asE = us_getTempData('autoscrobbleError');
if (asE) {
if (asE == 'failed') {
messageText += '<div class="us_error">AutoScrobble failed. Please edit info.</div>';
}
if (asE == 'noMusic') {
messageText += '<div class="us_error">Track not found on Last.fm.</div>';
}
if (asE == 'bad') {
messageText += '<div class="us_error">Video title is not in a valid format to be scrobbled.</div>';
}
}
if (us_getValue('us_autoscrobble_active', 0) == 1) {
checkedText = ' checked';
}
if (((feedback == 'found') && (trackInfoFromDB))) {
databaseFoundText = '<div id="foundInDBIcon" title="Track information retrieved from personal database"></div>';
}
if (us_getTempData('is_full_album') == 'yes') {
databaseFoundText = `<div id="fullAlbumIcon" title="Video was recognized as a full album">Full Album: Track ${us_getTempData('full_album_track_nr')} of ${us_getTempData('full_album_track_count')}</div>`;
}
let cont = `
<div id="us_loginbox_form">${databaseFoundText}
${messageText}
<form name="us_scrobbleform" onSubmit="return false">
Artist: <input type="text" name="artist" value="${artist}" /><br />
Track: <input type="text" name="track" value="${track}" /><br/>
<button id="us_quickchange" title="Artist ↔ Track"></button>
<button type="button" id="us_more" title="more options">+</button>
<p id="us_hiddenform" class="us_hidden">
Album title: <input type="text" name="album" value="${album}" /><br />
</p>
</form>
</div>
<div class="us_submitbuttons">
<div class="us_submitbuttons_box_left" title="Activate automatic scrobbling?"><input id="us_autoscrobble" name="us_autoscrobble" type="checkbox"${checkedText}><label for="us_autoscrobble">Auto</label></div>
${scrobbleStatus}
<input id="us_submit" value="Scrobble" type="submit" />
</div>
`;
us_boxcontent('Scrobble to last.fm - ' + us_getValue('us_username'), cont);
document.getElementById('us_quickchange').addEventListener('click', us_quickchange);
document.getElementById('us_submit').addEventListener('click', us_scrobblenp);
document.getElementById('us_autoscrobble').addEventListener('click', function() {
if (this.checked) {
us_saveValue('us_autoscrobble_active', 1);
} else {
us_saveValue('us_autoscrobble_active', 0);
}
});
document.getElementById('us_more').addEventListener('click', us_showmoreform);
if (document.getElementById('us_abortScrobbling')) {
document.getElementById('us_abortScrobbling').addEventListener('click', us_abortScrobbling);
}
if (us_getValue('us_more_options_show_or_hide') == 'show') {
us_showmoreform();
}
}
// little box show info-messages
function us_infoBox(cont) {
let inbox;
if (!document.getElementById('us_infobox')) {
inbox = createIdElement('div', 'us_infobox');
inbox.classList.add('us_infobox');
document.body.appendChild(inbox);
} else {
inbox = document.getElementById('us_infobox');
inbox.classList.remove('us_box_hidden');
}
inbox.addEventListener('click', us_closeinfobox);
inbox.style.cursor = 'pointer';
inbox.title = 'Click to Close';
inbox.innerHTML = cont;
}
// closes the box with fadeout effect
function us_closebox() {
let object = document.getElementById('us_loginbox');
object.classList.add('us_box_hidden');
}
// closes the info-box with fadeout effect
function us_closeinfobox() {
let object = document.getElementById('us_infobox');
object.classList.add('us_box_hidden');
}
// shows the optional data fields
function us_showmoreform() {
let i1 = document.getElementById('us_hiddenform');
let a = document.getElementById('us_more');
if (i1.classList.contains('us_hidden')) {
i1.classList.remove('us_hidden');
a.innerHTML = '−';
us_saveValue('us_more_options_show_or_hide', 'show');
} else {
i1.classList.add('us_hidden');
a.innerHTML = '+';
us_saveValue('us_more_options_show_or_hide', 'hide');
}
}
/**
* Fills window with title and content
*/
function us_boxcontent(title, content) {
let loginbox = document.getElementById('us_loginbox');
if (!loginbox) {
return false;
}
if (loginbox.classList.contains('us_box_hidden')) {
loginbox.classList.remove('us_box_hidden');
}
loginbox.innerHTML = `
<h3 id="us_box_head">${title}<ul>
<li><button type="button" title="Close" id="us_box_close" class="round_button"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M2.4,2.4 21.6,21.6 M2.4,21.6 21.6,2.4" style="fill:none;stroke-width:5;stroke:#fff"/></svg></button></li>
<li><button type="button" title="Settings" id="us_box_settings" class="round_button"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 479.8 479.8" fill="#fff"><path d="m479.3 116.6c-0.4-4.3-3.2-7.9-7.2-9.4-4-1.5-8.5-0.5-11.6 2.6l-62.2 62-69-21.8-21.9-68.7 62.2-62c3-3 4-7.5 2.6-11.6-1.5-4-5.1-6.8-9.4-7.2C324.9-2.7 287.9 10.7 261.2 37.4 224.5 73.9 214.8 127.3 231.9 172.8c-1.9 1.6-3.7 3.3-5.5 5.1L18.5 373.2c-0.1 0.1-0.1 0.1-0.2 0.2-24.4 24.3-24.4 64 0 88.3 24.4 24.3 64 24 88.3-0.3 0.1-0.1 0.2-0.2 0.3-0.3L301.3 252.5c1.8-1.8 3.4-3.6 4.9-5.5 45.7 17.2 99.3 7.5 136-29.1 26.8-26.7 40.4-63.6 37-101.3zM75.3 435.4c-9 9-23.6 9-32.6 0-9-9-9-23.5 0-32.5 9-9 23.6-9 32.6 0 8.9 9 8.9 23.5 0 32.5z"/></svg></button></li>
<li><button type="button" title="Help" id="us_box_help" class="round_button"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="#fff"><path d="m10.1 23.9h4v-4h-4zm2-24c-4.4 0-8 3.6-8 8h4c0-2.2 1.8-4 4-4 2.2 0 4 1.8 4 4 0 4-6 3.5-6 10h4c0-4.5 6-5 6-10 0-4.4-3.6-8-8-8z"/></svg></button></li>
</ul></h3>
<div>${content}</div>
`;
document.getElementById('us_box_close').addEventListener('click', us_closebox);
document.getElementById('us_box_settings').addEventListener('click', us_settings);
document.getElementById('us_box_help').addEventListener('click', us_help);
let position = us_getValue('us_boxPosition').split(';');
let initialPositionX = parseInt(position[0]);
let initialPositionY = parseInt(position[1]);
us_draggable(
loginbox,
document.getElementById('us_box_head'),
initialPositionX,
initialPositionY,
function persistPosition(elementX, elementY) {
us_saveValue('us_boxPosition', elementX + ';' + elementY);
}
);
}
/**
* Show the help-window
*/
function us_help() {
let cont = `
<p class="us_left">Documentation, Changelog and more can be found on the <a target="_blank" href="http://www.lukash.de/youscrobbler" title="YouScrobbler on lukash.de">YouScrobbler Website</a>.</p>
<h4>Feedback</h4>
<p class="us_left">Suggestions and other Questions can be posted in the <a target="_blank" href="http://www.last.fm/group/YouScrobbler/forum" title="YouScrobbler Forum">Forum</a>.</p>
<h4>Links</h4>
<p class="us_left">
<a target="_blank" href="http://www.lukash.de/youscrobbler" title="YouScrobbler on lukash.de">YouScrobbles Website</a><br/>
<a target="_blank" href="http://www.last.fm/group/YouScrobbler" title="Last.fm Group">Last.fm Group</a><br/>
<a target="_blank" href="https://github.com/floblik/YouScrobbler" title="GitHub">GitHub repo</a><br/>
</p>
`;
us_boxcontent('About - YouScrobbler ' + VERSION, cont);
}
/**
* Show the settings-window
*/
function us_settings() {
let maxEntries = us_getValue('database.maxEntries', 5000);
let cont = `
<div id="us_loginbox_form"><form name="us_settings_form" onSubmit="return false"><table><tr>
<td class="us_settings_grp us_settings_grp_left">
<div class="us_settings_grp_heading">General</div>
<input type="checkbox" id="us_settings_asFailNotification" name="us_settings_asFailNotification"/><label for="us_settings_asFailNotification">error notification</label><br/>
<input type="checkbox" id="us_settings_scrobblingNotification" name="us_settings_scrobblingNotification"/><label for="us_settings_scrobblingNotification">scrobble notification</label><br/>
<hr/>
<label for="scrobble_at">scrobble at </label><select name="scrobble_at" id="scrobble_at">
<option id="scrobble_at10" value="10">10</option>
<option id="scrobble_at25" value="25">25</option>
<option id="scrobble_at50" value="50">50</option>
<option id="scrobble_at75" value="75">75</option>
<option id="scrobble_at95" value="95">95</option>
</select><span>%</span><br/>
<input type="checkbox" id="us_settings_autoCorrect" name="us_settings_autoCorrect"/><label for="us_settings_autoCorrect">last.fm auto correct</label><br/>
<hr/>
<div>
<input type="radio" id="us_settings_color_red" name="us_settings_color" value="red" /><label for="us_settings_color_red">Red</label>
<input type="radio" id="us_settings_color_black" name="us_settings_color" value="black" /><label for="us_settings_color_black">Black</label>
</div>
</td>
<td class="us_settings_grp us_settings_grp_right">
<div class="us_settings_grp_heading us_settings_grp_database" title="Your custom edited track information">Database</div>
<span>Size: ${us_getValue('database.id').split(' ').length - 1} / <select name="databaseMaxLength" id="databaseMaxLength">
<option id="databaseMaxLength500" value="500">500</option>
<option id="databaseMaxLength5000" value="5000">5000</option>
<option id="databaseMaxLength-1" value="-1">unlimited</option>
</select></span><br/><br/>
<div class="us_settings_grp_heading">About</div>
<span>Version: ${VERSION}</span><br/>
<span id="us_manualupdate"><button type="button" id="us_manualupdate_link">Check for Update</button></span>
</td>
</tr></table></form></div>
<div class="us_submitbuttons"><input type="submit" id="us_resetlogin" value="Reset Login"/></div>
`;
us_boxcontent('Settings', cont);
let us_settings_color = 'us_settings_color_' + us_getValue('us_color');
document.getElementById(us_settings_color).setAttribute('checked', 'checked');
if (us_getValue('asFailNotification', 0) || us_getValue('asFailNotification') == 'yes') {
document.getElementById('us_settings_asFailNotification').setAttribute('checked', 'checked');
}
if (us_getValue('scrobblingNotification')) {
document.getElementById('us_settings_scrobblingNotification').setAttribute('checked', 'checked');
}
if (us_getValue('us_autocorrect_tracks')) {
document.getElementById('us_settings_autoCorrect').setAttribute('checked', 'checked');
}
document.getElementById('databaseMaxLength' + maxEntries.toString()).selected = true;
if (document.getElementById('scrobble_at' + us_getValue('scrobble_at', 75).toString())) {
document.getElementById(('scrobble_at' + us_getValue('scrobble_at', 75).toString())).selected = true;
} else {
document.getElementById('scrobble_at75').selected = true;
}
document.getElementById('us_resetlogin').addEventListener('click', us_resetlogin);
document.getElementById('us_manualupdate_link').addEventListener('click', function() {
document.getElementById('us_manualupdate').innerHTML = '<span class="us_status_small">checking</span>';
updateCheck(true);
});
// Save settings
document.getElementById('us_settings_color_red').addEventListener('change', function() {
us_saveValue('us_color', 'red');
document.getElementById('us_start_scrobblebutton_text').classList.remove('black_icon');
});
document.getElementById('us_settings_color_black').addEventListener('change', function() {
us_saveValue('us_color', 'black');
document.getElementById('us_start_scrobblebutton_text').classList.add('black_icon');
});
document.getElementById('us_settings_asFailNotification').addEventListener('change', function() {
us_saveValue('asFailNotification', document.getElementById('us_settings_asFailNotification').checked);
});
document.getElementById('us_settings_autoCorrect').addEventListener('change', function() {
us_saveValue('us_autocorrect_tracks', document.getElementById('us_settings_autoCorrect').checked);
});
document.getElementById('us_settings_scrobblingNotification').addEventListener('change', function() {
us_saveValue('scrobblingNotification', document.getElementById('us_settings_scrobblingNotification').checked);
});
document.getElementById('databaseMaxLength').addEventListener('change', function() {
let el = document.getElementById('databaseMaxLength');
let text = el.options[el.selectedIndex].value;
us_saveValue('database.maxEntries', text);
});
document.getElementById('scrobble_at').addEventListener('change', function() {
let el = document.getElementById('scrobble_at');
let text = el.options[el.selectedIndex].value;
us_saveValue('scrobble_at', text);
});
}
/**
* --- 4. Scrobbling and Login ---
*/
function isMusicVideo(infoResult) {
let artist = us_getTempData('artist').replace(' ', '+');
let track = us_getTempData('track').replace(' ', '+');
let url = 'http://ws.audioscrobbler.com/2.0/?method=track.getInfo&api_key=' + APIKEY + '&artist=' + artist + '&track=' + track + '&autocorrect=1&format=json';
GM_xmlhttpRequest({
method: 'GET',
url: url,
onload: function(response) {
if (response.responseText) {
let json = JSON.parse(response.responseText);
if (json.track && (json.track.name != us_getTempData('track') || json.track.artist.name != us_getTempData('artist')) && us_getValue('us_autocorrect_tracks') == 'yes' && !json.error) {
us_saveTempData('track', json.track.name);
us_saveTempData('artist', json.track.artist.name);
}
if (json && json.track || trackInfoFromDB) {
tryAutoScrobbleCallback(infoResult, true);
return true;
} else
if (json.error) {
tryAutoScrobbleCallback('noMusic', false);
return false;
}
}
tryAutoScrobbleCallback(infoResult, false);
return false;
},
onerror: function() {
tryAutoScrobbleCallback(infoResult, false);
return false;
},
ontimeout: function() {
tryAutoScrobbleCallback(infoResult, false);
return false;
}
});
}
/**
* Tries to AutoScrobble video if user is logged in, its a music video and the trackinfo was found
*/
function tryAutoScrobble() {
if (us_getValue('us_autoscrobble_active', 0) == 1) {
let response = getTrackInfo();
if (us_getTempData('is_full_album') != 'yes') {
isMusicVideo(response);
}
}
}
function tryAutoScrobbleCallback(response, musicVideo) {
if ((isLoggedIn()) && ((trackInfoFromDB) || ((response == 'found') && (musicVideo)))) {
// save time page was loaded aka playstart time in ctime and gay format
let time = new Date();
us_saveTempData('us_playstart_s', Math.round(time.getTime() / 1000));
us_scrobble(decodeURIComponent(us_getTempData('artist')), decodeURIComponent(us_getTempData('track')), '', '', 0, 0, 1);
return;
} else if (response == 'bad') {
us_saveTempData('autoscrobbleError', 'bad');
scrobble_statusbar('failed');
} else if (!musicVideo) {
us_saveTempData('autoscrobbleError', 'noMusic');
scrobble_statusbar('failed');
} else {
us_saveTempData('autoscrobbleError', 'failed');
scrobble_statusbar('failed');
}
if (us_getValue('asFailNotification') || us_getValue('asFailNotification') == 'yes') {
us_showBox();
}
}
function scrobble_statusbar(status) {
let statusBar = document.getElementById('us_scrobble_statusbar');
if (status == 'scrobble') {
statusBar.classList.remove('us_status_hidden');
statusBar.classList.remove('us_status_failed');
} else if (status == 'failed') {
statusBar.classList.remove('us_status_hidden');
statusBar.classList.add('us_status_failed');
} else if (status == 'hide') {
statusBar.classList.add('us_status_hidden');
}
}
/**
* Redirects the user to Last FM to authenticate. When they allow they will
* be directed back and an authentication token will be added to the URL.
* adapted from ScrobbleSmurf
*/
function us_authenticate() {
let tokenRegex = /(\?|&)token=[^&?]*/gi;
let currentURL = document.URL.replace(tokenRegex, '');
if (currentURL.indexOf('?') === -1) {
currentURL = currentURL.replace('&', '?');
}
let redirectURL = lastFmAuthenticationUrl + '?api_key=' + APIKEY + '&cb=' + currentURL;
window.location.href = redirectURL;
}
/**
* Attempts to get a Last FM token from the URL. If so authenticate the user.
* adapted from ScrobbleSmurf and edited
*/
function tryGetAuthToken() {
let url = currentURL;
let tokenRegex = /(\?|&)token=[^]{32}/gi;
let matches = url.match(tokenRegex);
if (matches == null) {
return;
}
let rawToken = matches[0];
let token = rawToken.substring(7); // 7, based on '?' or '&' and 'token='.
GM_xmlhttpRequest({
method: 'GET',
url: authenticationSessionUrl + '?token=' + token,
headers: {
'Accept': 'text/html'
},
onload: function(responseDetails) {
let feedback = responseDetails.responseText;
if (((feedback.indexOf('api-error')) == -1) && ((feedback.indexOf('token-error')) == -1)) {
let retrievedData = responseDetails.responseText.split(' - ');
us_saveValue('us_username', retrievedData[0]);
us_saveValue('us_sessionKey', retrievedData[1]);
us_showBox(true);
} else {
us_showBox();
us_resetlogin(feedback);
}
}
});
}
/**
* Scrobbles a song using the saved track information
*/
function us_scrobble(artist, track, album, mbid, retry, queued, auto, full_album_scrobble) {
let secs = us_getTempData('us_secs');
if ((us_getTempData('scrobbled')) == 1 && !queued) {
us_saveTempData('us_leftToPlay', parseInt(us_getTempData('us_secs') * (us_getValue('scrobble_at')) * 0.01));
us_saveTempData('scrobbled', 0);
}
let args = '?artist=' + encodeURIComponent(artist) + '&sk=' + us_getValue('us_sessionKey') +
'×tamp=' + us_getTempData('us_playstart_s') + '&track=' + encodeURIComponent(track) + '&duration=' + secs + '&yt_vid_id=' + getYouTubeVideoId();
if (album != 0 && album != '') {
args += '&album=' + encodeURIComponent(album);
}
if (mbid != 0 && mbid != '') {
mbid == false;
args += '&mbid=' + encodeURIComponent(mbid);
}
let time_left_to_scrobble = us_getTempData('us_leftToPlay');
if (time_left_to_scrobble > 0) {
TO1Helper = true;
if (retry == 0) {
if (auto == 1 && us_getValue('scrobblingNotification')) {
us_infoBox(`<div><span class="us_trackinfo"><span id="us_artist_display">${artist}</span> <span class="sep"> - </span> <span id="us_track_display">${track}</span></span> ${time_left_to_scrobble} s <img src="" alt="queued" /></div>`);
window.setTimeout(function() {
us_closeinfobox();
}, 5000);
} else if (us_getValue('scrobblingNotification')) {
us_boxcontent('Queued...', `<div class="us_done">This will be scrobbled in ${time_left_to_scrobble} seconds. </div>`);
window.setTimeout(function() {
us_closebox();
}, 3000);
}
}
} else {
TO1Helper = false;
if (queued != 1) {
us_boxcontent('Scrobbling...', loadgif);
}
GM_xmlhttpRequest({
method: 'GET',
url: scrobbleSongUrl + args,
onload: function(responseDetails) {
scrobbleFeedback(responseDetails, artist, track, queued, full_album_scrobble);
},
onerror: function() {
GM_xmlhttpRequest({
method: 'GET',
url: scrobbleSongUrl + args,
onload: function(responseDetails) {
scrobbleFeedback(responseDetails, artist, track, queued, full_album_scrobble);
},
onerror: function() {
us_infoBox('<div class="us_error">Server error</div>');
window.setTimeout(function() {
us_closeinfobox();
}, 10000);
}
});
}
});
}
}
/**
* Feedback on scrobbling
*/
function scrobbleFeedback(responseDetails, artist, track, queued, full_album_scrobble) {
let feedback = responseDetails.responseText;
us_saveTempData('scrobbled', 1);
if ((feedback.indexOf('<lfm status="ok"')) != -1) {
TO1Helper = false;
if (document.getElementById('scrobbleStatus_parent')) {
document.getElementById('scrobbleStatus_parent').innerHTML = 'scrobbled';
}
if (queued != 1) {
us_boxcontent('OK!', `<div class="us_done"><span class="us_trackinfo"><span id="us_artist_display">${artist}</span> <span class="sep">-</span> <span id="us_track_display">${track}</span></span> scrobbled.</div>`);
window.setTimeout(function() {
us_closebox();
}, 2000);
} else {
if (us_getValue('scrobblingNotification', 0)) {
us_infoBox(`<div><span class="us_trackinfo">${artist} <span class="sep">-</span> ${track}</span> scrobbled. <img src="%2BpODGCLbrfsyH5BAAAAAAALAAAAAAQABAAAANhCLrcHmKUEZw6g4YtT8PEERBEcBCFtwyh4L5nsQTD8cLCURCKducKkgxQgACBAIIgYFAUXQ3CYNkkjgS8YIZUpdlYyQxlt9jRxBla9WLIKVlq1eJgMI8KBnnUwDdkLYAMCQA7" alt="done" /></div>`);
window.setTimeout(function() {
us_closeinfobox();
}, 3000);
}
}
} else {
if (queued != 1) {
us_boxcontent('Error', `<div class="us_error">${feedback}</div>`);
} else {
us_infoBox(`<div class="us_error">Error: ${feedback}</div>`);
window.setTimeout(function() {
us_closeinfobox();
}, 10000);
}
}
if (full_album_scrobble || us_getTempData('is_full_album') == 'yes') {
TO1Helper = true;
us_saveTempData('scrobbled', 0);
scrobble_statusbar('hide');
us_closebox();
let global_album = JSON.parse(us_getTempData('global_album'));
let track_num = parseInt(us_getTempData('full_album_track_nr'));
us_saveTempData('artist', global_album.tracks.track[track_num].artist.name);
us_saveTempData('track', global_album.tracks.track[track_num].name);
us_saveTempData('us_secs', global_album.tracks.track[track_num].duration);
let album_left_to_play = global_album.tracks.track[track_num].duration - 1;
let time = new Date();
us_saveTempData('us_leftToPlay', album_left_to_play);
us_saveTempData('us_playstart_s', Math.round(time.getTime() / 1000));
track_num++;
us_saveTempData('full_album_track_nr', track_num);
us_scrobble(decodeURIComponent(us_getTempData('artist')), decodeURIComponent(us_getTempData('track')), decodeURIComponent(us_getTempData('album')), decodeURIComponent(us_getTempData('mbid')), 0, 1, 1, 1);
}
}
/**
* Temporary save track information from the form
*/
function us_scrobblenp() {
let formArtist = document.forms[0].elements[0].value;
let formTrack = document.forms[0].elements[1].value;
let formAlbum = document.forms[0].elements[2].value;
if ((formArtist != '') && (formTrack != '')) {
if (formArtist != decodeURIComponent(us_getTempData('artist')) || formTrack != decodeURIComponent(us_getTempData('track')) || formAlbum != decodeURIComponent(us_getTempData('album'))) {
saveDatabaseData(getYouTubeVideoId(), formArtist, formTrack, formAlbum);
}
us_saveTempData('artist', encodeURIComponent(formArtist));
us_saveTempData('track', encodeURIComponent(formTrack));
if (!formAlbum) {
formAlbum = '';
} else {
us_saveTempData('album', encodeURIComponent(formAlbum));
}
us_scrobble(decodeURIComponent(us_getTempData('artist')), decodeURIComponent(us_getTempData('track')), formAlbum, '', 0, 0, 0);
} // empty input
}
/**
* Abort scrobbling process
*/
function us_abortScrobbling() {
if (TO1Helper) {
TO1Helper = false;
if (document.getElementById('scrobbleStatus_parent')) {
let element = document.getElementById('scrobbleStatus_parent');
element.parentNode.removeChild(element);
}
}
scrobble_statusbar('hide');
}
/**
* Unset the saved login info + show login form, and maybe show errors
*/
function us_resetlogin(error) {
us_saveValue('us_username', '');
us_saveValue('us_sessionKey', '');
let cont = '';
let resetInfo = '';
if (!error) {
resetInfo = '<div class="us_done">Successfully reset the login credentials</div><br />';
}
if ((error != '[object MouseEvent]') && (error != '[object XPCNativeWrapper [object MouseEvent]]')) {
cont = `<p class="us_error">Error: ${error}</p>`;
}
cont += `
<div id="us_loginbox_form">
${resetInfo}
<span>Click Login below to authenticate your account</span><br/><br/>
<em>Note: You will leave this site and be redirected here after having logged in to Last.fm.</em><br/><br />
</div>
<div class="us_submitbuttons"><input id="us_submit" value="Authenticate" type="submit" /></div>
`;
us_boxcontent('Login to last.fm', cont);
document.getElementById('us_submit').addEventListener('click', us_authenticate);
}
/**
* --- 5. Information request ---
*/
/**
* Check whether user credentials are stored or not.
*/
function isLoggedIn() {
if ((!us_getValue('us_username', 0)) || (!us_getValue('us_sessionKey', 0))) {
return false;
}
return true;
}
/**
* Gets the current YouTube video ID from the browser URL.
*/
function getYouTubeVideoId() {
let regex = /(\?|%3F|&|%26)v=[^?&#]*/gi;
let removeRegex = /(\?|%3F|&|%26)v=/gi;
let matches = document.URL.match(regex);
if (matches == null) {
let playerNode;
if (document.getElementById('c4-player')) {
playerNode = document.getElementById('c4-player');
} else {
playerNode = document.getElementById('movie_player');
}
matches = playerNode.getVideoUrl().match(regex);
if (matches == null) {
return null;
}
}
let vidId = matches[0].replace(removeRegex, '');
return vidId;
}
/**
* Detects the track information from the video title and temporarily saves it
*/
function getTrackInfo() {
let feedback;
if ((us_getTempData('artist') != 0) || (us_getTempData('track') != 0)) {
feedback = 'found';
} else {
let titleContentOriginal;
if (location.href.indexOf('youtube.com/user/') != -1) {
if (document.getElementById('playnav-curvideo-title')) {
titleContentOriginal = document.getElementById('playnav-curvideo-title').getElementsByTagName('a')[0].textContent;
} else if (document.getElementsByClassName('channels-featured-video-details tile')[0]) {
titleContentOriginal = document.getElementsByClassName('channels-featured-video-details tile')[0].getElementsByTagName('a')[0].textContent;
}
} else {
// Feather check
if (document.getElementsByClassName('title style-scope ytd-video-primary-info-renderer')[0]) {
titleContentOriginal = document.getElementsByClassName('title style-scope ytd-video-primary-info-renderer')[0].textContent;
} else if (document.getElementById('eow-title')) {
titleContentOriginal = document.getElementById('eow-title').textContent;
} else if (document.getElementById('watch-headline-title')) {
titleContentOriginal = document.getElementById('watch-headline-title').textContent;
} else if (document.getElementById('vt')) {
titleContentOriginal = document.getElementById('vt').textContent;
}
}
// Retrieve track information from database
if (getDatabaseData() == true) {
feedback = 'found';
trackInfoFromDB = true;
} else {
// New detection of track information
// remove (*) and/or [*] to remove unimportant data
let titleContent = titleContentOriginal.replace(/ *\([^)]*\) */g, ' ');
titleContent = titleContent.replace(/ *\[[^)]*\] */g, ' ');
// remove HD info
titleContent = titleContent.replace(/\W* HD( \W*)?/, '');
titleContent = titleContent.replace(/\W* HQ( \W*)?/, '');
// get remix info
let remixInfo = titleContentOriginal.match(/\([^)]*(?:remix|mix|cover|version|edit|booty?leg)\)/i);
let musicInfo = titleContent.split(' - ');
if (musicInfo.length == 1) {
musicInfo = titleContent.split('-');
}
if (musicInfo.length == 1) {
musicInfo = titleContent.split('–');
}
if (musicInfo.length == 1) {
musicInfo = titleContent.split(':');
}
if (musicInfo.length == 1) {
musicInfo = titleContent.split(' "');
}
// format feat. info
for (let i = 0; i < musicInfo.length; i++) {
musicInfo[i] = musicInfo[i].replace(/ feat. /, ' feat. ');
musicInfo[i] = musicInfo[i].replace(/ feat /, ' feat. ');
musicInfo[i] = musicInfo[i].replace(/ ft. /, ' feat. ');
musicInfo[i] = musicInfo[i].replace(/ ft /, ' feat. ');
}
// remove " and ' from musicInfo
for (let i = 0; i < musicInfo.length; i++) {
musicInfo[i] = musicInfo[i].replace(/^\s*"|"\s*$/g, '');
musicInfo[i] = musicInfo[i].replace(/^\s*'|'\s*$/g, '');
}
if ((musicInfo.length == 1) || (musicInfo[0] == false) || (musicInfo[1] == false)) {
musicInfo[0] = '';
musicInfo[1] = '';
feedback = 'notFound';
} else {
feedback = 'found';
}
musicInfo[1] = musicInfo[1].replace(/(\.avi)$/gi, '');
musicInfo[1] = musicInfo[1].replace(/(\.wmv)$/gi, '');
musicInfo[1] = musicInfo[1].replace(/(\.mp4)$/gi, '');
musicInfo[1] = musicInfo[1].replace(/(\.mpeg4)$/gi, '');
musicInfo[1] = musicInfo[1].replace(/(\.mov)$/gi, '');
musicInfo[1] = musicInfo[1].replace(/(\.3gpp)$/gi, '');
musicInfo[1] = musicInfo[1].replace(/(\.flv)$/gi, '');
musicInfo[1] = musicInfo[1].replace(/(\.webm)$/gi, '');
// Full Album Video
if (titleContentOriginal.match(/Full Album/i)) {
musicInfo[1] = musicInfo[1].replace(/Full Album/i, '');
} else {
// move feat. info from artist to track
if (musicInfo[0].match(/ feat.* .*/)) {
musicInfo[1] = musicInfo[1] + musicInfo[0].match(/ feat.* .*/);
musicInfo[0] = musicInfo[0].replace(/ feat.* .*/, '');
}
// add remix info
if (remixInfo && remixInfo.length == 1) {
musicInfo[1] += ' ' + remixInfo[0];
}
}
// delete spaces
musicInfo[0] = musicInfo[0].replace(/^\s\s*/, '').replace(/\s\s*$/, '');
musicInfo[1] = musicInfo[1].replace(/^\s\s*/, '').replace(/\s\s*$/, '');
if (us_getValue('us_autoscrobble_active', 0) == 1) {
if ((musicInfo.length != 2)) {
feedback = 'bad';
}
}
if (!us_getTempData('artist') && musicInfo[0] != 0) {
us_saveTempData('artist', encodeURIComponent(musicInfo[0]));
}
if (!us_getTempData('track') && musicInfo[1] != 0) {
us_saveTempData('track', encodeURIComponent(musicInfo[1]));
}
}
// Full Album Video
if (titleContentOriginal.match(/Full Album/i)) {
us_saveTempData('is_full_album', 'yes');
getAlbumInfo();
}
}
return feedback;
}
/**
* Fetch full album info from last.fm API
*/
function getAlbumInfo() {
let artist = decodeURIComponent(us_getTempData('artist'));
let albumName = decodeURIComponent(us_getTempData('track'));
let url = 'http://ws.audioscrobbler.com/2.0/?method=album.getinfo&api_key=' + APIKEY + '&artist=' + artist.replace(' ', '+') + '&album=' + albumName.replace(' ', '+') + '&format=json';
GM_xmlhttpRequest({
method: 'GET',
url: url,
onload: function(response) {
// console.log(response); //TODO: if Album not found -> reset
if (response.responseText) {
let json = JSON.parse(response.responseText);
response = response.responseText;
let album = json.album;
if (album.artist == artist && album.name == albumName && album.tracks.track.length > 1) {
us_saveTempData('is_full_album', 'yes');
us_saveTempData('global_album', JSON.stringify(album));
us_saveTempData('full_album_track_nr', 1);
let track_index = 0;
us_saveTempData('full_album_track_count', album.tracks.track.length);
us_saveTempData('us_secs', album.tracks.track[track_index].duration);
let album_left_to_play = album.tracks.track[track_index].duration - 1;
us_saveTempData('us_leftToPlay', album_left_to_play);
us_saveTempData('artist', album.tracks.track[track_index].artist.name);
us_saveTempData('track', album.tracks.track[track_index].name);
response = getTrackInfo();
isMusicVideo(response);
} else {
us_saveTempData('is_full_album', 0);
}
}
},
onerror: function() {
// alert("failed");
},
ontimeout: function() {
// tryAutoScrobbleCallback(infoResult, false);
}
});
}
/**
* database track information
*/
function getDatabaseData() {
let id = getYouTubeVideoId();
if ((us_getValue('database.id', 0) != 0) && (us_getValue('database.id', 0).search(id) != -1)) {
let ids = us_getValue('database.id', 0).split(' ');
let artists = us_getValue('database.artist', 0).split(' ');
let tracks = us_getValue('database.track', 0).split(' ');
let index = 0;
for (let i = 0; i < ids.length; i++) {
if (ids[i] == id) {
index = i;
i = ids.length;
}
}
if (!us_getTempData('artist')) {
us_saveTempData('artist', artists[index]);
}
if (!us_getTempData('track')) {
us_saveTempData('track', tracks[index]);
}
ids.splice(ids.length, 0, ids[index]);
ids.splice(index, 1);
artists.splice(artists.length, 0, artists[index]);
artists.splice(index, 1);
tracks.splice(tracks.length, 0, tracks[index]);
tracks.splice(index, 1);
us_saveValue('database.id', ids.join(' '));
us_saveValue('database.artist', artists.join(' '));
us_saveValue('database.track', tracks.join(' '));
// retrieve Album title
if (us_getValue('database_additional.id', 0).search(id) != -1) {
let Aids = us_getValue('database_additional.id', 0).split(' ');
let albumtitles = us_getValue('database_additional.albumtitle', 0).split(' ');
let index = 0;
for (let i = 0; i < Aids.length; i++) {
if (Aids[i] == id) {
index = i;
i = Aids.length;
}
}
if (!us_getTempData('album')) {
us_saveTempData('album', albumtitles[index]);
}
}
return true;
} else {
return false;
}
}
function saveDatabaseData(id, artist, track, album) {
// Edit existing entry
if ((us_getValue('database.id', 0) != 0) && (us_getValue('database.id', 0).search(id) != -1)) {
let ids = us_getValue('database.id', 0).split(' ');
let artists = us_getValue('database.artist', 0).split(' ');
let tracks = us_getValue('database.track', 0).split(' ');
let index = 0;
for (let i = 0; i < ids.length; i++) {
if (ids[i] == id) {
index = i;
i = ids.length;
}
}
ids.splice(ids.length, 0, ids[index]);
ids.splice(index, 1);
artists.splice(artists.length, 0, encodeURIComponent(artist));
artists.splice(index, 1);
tracks.splice(tracks.length, 0, encodeURIComponent(track));
tracks.splice(index, 1);
us_saveValue('database.id', ids.join(' '));
us_saveValue('database.artist', artists.join(' '));
us_saveValue('database.track', tracks.join(' '));
} else {
// New entry
let ids = us_getValue('database.id', 0).split(' ');
if ((us_getValue('database.maxEntries', 5000) == '-1') || (ids.length < us_getValue('database.maxEntries', 5000))) {
// New Entry
us_saveValue('database.id', (us_getValue('database.id', 0) + ' ' + id));
us_saveValue('database.artist', (us_getValue('database.artist', 0) + ' ' + encodeURIComponent(artist)));
us_saveValue('database.track', (us_getValue('database.track', 0) + ' ' + encodeURIComponent(track)));
} else {
// Already maximum number of entries -> delete oldest and insert new
let artists = us_getValue('database.artist', 0).split(' ');
let tracks = us_getValue('database.track', 0).split(' ');
ids.splice(ids.length, 0, id);
ids.splice(0, 1);
artists.splice(artists.length, 0, encodeURIComponent(artist));
artists.splice(0, 1);
tracks.splice(tracks.length, 0, encodeURIComponent(track));
tracks.splice(0, 1);
us_saveValue('database.id', ids.join(' '));
us_saveValue('database.artist', artists.join(' '));
us_saveValue('database.track', tracks.join(' '));
}
}
// Save additional information about the track (Album title)
if (album != 0) {
if ((us_getValue('database_additional.id', 0) != 0) && (us_getValue('database_additional.id', 0).search(id) != -1)) {
let Aids = us_getValue('database_additional.id', 0).split(' ');
let albumtitles = us_getValue('database_additional.albumtitle', 0).split(' ');
let index = 0;
for (let i = 0; i < Aids.length; i++) {
if (Aids[i] == id) {
index = i;
i = Aids.length;
}
}
Aids.splice(Aids.length, 0, id);
Aids.splice(index, 1);
albumtitles.splice(albumtitles.length, 0, encodeURIComponent(album));
albumtitles.splice(index, 1);
us_saveValue('database_additional.id', Aids.join(' '));
us_saveValue('database_additional.albumtitle', albumtitles.join(' '));
} else {
us_saveValue('database_additional.id', us_getValue('database_additional.id', '') + ' ' + encodeURIComponent(id));
us_saveValue('database_additional.albumtitle', us_getValue('database_additional.albumtitle', '') + ' ' + encodeURIComponent(album));
}
}
}
/**
* --- 6. Miscellaneous ---
*/
/**
*
*
*/
function us_ajax_scanner() {
let leftToPlay = parseInt(us_getTempData('us_leftToPlay'));
let secs = parseInt(us_getTempData('us_secs'));
let scrobble_at = parseInt(us_getValue('scrobble_at'));
if (!getYouTubeVideoId()) {
us_saveTempData('us_reset_now', '1');
setTimeout(function() {
us_ajax_scanner();
}, 5000);
} else {
setTimeout(function() {
us_ajax_scanner();
}, 1000);
if (us_getTempData('video_is_playing', 0) == 1 && us_getTempData('us_reset_now') != '1' && leftToPlay >= 1 && !us_getTempData('autoscrobleerror')) {
us_saveTempData('us_leftToPlay', parseInt(leftToPlay - 1));
if (document.getElementById('scrobbleStatus')) {
document.getElementById('scrobbleStatus').innerHTML = leftToPlay - 1;
}
if (TO1Helper) {
scrobble_statusbar('scrobble');
if (us_getTempData('is_full_album') == 'yes') {
document.getElementById('us_scrobble_statusbar').style.width = Math.round(100 - 100 * ((leftToPlay - 1) / (secs * (100) * 0.01))) + '%';
} else {
document.getElementById('us_scrobble_statusbar').style.width = Math.round(100 - 100 * ((leftToPlay - 1) / (secs * ((scrobble_at) * 0.01)))) + '%';
}
}
}
if (us_getTempData('video_end_reached') == 'yes' && us_getTempData('video_is_playing', 0) == 1) {
us_saveTempData('us_reset_now', '1');
us_saveTempData('video_end_reached', 0);
}
leftToPlay = us_getTempData('us_leftToPlay');
if (leftToPlay <= 0 && us_getTempData('scrobbled') != 1 && TO1Helper && secs > 30) {
leftToPlay = 0;
TO1Helper = false;
if (document.getElementById('scrobbleStatus')) {
document.getElementById('scrobbleStatus_parent').innerHTML = 'submitting...';
}
us_scrobble(decodeURIComponent(us_getTempData('artist')), decodeURIComponent(us_getTempData('track')), decodeURIComponent(us_getTempData('album')), decodeURIComponent(us_getTempData('mbid')), 0, 1, 1);
}
// check for reset -> ajax youtube change
if (us_getTempData('us_reset_now') == '1') {
us_saveTempData('us_reset_now', '0');
us_reset();
}
}
}
/**
* Quickchange artist ↔ track in scrobble-form
*/
function us_quickchange() {
let artist = document.forms[0].elements[0].value;
document.forms[0].elements[0].value = document.forms[0].elements[1].value;
document.forms[0].elements[1].value = artist;
}
/**
* Checks if its first run or if updated
*/
function checkFirstRun() {
let localVersion = us_getValue('us_local_version', 0);
if (localVersion == 0) {
initPreferences();
us_showBox();
let cont = `
<div id="us_loginbox_form">
<h4>Welcome to YouScrobbler!</h4><br/>
<span>Join the <a target="_blank" href="http://www.last.fm/group/YouScrobbler">Last.fm Group</a> to stay up to date.</span><br/><br/>
<span>Description and documentation can be found on the <a target="_blank" href="http://www.lukash.de/youscrobbler">Homepage</a>.</span><br/><br/>
</div>
<div class="us_submitbuttons"><input id="us_submit" value="Next" type="submit" /></div>
`;
us_boxcontent('First Run', cont);
document.getElementById('us_submit').addEventListener('click', us_showBox);
us_saveValue('us_local_version', VERSION);
} else if (localVersion < VERSION) {
initPreferences();
us_showBox();
let cont = `
<div id="us_loginbox_form">
<div class="us_done">Welcome to YouScrobbler ${VERSION}.</div><br/>
<span>Changelog can be found on the <a target="_blank" href="http://www.lukash.de/youscrobbler">Web page</a>.</span><br/><br/><br/>
<h4>Join the <a target="_blank" href="http://www.last.fm/group/YouScrobbler" title="Last.fm Group">Last.fm Group</a> to stay tuned</h4><br />
</div>
<div class="us_submitbuttons"><input id="us_submit" value="Close" type="submit" /></div>
`;
us_boxcontent('Successfully Updated', cont);
document.getElementById('us_submit').addEventListener('click', us_closebox);
us_saveValue('us_local_version', VERSION);
}
}
/**
* --- 7. Update ---
*
* Edited version of Script Update Checker (http://userscripts.org/scripts/show/20145)
*/
function updateCheck(forced) {
let update_interval = 86400000;
if ((forced) || ((parseInt(us_getValue('us_last_update', '0')) + update_interval) <= parseInt(((new Date()).getTime())))) { // Checks every day (24 h * 60 m * 60 s * 1000 ms)
try {
GM_xmlhttpRequest(
{
method: 'GET',
url: updateUrl(),
onload: function(resp) {
let local_version, remote_version, response;
response = resp.responseText;
us_saveValue('us_last_update', new Date().getTime() + '');
remote_version = response.split(' ')[0];
local_version = VERSION;
let scriptDownloadUrl = 'http://youscrobbler.lukash.de/youscrobbler_' + remote_version.replace(/\./g, '') + '.user.js';
if (remote_version > local_version) {
let cont = `
<div id="us_loginbox_form">
<span>YouScrobbler ${remote_version} is available.</span><br/>
<span>Changes are applied after new page load.</span><br/><br/>
Problem updating?<br/>
Try via Greasemonkey (Addons/UserScripts)
</div>
<div class="us_submitbuttons"><input id="us_submit" value="Install Update" type="submit" /></div>
`;
us_boxcontent('Update available', cont);
document.getElementById('us_submit').addEventListener('click', function() {
window.open(scriptDownloadUrl, '_blank');
});
} else if (forced) {
document.getElementById('us_manualupdate').firstChild.innerHTML = 'No Update available';
}
},
onerror: function() {
if ((forced)) {
let cont = `
<div id="us_loginbox_form">
<div class="us_error">Checking for an update has failed. Try again later.</div><br/><br/><br/>
</div>
<div class="us_submitbuttons"><input id="us_submit" value="Check again" type="submit" /></div>
`;
us_showBox();
us_boxcontent('Checking for update failed', cont);
document.getElementById('us_submit').addEventListener('click', function() {
updateCheck(true);
});
} else {
us_infoBox('<div class="us_error">Checking for update failed</div>');
window.setTimeout(function() {
us_closeinfobox();
}, 5000);
}
}
});
} catch (err) {
if ((forced)) {
let cont = `
<div id="us_loginbox_form"><div class="us_error">Checking for an Updated has failed. Try again later.<br/>
Error:<br/>
${err}
</div></div><br/><br/><br/>
<div class="us_submitbuttons"><input id="us_submit" value="Check again" type="submit" /></div>
`;
us_showBox();
us_boxcontent('Checking for update failed', cont);
document.getElementById('us_submit').addEventListener('click', function() {
updateCheck(true);
});
} else {
us_infoBox(`<div class="us_error">Checking for update failed: ${err}</div>`);
window.setTimeout(function() {
us_closeinfobox();
}, 5000);
}
}
}
}
if (window.top === window.self) {
init();
updateCheck(false);
}