Sunday, February 26, 2012

Youtube Tweak


// ==UserScript==
// @id             YoutubeTweak@Lunatrius
// @name           Youtube Tweak
// @version        0.3.1
// @namespace      Lunatrius
// @author         Lunatrius <lunatrius@gmail.com>
// @description    Add a delete button, automatically delete watched videos.
// @match          http://www.youtube.com/
// @match          http://www.youtube.com/?*
// @match          http://www.youtube.com/watch*
// @match          https://www.youtube.com/
// @match          https://www.youtube.com/?*
// @match          https://www.youtube.com/watch*
// @include        http://www.youtube.com/
// @include        http://www.youtube.com/?*
// @include        http://www.youtube.com/watch*
// @include        https://www.youtube.com/
// @include        https://www.youtube.com/?*
// @include        https://www.youtube.com/watch*
// @icon           
// @run-at         document-end
// ==/UserScript==

/*jslint devel: true, browser: true, regexp: true, continue: true, plusplus: true, maxerr: 50, indent: 4 */
/*global XPathResult: false */

(function () {
 "use strict";

 // add an item to the array if it doesn't exist
 Array.prototype.add = function (element) {
  var i;
  for (i = 0; i < this.length; i++) {
   if (this[i] === element) {
    return false;
   }
  }
  this.push(element);
  return true;
 };

 // remove all matching items
 Array.prototype.remove = function (element) {
  var removed = [], i;
  for (i = 0; i < this.length; i++) {
   if (this[i] === element) {
    removed.push(this.splice(i--, 1)[0]);
   }
  }
  return removed;
 };

 // return true if the element was found
 Array.prototype.exists = function (element) {
  var i;
  for (i = 0; i < this.length; i++) {
   if (this[i] === element) {
    return true;
   }
  }
  return false;
 };

 // some variables
 var msg, debug, DomStorage, YoutubeTweak, yt, cfg, saveAmount;

 // inicialization
 yt = null;
 debug = true;
 saveAmount = 500;

 // correct a few things
 if (saveAmount > 0) {
  saveAmount = -saveAmount;
 }

 // debug function
 msg = debug ? function (message) {
  console.log(JSON.stringify(message));
 } : function () {};

 /***************************************************************
  * Name: DOM Storage Wrapper Class
  * Author: Lunatrius <lunatrius@gmail.com>
  *
  * Public members:
  *     ctor({"session"|"local"}[, <namespace>])
  *     set(<key>, <value>)
  *     get(<key>, <default value>)
  *     remove(<key>)
  *     keys()
  *     removeAll()
  ***************************************************************/
 DomStorage = function (type, namespace) {
  // variables
  var t = this,
   s,
   value,
   keys,
   key,
   i;

  // check if type if specified
  if (typeof (type) !== "string") {
   type = "session";
  }

  // check if namespace if specified
  if (!namespace || (typeof (namespace) !== "string" && typeof (namespace) !== "number")) {
   namespace = "script";
  }

  // bind the storage of a specific type to a variable
  switch (type) {
  case "local":
   s = localStorage;
   break;

  default: // case "session":
   s = sessionStorage;
   break;
  }

  // concat the namespace string
  namespace = ["domstorage", namespace].join(".").replace(/[^a-z0-9\.\-]/gi, "_");

  // set a key with a value
  t.set = function (key, value) {
   s.setItem([namespace, key].join(".").replace(/[^a-z0-9\.\-]/gi, "_"), JSON.stringify(value));
  };

  // get the value of a key, if there is none return the default value
  t.get = function (key, defaultValue) {
   value = JSON.parse(s.getItem([namespace, key].join(".").replace(/[^a-z0-9\.\-]/gi, "_")));
   if (value !== null && value !== undefined) {
    return value;
   }
   return defaultValue;
  };

  // remove a key
  t.remove = function (key) {
   s.removeItem([namespace, key].join(".").replace(/[^a-z0-9\.\-]/gi, "_"));
  };

  // return an array of keys
  t.keys = function () {
   keys = [];
   for (i = s.length - 1; i >= 0; i--) {
    key = s.key(i).replace([namespace, ""].join("."), "");
    if (key !== s.key(i)) {
     keys.push(key);
    }
   }
   keys.sort();
   return keys;
  };

  // remove all keys
  t.removeAll = function () {
   keys = t.keys();
   for (i = keys.length - 1; i >= 0; i--) {
    t.remove(keys[i]);
   }
  };
 };

 // the core of the script
 YoutubeTweak = function () {
  // delarations
  var t = this;


  // initialization
  t.style = null;

  // hotkey handler
  t.hotkey = function (e) {
   var value = null;
   if (e.altKey === true) {
    switch (e.keyCode) {
    case 49:
     value = cfg.get("removewatched", true);
     if (confirm("Do you want to " + (value ? "disable" : "enable") + " the removal of watched videos?")) {
      value = !value;
      cfg.set("removewatched", value);
      alert((value ? "Enabled" : "Disabled") + " the removal of watched videos.");
     }
     break;

    case 50:
     value = cfg.get("removeignored", true);
     if (confirm("Do you want to " + (value ? "disable" : "enable") + " the removal of ignored videos?")) {
      value = !value;
      cfg.set("removeignored", value);
      alert((value ? "Enabled" : "Disabled") + " the removal of ignored videos.");
     }
     break;
    }
   }
  };

  // initialization entry point
  t.init = function () {
   cfg = new DomStorage("local", "YoutubeTweak");
   try {
    window.addEventListener("keydown", t.hotkey, false);
    window.wrappedJSObject.ytt = cfg;
    document.head = document.head || document.getElementsByTagName("head")[0] || document.documentElement;
   } catch (ex) {
   }

   t.main();
  };

  // main function
  t.main = function () {
   var videoId;

   // video page
   if (location.pathname === "/watch") {
    videoId = t.getVideoId(location.search);
    if (videoId) {
     t.addWatched(videoId);
    }
   }

   // main page
   if (location.pathname === "/") {
    // add the button design
    t.addStyle(".yt-tweak-button", {
     "position": "absolute",
     "top": "16px",
     "right": "2px",
     "width": "12px",
     "height": "12px",
     "cursor": "pointer",
     "background": "url('')"
    });

    // add the button design - hover
    t.addStyle(".yt-tweak-button:hover", {
     "background": "url('')"
    });

    // execute the functions and create an interval
    t.homeFunctions();
    setInterval(t.homeFunctions, 3000);
   }
  };

  // add a video to the watched list
  t.addWatched = function (videoId) {
   try {
    var watched = cfg.get("watched", []);
    watched.add(videoId);
    cfg.set("watched", watched.splice(saveAmount));
   } catch (ex) {
    msg(ex);
   }
  };

  // add a video to the ignored list
  t.addIgnored = function (videoId) {
   try {
    var ignored = cfg.get("ignored", []);
    ignored.add(videoId);
    cfg.set("ignored", ignored.splice(saveAmount));
   } catch (ex) {
    msg(ex);
   }
  };

  // check if a videos was watched
  t.existsWatched = function (videoId) {
   try {
    var watched = cfg.get("watched", []);
    return watched.exists(videoId);
   } catch (ex) {
    msg(ex);
    return false;
   }
  };

  // check if a videos was ignored
  t.existsIgnored = function (videoId) {
   try {
    var ignored = cfg.get("ignored", []);
    return ignored.exists(videoId);
   } catch (ex) {
    msg(ex);
    return false;
   }
  };

  // clean any whitespaces
  t.strip = function (string) {
   string = string.replace(/\s+/g, " ");
   string = string.replace(/^\s+/g, "");
   string = string.replace(/\s+$/g, "");
   return string;
  };

  // get the video ID (v argument)
  t.getVideoId = function (url) {
   var videoId = url.match(/v=([^&]+)/);
   if (videoId && videoId.length === 2) {
    return videoId[1];
   } else {
    return null;
   }
  };

  // append CSS style
  t.addStyle = function (cssRule, cssProperties) {
   var cssProp = [], i;
   if (t.style === null) {
    t.style = document.createElement("style");
    t.style.id = "YoutubeTweak";
    t.style.textContent = "";
    document.head.appendChild(t.style);
   }

   for (i in cssProperties) {
    if (cssProperties.hasOwnProperty(i)) {
     cssProp.push(i + ": " + cssProperties[i] + " !important;");
    }
   }

   t.style.textContent += cssRule + " {\n\t" + cssProp.join("\n\t") + "\n}" + "\n\n";
  };

  // used for the inverval timer
  t.homeFunctions = function () {
   t.addIgnoreButton();
   t.expandLists();

   t.deleteVideos();
   // t.deletePlaylists();
   // t.deleteListDescriptions();
   // t.deleteListItems();
   // t.deleteLists();
   // t.deleteMainListItems();
  };

  // generate a function for one button
  t.addIgnoredHandler = function (videoId) {
   return function () {
    t.addIgnored(videoId);

    if (this.className === "yt-tweak-button") {
     this.parentNode.parentNode.parentNode.parentNode.removeChild(this.parentNode.parentNode.parentNode);
    }
   };
  };

  // add a button
  t.addIgnoreButton = function () {
   var videoSnap, i, videoId, button;

   videoSnap = document.evaluate("//div[contains(@class,'feed-page')]/ul/li/div[contains(@class,'feed-item-container')]/div[contains(@class,'feed-item-outer')]/div[contains(@class,'feed-item-main')]/div/div/h4/a[contains(@class,'title') and not(@button)]", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);

   for (i = 0; i < videoSnap.snapshotLength; i++) {
    button = document.createElement("div");
    button.classList.add("yt-tweak-button");

    videoSnap.snapshotItem(i).parentNode.parentNode.parentNode.parentNode.appendChild(button);
    videoSnap.snapshotItem(i).setAttribute("button", "yes");

    videoId = t.getVideoId(videoSnap.snapshotItem(i).href);

    if (videoId !== null && typeof (videoId) === "string") {
     button.addEventListener("click", t.addIgnoredHandler(videoId), false);
    }
   }
  };

  // try to delete all video on the page
  t.deleteVideos = function () {
   var videoSnap, i, videoId;

   videoSnap = document.evaluate("//div[contains(@class,'feed-page')]/ul/li/div[contains(@class,'feed-item-container')]/div[contains(@class,'feed-item-outer')]/div[contains(@class,'feed-item-main')]/div/div/h4/a[contains(@class,'title')]", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);

   for (i = 0; i < videoSnap.snapshotLength; i++) {
    videoId = t.getVideoId(videoSnap.snapshotItem(i).href);

    if (videoId !== null && typeof (videoId) === "string") {
     t.deleteVideo((t.existsWatched(videoId) && cfg.get("removewatched", true) === true) || (t.existsIgnored(videoId) && cfg.get("removeignored", true)), videoId);
    }
   }
  };

  // delete one video on the page
  t.deleteVideo = function (deleted, videoId) {
   if (deleted === true) {
    var videoSnap, node, container, i;

    videoSnap = document.evaluate("//div[contains(@class,'feed-page')]/ul/li/div[contains(@class,'feed-item-container')]/div[contains(@class,'feed-item-outer')]/div[contains(@class,'feed-item-main')]/div/div/h4/a[contains(@class,'title') and contains(@href,'v=" + videoId + "')]/../../../..", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);

    for (i = 0; i < videoSnap.snapshotLength; i++) {
     node = videoSnap.snapshotItem(i).parentNode.parentNode.parentNode;
     container = node.parentNode.parentNode;

     node.parentNode.removeChild(node);
     if (container.firstElementChild.childElementCount === 0) {
      // console.log("deleting empty container...");
      container.parentNode.removeChild(container);
     }
     // videoSnap.snapshotItem(i).parentNode.parentNode.parentNode.parentNode.removeChild(videoSnap.snapshotItem(i).parentNode.parentNode.parentNode);
    }
   }
  };
/*
  // delete playlists - these are useless as everything is ALREADY listed
  t.deletePlaylists = function () {
   var snapPlaylists, item, i;
   snapPlaylists = document.evaluate("//div[contains(@class,'feed-item-content-playlist')]", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
   for (i = 0; i < snapPlaylists.snapshotLength; i++) {
    item = snapPlaylists.snapshotItem(i);
    item.parentNode.removeChild(item);
   }
   snapPlaylists = document.evaluate("//div[contains(@class,'feed-item-main')]", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
   for (i = 0; i < snapPlaylists.snapshotLength; i++) {
    item = snapPlaylists.snapshotItem(i);
    if (item.childElementCount === 2 || item.childElementCount === 1) {
     item.parentNode.removeChild(item);
    }
   }
  };

  // delete list descriptions
  t.deleteListDescriptions = function () {
   var snapListDesc, item, i;
   snapListDesc = document.evaluate("//div[contains(@class,'feed-item-description')]/div", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
   for (i = 0; i < snapListDesc.snapshotLength; i++) {
    item = snapListDesc.snapshotItem(i);
    if (item.children.length === 0) {
     item.parentNode.parentNode.removeChild(item.parentNode);
    }
   }
  };

  // delete all remaining trash elements
  t.deleteListItems = function () {
   var snapListItems, item, i;
   snapListItems = document.evaluate("//div[contains(@class,'feed-page')]/ul/li/div/div", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
   for (i = 0; i < snapListItems.snapshotLength; i++) {
    item = snapListItems.snapshotItem(i);
    if (item.children.length === 0) {
     item.parentNode.parentNode.parentNode.removeChild(item.parentNode.parentNode);
    }
   }
  };

  // delete remaining list elements
  t.deleteLists = function () {
   var snapListItems, item, i;
   snapListItems = document.evaluate("//div[contains(@class,'feed-page')]/ul", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
   for (i = 0; i < snapListItems.snapshotLength; i++) {
    item = snapListItems.snapshotItem(i);
    if (item.children.length === 0) {
     item.parentNode.parentNode.removeChild(item.parentNode);
    }
   }
  };

  // delete remaining main list elements
  t.deleteMainListItems = function () {
   var snapMainListItems, item, i;
   snapMainListItems = document.evaluate("//li[contains(@class,'feed-item-container')]", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
   for (i = 0; i < snapMainListItems.snapshotLength; i++) {
    item = snapMainListItems.snapshotItem(i);
    if (!/watch([^"]+)v=([^&]+)&/i.test(item.innerHTML)) {
     item.parentNode.removeChild(item);
    }
   }
  };
*/
  // auto expand the lists ONCE
  t.expandLists = function () {
   var spanSnap, item, i;

   spanSnap = document.evaluate("//a[contains(@class,'show-more') and not(@done)]", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);

   for (i = 0; i < spanSnap.snapshotLength; i++) {
    item = spanSnap.snapshotItem(i);
    try {
     item.click();
    } catch (exA) {}
    try {
     item.wrappedJSObject.click();
    } catch (exB) {}
    item.setAttribute("done", "yes");
   }
  };

  // execute the script
  t.init();
 };

 // filter out and errors
 try {
  // only run on the specified domain
  if (/www\.youtube\.com/i.test(location.host) && window.top === window.self) {
   // only run on the main and video pages
   if (location.pathname === "/" || location.pathname === "/watch") {
    // create a new instance of the core
    yt = new YoutubeTweak();
   }
  }
 } catch (ex) {
  msg(ex);
 }
}());

0 comments:

Post a Comment