Monday, December 5, 2011

ScrobbleSmurf


// ==UserScript==
// @name           SrobbleSmurf
// @namespace      scrobble
// @description    Scrobble songs from YouTube.
// @include        http://youtube.com/watch?*v=*
// @include        http://www.youtube.com/watch?*v=*
// @include        http://youtube.com/watch#!*v=*
// @include        http://www.youtube.com/watch#!*v=*
// Version 1.1.2
// ==/UserScript==
/* You can contact me on Twitter @DDaanV or Last.FM (DDaan90).
 * - Don't steal all of my code and say it's yours. 
 * - You may steal parts of my code. Credits are not required. 
 * - Suggestions and improvements are welcome. 
 * - This code is free, when you take parts of my code expect it will remain free.
 */
/* TODO's: 
 * - Create more reliable timer (use system time).
 * - Use JSON parser instead of 'eval' which should increase performance.
 * - Server-side: Attemp to swap artist and title when scrobbling failed.
 * - Scrobble from YouTube user channels. (http://www.youtube.com/user/* && http://youtube.com/user/*)
 * - Better error handling. 
 */
var cosmicPanda = false;
var showTime = 0;
var scrobLength = -1;
var noScrobble = false;
var iTunesArtistAndTitleFound = false;

var scrobbleAutoCollapse = true;
var scrobblePercentage = 50;
var scrobbleEnabled = true;
var scrobbleFilter = "";

var apiKey = "870da3cecaf2a63062b474a93ace2285";
var lastFmAuthenticationUrl = "http://www.last.fm/api/auth";
var authenticationSessionUrl = "http://daan.hostei.com/lastfm/auth/getSession";
var scrobbleSongUrl = "http://daan.hostei.com/lastfm/scrobbleSong";
var altScrobbleSongUrl = "http://ddaan.nl/lastfm/scrobbleSong";
var youtubeApiUrl = "http://gdata.youtube.com/feeds/api/videos/";

//--General functions.
/**
 * Initialisation function, should be called only once when the document is loaded. 
 * The ScrobbleSmurf plug-in will be added to the YouTube page.
 */
function initialize() {
 try {
  localStorageEnabledCheck();
  fixVideoEmbedOpaqueness();
  loadSavedFormValues();
  cosmicPandaLayout();
  addStyle();
  addLastFmOptions();
  addLastFmButton();
  addMenuPop();
  activateLastFmButtons();
  var song = getSongInformation();
  fillSongInformation(song);
  videoApiRequest();
  
  if(!isLoggedIn()) {
   showLoginHideReset();
   tryGetAuthToken();
  } else {
   setUsernameSpan();
   hideLoginShowReset();
  }
 } catch (e) {
  console.log(e);
 }
}

/**
 * Shows a message in the ScrobbleSmurf menu.
 */
function menuAlert(text) {
 text = text.replace(/\n/gi, "<br/>");
 var scrobbCount = document.getElementById("scrobble-counter");
 var scrobbMessg = document.getElementById("scrobble-message");
 var scrobbMsg = document.getElementById("scrobble-msg");
 scrobbCount.style.display = "none";
 scrobbMessg.style.display = "block";
 scrobbMsg.innerHTML = text + ".";
}
//--Google Chrome compatibility functions.
/**
 * Alerts the user when the browser does not support HTML5 local storage.
 */
function localStorageEnabledCheck()
{
 var localStorageEnabled = window.localStorage;
 if(!localStorageEnabled) {
  alert("Your browser does not support HTML5 local storage.\n" +
   "This is required to use ScrobbleSmurf V1.1 and up.");
 }
}
/**
 * Adds wmode=opaque to which the video is embedded. 
 * Used to overlay the options in Google Chrome.
 */
function fixVideoEmbedOpaqueness()
{
  var wmodeParam = document.createElement("param");
  wmodeParam.setAttribute("name", "wmode");
  wmodeParam.setAttribute("value", "opaque");

  var div = document.getElementById("watch-player");
  var videoEmbed = div.getElementsByTagName("embed")[0];
  var parent = videoEmbed.parentNode;
  
  parent.removeChild(videoEmbed);
  
  videoEmbed.setAttribute("wmode", "opaque");
  videoEmbed.appendChild(wmodeParam);
  
  parent.appendChild(videoEmbed);
}

//--Functions for the additional menu pop up.
/**
 * Adds the more options pop up and background to the YouTube page. 
 * By default the pop up is hidden, to show it call the moreOptions method.
 */
function addMenuPop()
{
 var page = document.getElementById("page");
 var pop = document.createElement("div");
 pop.setAttribute("id", "lastfm-moreoptions-pop");
 pop.style.visibility = "hidden";
 pop.innerHTML = "<h2>More Last FM Options</h2>\n" + 
 "<form id=\"lastfm-moreoptions-form\">\n" +
 "<table class=\"lastfm-moreoptions-table\"><tr>\n" +
 " <td>Collapse ScrobbleSmurf menu</td>\n" +
 " <td><input name=\"lastfm-moreoptions-collapse\" type=\"radio\" value=\"collapse\">Collapse the menu by default.<br/>\n" + 
 " <input name=\"lastfm-moreoptions-collapse\" type=\"radio\" value=\"expand\">Expand the menu by default.</td>\n" +
 "</tr><tr>\n" +
 " <td>Scrobble percentage</td>\n" +
 " <td><select style=\"width:100%\" name=\"lastfm-moreoptions-percentage\">\n" +
 "  <option value=\"20\">20 percent</option>\n" +
 "  <option value=\"30\">30 percent</option>\n" +
 "  <option value=\"40\">40 percent</option>\n" +
 "  <option value=\"50\">50 percent</option>\n" +
 "  <option value=\"60\">60 percent</option>\n" +
 "  <option value=\"70\">70 percent</option>\n" +
 "  <option value=\"80\">80 percent</option>\n" +
 " </select></td>\n" +
 "</tr><tr>\n" +
 " <td>Words to remove<br/><i>(separate by semicolon)<i></td>\n" +
 " <td><input style=\"width:100%\" id=\"lastfm-moreoptions-filter\" name=\"lastfm-moreoptions-filter\" type=\"text\"></input></td>\n" +
 "</tr><tr>\n" +
 " <td>Enable all scrobbling</td>\n" +
 " <td><input name=\"lastfm-moreoptions-scrobble\" type=\"radio\" value=\"yes\">Yes, scrobble my songs to Last FM<br/>\n" + 
 " <input name=\"lastfm-moreoptions-scrobble\" type=\"radio\" value=\"no\">No, do not scrobble my songs</td>\n" +
 "</tr></table>\n" +
 "</form>\n" +
 "<div class=\"lastfm-moreoptions-controls\">\n" + 
 " <button class=\"yt-uix-button yt-uix-tooltip\" id=\"lastfm-moreoptions-ok\" title=\"Save changes\"><span class=\"yt-uix-button-content\">Save</span></button>\n" +
 " <button class=\"yt-uix-button yt-uix-tooltip\" id=\"lastfm-moreoptions-cancel\" title=\"Discard changes\" style=\"margin-left: 14px;\"><span class=\"yt-uix-button-content\">Cancel</span></button>\n" +
 "</div>\n"; 
 page.appendChild(pop);
 addMenuBack(pop);
 centerPopup(pop);
 
 setDefaultSelectValue("lastfm-moreoptions-percentage", scrobblePercentage);
 activatePopupControls();
}
/**
 * Add the more options pop up background to the YouTube page.
 */
function addMenuBack(menuElement)
{
 var overlay = document.createElement("div");
 overlay.setAttribute("id", "lastfm-moreoptions-back");
 overlay.style.visibility = "hidden";
 menuElement.parentNode.appendChild(overlay);
 
 var body = document.body, html = document.documentElement;
 var height = Math.max(body.scrollHeight, body.offsetHeight, html.clientHeight, html.scrollHeight, html.offsetHeight);
 overlay.style.height = height + "px";
}
/**
 * Attemps to center the more options pop up.
 */
function centerPopup(popup) {
 var windowWidth = document.documentElement.clientWidth;
 var windowHeight = document.documentElement.clientHeight;
 
 var popupWidth = parseInt(popup.clientWidth);
 var popupHeight = parseInt(popup.clientHeight);
 
 popup.style.position = "absolute";
 var top = windowHeight/2-popupHeight/2;
 var left = windowWidth/2-popupWidth/2;
 popup.style.top = (top > 0 ? top : 0) + "px";
 popup.style.left = (left > 0 ? left : 0) + "px";
}
/**
 * Attemps to center the more options pop up.
 */
function activatePopupControls()
{
 var okButton = document.getElementById("lastfm-moreoptions-ok");
 var cancelButton = document.getElementById("lastfm-moreoptions-cancel");
 
 okButton.addEventListener("click", saveCurrentForm, false);
 cancelButton.addEventListener("click", function() { moreOptions(true); }, false);
}
/**
 * Saves the currently selected values from the more options menu pop up to the user's hard disk.
 */
function saveCurrentForm()
{
 var sform = document.getElementById("lastfm-moreoptions-form");
 
 var collapse = getSelectedRadioButton("lastfm-moreoptions-collapse");
 var percentage = sform.elements["lastfm-moreoptions-percentage"].value;
 var filter = sform.elements["lastfm-moreoptions-filter"].value;
 var enable = getSelectedRadioButton("lastfm-moreoptions-scrobble");
 
 localStorage.setItem("lastfmExpand", (collapse != "expand"));
 localStorage.setItem("lastfmPercentage", percentage);
 localStorage.setItem("lastfmFilter", filter);
 localStorage.setItem("lastfmScrobble", enable != "no");
  
 moreOptions(true); 
 loadSavedFormValues();
}
/**
 * Loads values from the more options menu pop up the user saved to it's hard disk. 
 */
function loadSavedFormValues()
{
 scrobbleAutoCollapse = localStorage.getItem("lastfmExpand");
 scrobblePercentage = localStorage.getItem("lastfmPercentage");
 scrobbleFilter = localStorage.getItem("lastfmFilter");
 scrobbleEnabled = localStorage.getItem("lastfmScrobble");
}
//--Functions to get additional video information from the YouTube API.
/**
 * Updates the cosmicPanda variable which is a different version of the YouTube layout.
 * Different layout requires different handling.
 */
function cosmicPandaLayout() {
 var r = document.getElementById("watch-frame-bottom");
 if(r != null) {
  cosmicPanda = true;
 }
}
/**
 * Gets the current YouTube video ID from the browser URL.
 */
function getYouTubeVideoId() {
 var regex = /(\?|&)v=[^\?&#]*/gi;
 var matches = document.URL.match(regex);
 if(matches == null) {
  return null;
 }
 var videoCode = matches[0].substring(3);
 return videoCode;
}
/**
 * Uses the YouTube API to get more information from the current video.
 */
function videoApiRequest() {
 var vidId = getYouTubeVideoId();
 if(vidId == null) {
  return;
 }
 GM_xmlhttpRequest({
  method: "GET",
  url: youtubeApiUrl + vidId + "?v=2&alt=jsonc",
  headers: {
   "Accept": "application/json"
  },
  onload: videoApiResponse,
  onerror: function() { "menuAlert(\"Could not connect \nto YouTube API.\")" }
 });
}
/**
 * Handles data returned from the YouTube API which provides more data from the video.
 */
function videoApiResponse(response) {
 var jsonObject = eval("(" + response.responseText + ")");
 var category = jsonObject.data.category.toLowerCase();
 if(countOccurencesInString(category, "music") < 1) {
  noScrobble = true;
  menuAlert("No music video");
  return;
 }
 
 if(scrobbleEnabled !== true && scrobbleEnabled !== "true") {
  dontScrobble();
 }
 
 scrobLength = Math.round(jsonObject.data.duration / (100 / scrobblePercentage) );
 timePlus();
 
 if(!iTunesArtistAndTitleFound) {
  var vidTitle = jsonObject.data.title;
  if(filterWords != null)
  {
   var filterWords = scrobbleFilter.split(";");
   
   for(word in filterWords) {
    var escapedFilterWords = escapeAllRegexSpecialChars(filterWords[word].trim());
    var filterRegex = new RegExp(escapedFilterWords, "gi");
    vidTitle = vidTitle.replace(filterRegex, "");
   }
  }
  
  var song = tryStripSongFromTitle(vidTitle);
  fillSongInformation(song);
 }
}
/**
 * Escapes all characters with a special meaning in Regular Expressions.
 */
function escapeAllRegexSpecialChars(regexString)
{
 var specials = [
      '/', '.', '*', '+', '?', '|',
      '(', ')', '[', ']', '{', '}', '\\'
    ];
    var regex = new RegExp(
      '(\\' + specials.join('|\\') + ')', 'g'
    );
 
 regexString= regexString.replace(regex, '\\$1'); 
 return regexString;
}
//--Scrobbling functions.
/**
 * Adds one to timer and updates the textspans. When the timeLeft is 0 (or less the song will be scrobbled).
 */
function timePlus() {
 showTime++;
 var timeLeft = scrobLength - showTime;
 
 var timeSpan = document.getElementById("scrobble-time");
 timeSpan.innerHTML = timeLeft;
 
 if(timeLeft > 0) {
  if(!noScrobble) {
   setTimeout(timePlus, 1000);
  }
 } else {
  scrobbleSong();
 }
}
/**
 * Srobbles a song using the values from the two textinput's from the form.
 */
function scrobbleSong() {
 var artist = document.getElementById("lastfm-artist").value;
 var title = document.getElementById("lastfm-title").value;
 if(artist == "" || title == "") {
  menuAlert("Insufficient artist \nand title information");
  return;
 }

 if(!isLoggedIn())
 {
  menuAlert("Please authenticate \nbefore scrobbling");
 }
 else
 {
  var user = getUserCredentials();
  var timestamp = Math.round(new Date().getTime()/1000.0);
  
  var args = "?artist=" + encodeURIComponent(artist) + "&sk=" + user.sessionkey + 
  "&timestamp=" + timestamp + "&track=" + encodeURIComponent(title);
  
  menuAlert("Scrobbling song");
   
  GM_xmlhttpRequest({
   method: "GET",
   url: altScrobbleSongUrl + args,
   onload: scrobbleFeedback,
   onerror: function() {
    GM_xmlhttpRequest({
     method: "GET",
     url: scrobbleSongUrl + args,
     onload: scrobbleFeedback,
     onerror: function() {
      menuAlert("Scrobbling failed");
     }
    });
   }
  });
 }
}
/**
 * Show the user a response to their Scrobble attempt.
 */
function scrobbleFeedback(response) {
 var text;
 if(countOccurencesInString(response.responseText, "\r?\n") > 0) {
  //Split the response, since my hosting puts in some spam.
  text = response.responseText.split(/\r?\n/gim)[0].replace(/#/g, "");
 } else {
  text = response.responseText;
 }
 var jsonObject = eval("(" + text + ")");
 
 if(jsonObject.error != null) {
  menuAlert("Error connecting to Last.FM.\n" + 
  "Error code: " + jsonObject.error + "\n" +
  "Message: " + jsonObject.message + ".");
 } else {
  var artist = jsonObject.scrobbles.scrobble.artist.text;
  var title = jsonObject.scrobbles.scrobble.track.text;
  menuAlert("Scrobbled song\nArtist: " + artist + "\nTitle: " + title);
 }
}
//--Login functions.
/**
 * Attempts to get a Last FM token from the URL. If so authenticate the user.
 */
function tryGetAuthToken() {
 var url = document.URL;
 var tokenRegex = /(\?|&)token=[0-9a-fA-F]{32}/gi;
 var matches = url.match(tokenRegex);
 if(matches == null) {
  return;
 }
 var rawToken = matches[0];
 var token = rawToken.substring(7); //7, based on '?' or '&' and 'token='.
 GM_xmlhttpRequest({
  method: "GET",
  url: authenticationSessionUrl + "?token=" + token,
  headers: {
   "Accept": "text/html" //My hosting puts in html spam
  },
  onload: getSessionKey
 });
}
/**
 * Gets a Last FM session key and store it for later reference.
 */
function getSessionKey(response) {
 //Split the response, since my hosting puts in some spam.
 var text = response.responseText.split(/\r?\n/gim)[0];
 var jsonObject = eval("(" + text + ")");
 
 if(jsonObject.error != null) {
  menuAlert("Error connecting to Last.FM.\n" + 
  "Error code: " + jsonObject.error + "\n" +
  "Message: " + jsonObject.message + ".");
 } else {
  localStorage.setItem("lastfmUsername", jsonObject.session.name);
  localStorage.setItem("lastfmSessionkey", jsonObject.session.key);
  
  setUsernameSpan();
  hideLoginShowReset();
 }
}
/**
 * Check whether user credentials are stored or not.
 */
function isLoggedIn() {
 var username = localStorage.getItem("lastfmUsername");
 var sessionk = localStorage.getItem("lastfmSessionkey");
 if(username == null || sessionk == null) {
  return false;
 }
 return true;
}
/**
 * Gets the user's stored credentials.
 */
function getUserCredentials() {
 var user = new Object();
 user.username = localStorage.getItem("lastfmUsername");
 user.sessionkey = localStorage.getItem("lastfmSessionkey");
 return user;
}
/**
 * Fills in the user's username on the Last FM form.
 */
function setUsernameSpan() {
 var usernameSpan = document.getElementById("lastfm-username");
 usernameSpan.innerHTML = getUserCredentials().username;
}
/**
 * Hides the 'login'-form, shows the 'reset login'-form
 */
function hideLoginShowReset() {
 var reset = document.getElementById("lastfm-options-resetlogin");
 var login = document.getElementById("lastfm-options-login");
 login.style.display = "none";
 reset.style.display = "block";
}
/**
 * Shows the 'login'-form, hides the 'reset login'-form
 */
function showLoginHideReset() {
 var reset = document.getElementById("lastfm-options-resetlogin");
 var login = document.getElementById("lastfm-options-login");
 reset.style.display = "none";
 login.style.display = "block";
}
//--Functions to edit the YouTube lay-out.
/**
 * Adds CSS style to the page used by this plug in.
 */
function addStyle() {
 var head = document.getElementsByTagName("head")[0];
 var style = document.createElement("style");
 head.appendChild(style);
 style.setAttribute("type", "text/css");
 style.innerHTML = "#masthead-lastfm { border: 1px solid #CCC; background: #F6F6F6; }\n" +
 "#lastfm-container { margin: 0 auto; width: 970px; }\n" +
 "ul#lastfm-options li { height: 90px; float: left; margin: 8px 0px; padding: 0px 24px; border-right: 1px solid #CCCCCC; }\n" + 
 "ul#lastfm-options p { padding-left: 1px; }\n" +
 "#lastfm-moreoptions { margin-bottom: 6px; }\n" +
 "#lastfm-moreoptions-pop { height:296px; width:432px; background:#FFF; border:2px solid #CCC; z-index:4; padding:12px; font-size:13px; }\n" +
 "#lastfm-moreoptions-back { position:absolute; height:100%; width:100%; top:0; left:0; background:#000; border:1px solid #333; z-index:3; opacity: 0.6; }\n" + 
 ".lastfm-moreoptions-table { border-collapse:collapse; }\n" + 
 ".lastfm-moreoptions-table td { border-bottom:1px solid #EEE; padding:8px 6px 10px; }\n" +
 ".lastfm-moreoptions-controls { text-align:center; margin-top:6px; }\n"; 
}
/**
 * Adds a button to the page to toggle the Last FM form.
 */
function addLastFmButton() {
 var buttonId = "lastfm-button";
 var container = document.getElementById("masthead-user-bar-container");
 var lastFmImgCode = "%3D";
 var htmlCode = "<div style=\"margin-left: 6px;\" id=\"masthead-user-bar\">" +
 "   <div id=\"masthead-user\">" +
 " <span  name=\"lastfm-expander\" class=\"yt-uix-expander yt-uix-expander-collapsed\" id=\"masthead-user-expander\">" +
 "     <span id=\"lastfm-button\">" +
 "     <span tabindex=\"1\" class=\"yt-uix-expander-head yt-rounded\" id=\"masthead-user-wrapper\">" +
 "  <button role=\"button\" class=\"yt-uix-button yt-uix-button-text yt-uix-button-toggle\" id=\"masthead-user-button\" type=\"button\">" +
 "      <span class=\"yt-uix-button-content\">" +
 "   <span id=\"masthead-user-image\">" +
 "       <span class=\"clip\">" +
 "    <span class=\"clip-center\">" +
 "        <img alt=\"Last.FM\" src=\"%3D\">" +
 "        <span class=\"vertical-center\">" +
 "        </span>" +
 "    </span>" +
 "       </span>" +
 "   </span>" +
 "   <span class=\"yt-uix-expander-arrow\">" +
 "   </span>" +
 "      </span>" +
 "  </button>" +
 "     </span>" +
 "  </span>" +
 " </span>" +
 " </div>" +
 "</div>";
 
 container.innerHTML += htmlCode;
 var button = document.getElementById(buttonId);
 button.addEventListener("click", toggleLastFmForm, false);
 
 if(scrobbleAutoCollapse !== true && scrobbleAutoCollapse !== "true") {
  toggleLastFmForm();
  var buttonCont = document.getElementById("lastfm-button").parentNode;
  var newClass = buttonCont.getAttribute("class") + "yt-uix-expander-collapsed";
  buttonCont.setAttribute("class", newClass);
 }
}
/**
 * Adds a division with the Last FM menu. 
 */
function addLastFmOptions() {
 var swapImgCode = "";
 var scrobbleSmurfImgCode = "%3D";
 var htmlCode = "<div id=\"masthead-lastfm\" style=\"display: none;\">\n" +
 " <div id=\"lastfm-container\">\n" +
 "  <ul id=\"lastfm-options\">\n" +
 "  <li id=\"lastfm-options-resetlogin\">\n" +
 "   <h3>Authenticate</h3>\n" +
 "   <p>Logged in as: <br/><span id=\"lastfm-username\"></span></p>\n" +
 "   <button class=\"yt-uix-button yt-uix-tooltip\" id=\"lastfm-reset\" title=\"Log out from Last FM\"><span class=\"yt-uix-button-content\">Reset</span></button>\n" +
 "  </li>\n" +
 "  <li id=\"lastfm-options-login\">" +
 "   <h3>Authenticate</h3>\n" +
 "   <p>To log in this page <br/>will be closed.</p>" +
 "   <button class=\"yt-uix-button yt-uix-tooltip\" id=\"lastfm-login\" title=\"Log in to Last FM\"><span class=\"yt-uix-button-content\">Authenticate</span></button>" +
 "  </li>\n" +
 "  <li>\n" +
 "   <h3>Miscellaneous</h3>\n" +
 "   <button class=\"yt-uix-button yt-uix-tooltip\" id=\"lastfm-moreoptions\" title=\"More options menu\"><span class=\"yt-uix-button-content\">More options</span></button><br/>\n" +
 "   <button class=\"yt-uix-button yt-uix-tooltip\" id=\"lastfm-man-scrobble\" title=\"Scrobble now\"><span class=\"yt-uix-button-content\">Scrobble manual</span></button>\n" +
 "  </li>\n" +
 "  <li>\n" +
 "   <h3>Scrobbling</h3>\n" +
 "   <table>\n" +
 "    <tbody>\n" +
 "    <tr><td><p>Artist:</p></td><td><input type=\"text\" id=\"lastfm-artist\"></td><td rowspan=\"2\">" +
 "     <img src=\"" + swapImgCode + "\" id=\"lastfm-swap-songinfo\" width=\"20px\" height=\"41px\"></td></tr>\n" +
 "    <tr><td><p>Title:</p></td><td><input type=\"text\" id=\"lastfm-title\"></td></tr>\n" +
 "    </tbody>\n" +
 "   </table>\n" +
 "  </li>\n" +
 "  <li style=\"max-width: 228px; border-right: none;\">\n" +
 "   <h3>Status</h3>\n" +
 "   <div id=\"scrobble-counter\">\n" +
 "    <p><span id=\"scrobble-in\">Scrobble in:</span> <span id=\"scrobble-time\"></span></p>\n" +
 "    <button class=\"yt-uix-button yt-uix-tooltip\" id=\"lastfm-noscrobble\" title=\"Stops the automatic scrobble timer\"><span class=\"yt-uix-button-content\">Don't scrobble</span></button>\n" +
 "   </div>\n" +
 "   <div id=\"scrobble-message\" style=\"display: none;\">\n" +
 "    <p id=\"scrobble-msg\"></p>\n" + 
 "   </div>\n" +
 "  </li>\n" +
 "  <li style=\"float: right; border-right: none;\">" +
 "   <div id=\"scrobblesmurf-logo\">\n" +
 "    <img src=\"" + scrobbleSmurfImgCode + "\"/>\n" +
 "   </div>\n" +
 "  </li>\n" +
 "  </ul>\n" +
 "  <div style=\"clear: both;\"></div>\n" +
 " </div>\n" +
 "</div>\n";
 var container = document.getElementById("masthead-container");
 container.innerHTML += htmlCode;
}
/**
 * Toggles the Last FM form added by this plug in. 
 * The stated is based on the Last FM button at the right top.
 */
function toggleLastFmForm() {
 var container = document.getElementById("masthead-lastfm");
 var buttonCont = document.getElementById("lastfm-button").parentNode;
 if(countOccurencesInString(buttonCont.getAttribute("class"), "yt-uix-expander-collapsed") >= 1) {
  container.style.display = "block";
 } else { 
  container.style.display = "none";
 }
}
//--Last FM form button listeners and actions.
/**
 * Add event listeners to all the menubuttons added by this plug-in.
 */
function activateLastFmButtons() {
 authenticateButton();
 resetAuthenicateButton();
 dontScrobbleButton();
 moreOptionsButton();
 manScrobbleButton();
 swapSongInfoButton();
}
/**
 * Add the event listener to the 'authenticate'-button.
 */
function authenticateButton() {
 var authButton = document.getElementById("lastfm-login");
 authButton.addEventListener("click", authenticate, false);
}
/**
 * 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.
 */
function authenticate() {
 var tokenRegex = /(\?|&)token=[^&\?]*/gi;
 var currentURL = document.URL.replace(tokenRegex, "");
 if(!stringContainsChar(currentURL, "?")) {
  currentURL = currentURL.replace("&", "?");
 }
 var redirectURL = lastFmAuthenticationUrl + "?api_key=" + apiKey +"&cb=" + currentURL;
 window.location.href = redirectURL;
}
/**
 * Add the event listener to the 'reset login'-button.
 */
function resetAuthenicateButton() {
 var resetButton = document.getElementById("lastfm-reset");
 resetButton.addEventListener("click", resetAuth, false);
}
/**
 * Resets the authentication. A user would have to authenticate again to scrobble.
 */
function resetAuth() {
 localStorage.removeItem("lastfmUsername");
 localStorage.removeItem("lastfmSessionkey");
 showLoginHideReset();
}
/**
 * Adds the event listener to the 'do not scrobble'-button.
 */
function dontScrobbleButton() {
 var dontScrobbleButton = document.getElementById("lastfm-noscrobble");
 dontScrobbleButton.addEventListener("click", dontScrobble, false);
}
/**
 * Disables scrobbling for this time.
 */
function dontScrobble() {
 noScrobble = true;
 
 var dontScrobbleButton = document.getElementById("lastfm-noscrobble");
 dontScrobbleButton.className += " yt-uix-button-toggled";
}
/**
 * Adds the event listener to the 'options'-button.
 */ 
function moreOptionsButton()
{
 var moreOptionsButton = document.getElementById("lastfm-moreoptions");
 moreOptionsButton.addEventListener("click", function(){ moreOptions(false); }, false);
}
/**
 * Shows or hides a pop-up containing some less common options for this plug-in.
 */
function moreOptions(hide)
{
 var back = document.getElementById("lastfm-moreoptions-back");
 var pop = document.getElementById("lastfm-moreoptions-pop");
 
 if(hide) {
  back.style.visibility = "hidden";
  pop.style.visibility = "hidden";
 } else {
  if(scrobbleAutoCollapse === true || scrobbleAutoCollapse === "true") {
   setRadioButtonChecked("lastfm-moreoptions-collapse", "collapse");
  } else {
   setRadioButtonChecked("lastfm-moreoptions-collapse", "expand");
  }
  document.getElementById("lastfm-moreoptions-filter").value = scrobbleFilter;
  if(scrobbleEnabled === true || scrobbleEnabled === "true") {
   setRadioButtonChecked("lastfm-moreoptions-scrobble", "yes");
  } else {
   setRadioButtonChecked("lastfm-moreoptions-scrobble", "no");
  }
 
  pop.style.visibility = "visible";
  back.style.visibility = "visible";
 }
}
/**
 * Adds the event listener to the 'manual scrobble'-button.
 */ 
function manScrobbleButton()
{
 var manScrobbButton = document.getElementById("lastfm-man-scrobble");
 manScrobbButton.addEventListener("click", scrobbleManual, false);
}
/**
 * Disables scrobbling for this time.
 */
function scrobbleManual() {
 noScrobble = true;
 scrobbleSong();
 
 var manScrobbButton = document.getElementById("lastfm-man-scrobble");
 manScrobbButton.removeEventListener('click', scrobbleManual, false);
 manScrobbButton.className += " yt-uix-button-toggled";
}
/**
 * Adds the event listener to the two arrows, the 'swap song info'-button.
 */
function swapSongInfoButton()
{
 var swapSongButton = document.getElementById("lastfm-swap-songinfo");
 swapSongButton.addEventListener("click", swapSongInfo, false);
}
/**
 * Swaps the artist field's value with the title field's value.
 */
function swapSongInfo()
{
 var artistInput = document.getElementById("lastfm-artist");
 var titleInput = document.getElementById("lastfm-title");
 
 var oldArtist = artistInput.value;
 var oldTitle = titleInput.value;
 
 artistInput.value = oldTitle;
 titleInput.value = oldArtist;
}
//--Functions to for song information.
/**
 * Fills the 'song'-form with the passed song.
 */
function fillSongInformation(song) {
 var artistInput = document.getElementById("lastfm-artist");
 var titleInput = document.getElementById("lastfm-title");
 
 if(song.artist != null) {
  artistInput.value = song.artist;
 }
 if(song.title != null) {
  titleInput.value = song.title;
 }
}
/**
 * Gets song information from the currently played YouTube video using the iTunes metadata.
 */
function getSongInformation() {
 var song = new Object();
 song.artist = tryGetArtistMeta();
 song.title = tryGetTitleMeta();
 if(song.artist && song.title) {
  iTunesArtistAndTitleFound = true;
 }
 return song; 
}
/**
 * Gets song artist from the metadata in the video description.
 */
function tryGetArtistMeta() {
 var xPath = "//ul[@id='watch-description-extra-info']/li[@class='full-link']/a/span[@class='metadata-info']/span";
 try { 
  var artistNodes = document.evaluate(xPath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
  var artist = artistNodes.singleNodeValue.firstChild.nodeValue;
 } catch (e) {
 }
 return artist;
}
/**
 * Gets song title from the metadata in the video description.
 */
function tryGetTitleMeta() {
 var xPath, title;
 if(cosmicPanda) {
  xPath = "//ul[@id='watch-description-extra-info']/li[2]/span[@class='metadata-info']";
 } else {
  xPath = "//ul[@id='watch-description-extra-info']/li[3]/span[@class='metadata-info']";
 }
 try {
  var songNodes = document.evaluate(xPath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
  var rawTitle = songNodes.singleNodeValue.firstChild.nodeValue; //rawSong still contains 'buy on' text.
  rawTitle = removeBreaks(rawTitle);
  title = stringBetweenChars(rawTitle, "\"");
  if(typeof(title) === "undefined") {
   title = stringBetweenChars(rawTitle, "'");
  }
 } catch (e) {
 }
 return title;
}
/**
 * Tries to strip a song's information from the passed video title. 
 * First looks for quoted text which will be used as song title. 
 * The rest of the title will then be used as artist. 
 * If it fails it split the title between a dash and uses the first part as artist. 
 * The second part will be used as title. 
 */
function tryStripSongFromTitle(title) {
 var song = new Object();
 if(countOccurencesInString(title, "\"") >= 2) {
  song = getSongFromQuotedTitle(title, "\"");
 } else if(countOccurencesInString(title, "'") >= 2) {
  song = getSongFromQuotedTitle(title, "'");
 } else if(stringContainsChar(title, "-")) {
  var split = title.split("-");
  song.artist = trim(split[0]);
  song.title = trim(split[1]);
 }
 return song;
}
/**
 * Gets a song from a video title that contains quotes.
 */
function getSongFromQuotedTitle(vidTitle, quoteString) {
 var song = new Object();
 song.title = stringBetweenChars(vidTitle, quoteString);
 var regex = new RegExp(quoteString + song.title + quoteString);
 song.artist = vidTitle.replace(regex, "");
 return song;
}
//--JavaScript common utilization functions.
/**
 * Removes breaks from a text.
 */
function removeBreaks(text) {
 return text.replace(/(\n|\r)/gm, "");
}
/**
 * Checks whether a text contains a character.
 */
function stringContainsChar(text, charac) {
 for(var i = 0; i < text.length; i++) {
  if(text[i] == charac) {
   return true;
  }
 }
 return false;
}
/**
 * Counts the occurences of a certain object in a text.
 */
function countOccurencesInString(text, toMatch) {
 var regex = new RegExp(toMatch, "g");
 var matches = text.match(regex);
 if(matches == null) {
  return 0;
 }
 return matches.length;
}
/**
 * Finds a text between two characters.
 */
function stringBetweenChars(text, charac) {
 var part, results = [];
 var regex = new RegExp(charac + "([^" + charac +"]+)", "g");
 while(part = regex.exec(text)) {
  results.push(part[1]);
 }
 results = results.splice(0, 1);
 if(typeof(results) === "string") {
  return results;
 } else {
  return results[0];
 }
}
/**
 * Trims a text from spaces.
 */
function trim(text) {
 return text.replace(/^\s*/, "").replace(/\s*$/, "");
}
/**
 * Gets the selected value from a radio button group.
 */
function getSelectedRadioButton(name) {
 var radioButtons = document.getElementsByName(name);
 for(x in radioButtons) {
  if(radioButtons[x].checked) {
   return radioButtons[x].value;
  }
 }
 return null;
}
/**
 * Sets all values from a radio button group to unchecked except the passed value's radio button.
 */
function setRadioButtonChecked(name, value) {
 var radioButtons = document.getElementsByName(name);
 for(x in radioButtons) {
  if(radioButtons[x].value == value) {
   radioButtons[x].checked = true;
  } else {
   radioButtons[x].checked = false;
  }
 }
}
/**
 * Sets a select input to a default value.
 */
function setDefaultSelectValue(name, value) {
 var selectInput = document.getElementsByName(name)[0];
 var options = selectInput.getElementsByTagName("option");
 for(var i = 0; i < options.length; i++) {
  if(options[i].value == value) {
   options[i].setAttribute("selected", "");
  } else {
   options[i].removeAttribute("selected");
  }
 }
 
 var parent = selectInput.parentNode;
 parent.removeChild(selectInput);
 parent.appendChild(selectInput);
}

initialize();

0 comments:

Post a Comment