Marti / uso - Anchor Bookmarks

(function () {

// ==UserScript==
// @name          uso - Anchor Bookmarks
// @namespace     http://userscripts.org/users/37004
// @description   Converts anchor tags with no attributes into bookmarks via the next available text node line
// @copyright     2010+, Marti Martz (http://userscripts.org/users/37004)
// @contributor   sizzlemctwizzle (http://userscripts.org/users/27715)
// @license       CC-BY-NC-SA-4.0; https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode
// @license       GPL-3.0+; http://www.gnu.org/licenses/gpl-3.0.txt
// @version       0.5.10.1eol
// @icon          https://www.gravatar.com/avatar/e615596ec6d7191ab628a1f0cec0006d?r=PG&s=48&default=identicon

// @homepageURL   https://github.com/Martii/UserScripts/tree/master/src/uso/Anchor%20Bookmarks
// @homepageURL   https://openuserjs.org/scripts/Marti/uso_-_Anchor_Bookmarks
// @homepageURL   http://userscripts.org/scripts/show/69725
// @supportURL    http://userscripts.org/topics/46795

// @include /^https?://userscripts\.org(?::\d{1,5})?//

// @exclude /^https?://userscripts\.org(?::\d{1,5})?/login/
// @exclude /^https?://userscripts\.org(?::\d{1,5})?/scripts/diff//
// @exclude /^https?://userscripts\.org(?::\d{1,5})?/scripts/version//

// @include http://userscripts.org:8080/*

// @include http://userscripts.org/*

// @include https://userscripts.org/*

// @exclude http://userscripts.org:8080/login*
// @exclude http://userscripts.org:8080/scripts/diff/*
// @exclude http://userscripts.org:8080/scripts/version/*

// @exclude http://userscripts.org/login*
// @exclude http://userscripts.org/scripts/diff/*
// @exclude http://userscripts.org/scripts/version/*

// @exclude https://userscripts.org/login*
// @exclude https://userscripts.org/scripts/diff/*
// @exclude https://userscripts.org/scripts/version/*

// ==/UserScript==

  var img = "";

  var headNode = document.evaluate(
      "//head",
      document.documentElement,
      null,
      XPathResult.FIRST_ORDERED_NODE_TYPE,
      null
  );
  if (headNode && headNode.singleNodeValue) {
    var styleNode = document.createElement("style");
    styleNode.setAttribute("type", "text/css");
    styleNode.setAttribute("media", "screen, projection");
    styleNode.textContent = ".bookmark { width: 16px; height: 16px; margin: 0.1em 0.2em 0; float: left; background: transparent url(" + img + ") no-repeat top left; opacity: 0.4; } .bookmark:hover { opacity: 1.0; }";
    headNode.singleNodeValue.appendChild(styleNode);
  }

  var xpr = document.evaluate(
      "//a[starts-with(@name,'comment-')]",
      document.body,
      null,
      XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
      null
  );

  if (xpr)
    for (var i = 0, thisNode; thisNode = xpr.snapshotItem(i++);) {
      var imgNode = document.createElement("img");
      imgNode.setAttribute("class", "bookmark");
      imgNode.setAttribute("src", "");
      imgNode.setAttribute("title", "link");
      thisNode.appendChild(imgNode);

      if (window.location.pathname.match(/^\/comments(.*)/i)) {
        var userid, useridNode = document.evaluate(
            ".//a[@user_id]",
            thisNode.parentNode,
            null,
            XPathResult.FIRST_ORDERED_NODE_TYPE,
            null
        );
        if (useridNode && useridNode.singleNodeValue) {
          var thatNode = useridNode.singleNodeValue;

          userid = thatNode.getAttribute("user_id");
          if (userid)
            thisNode.setAttribute("href", "/users/" + userid + "/comments#" + thisNode.name); // NOTE: Comments are not fully linked to source page so use public profile comments
        }
        if (!userid)
          thisNode.setAttribute("href", window.location.pathname + window.location.search + "#" + thisNode.name); // NOTE: Fallback to actual fragment in case of USO gen err
      }
      else
        thisNode.setAttribute("href", window.location.pathname + "#" + thisNode.name); // NOTE: Fallback for no pagination
    }

  // Fix missing recent /posts linkage if still missing
  if (window.location.pathname.match(/(.*)\/posts\/?$/i)) {
    var xpr = document.evaluate(
        "//tr[starts-with(@id,'posts-')]",
        document.body,
        null,
        XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
        null
    );
    if (xpr)
      for (var i = 0, thisNode; thisNode = xpr.snapshotItem(i++);) {
        var targetNode = document.evaluate(
            ".//abbr[@class='updated']",
            thisNode,
            null,
            XPathResult.FIRST_ORDERED_NODE_TYPE,
            null
        );
        if (targetNode && targetNode.singleNodeValue) {
          var abbrNode = targetNode.singleNodeValue;

          if (abbrNode.parentNode.getAttribute("rel") != "bookmark") {
            var aNode = document.createElement("a");
            aNode.setAttribute("rel", "bookmark");
            aNode.setAttribute("href", "/" + thisNode.getAttribute("id").replace("-", "/"));

            abbrNode.parentNode.appendChild(aNode);
            aNode.appendChild(abbrNode);
          }
        }
      }
  }

  // Add custom bookmarks if present
  var pathname, portion;
  switch ((pathname = window.location.pathname)) {
    case undefined:
    default:
      break;

    case (portion = pathname.match(/^\/scripts(.*)/i)) ? portion[0] : undefined:
      var tabid = (portion = portion[1].match(/^\/(show|reviews|issues)\/(.*)/i)) ? portion[1] : undefined;
      switch (tabid) {
        case "show":
          var contextNode = document.evaluate(
              "//div[@id='full_description']",
              document.body,
              null,
              XPathResult.FIRST_ORDERED_NODE_TYPE,
              null
          );
          if (contextNode && contextNode.singleNodeValue) {
            contextNode = contextNode.singleNodeValue;
            addBookmarks(contextNode, "bookmark-");
          }
          break;
        case "reviews":
          var contextNode = document.evaluate(
              "//div[@class='review']/div[@class='body']",
              document.body,
              null,
              XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
              null
          );
          for (var i = 0; i < contextNode.snapshotLength; ++i) {
            var thisNode = contextNode.snapshotItem(i);

            var reviewid = thisNode.previousSibling.previousSibling.previousSibling.previousSibling.getAttribute("id").match(/reviews-(\d+)-status/i)[1];
            addBookmarks(thisNode, "bookmark-" + reviewid + "-");
          }
          break;
        case "issues":
          var contextNode = document.evaluate(
              "//p[contains(@id, 'issuecomments-')]",
              document.body,
              null,
              XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
              null
          );
          for (var i = 0; i < contextNode.snapshotLength; ++i) {
            var thisNode = contextNode.snapshotItem(i);

            var commentid = thisNode.getAttribute("id").match(/issuecomments-(\d+)/i)[1];
            addBookmarks(thisNode.nextSibling, "bookmark-" + commentid + "-");
          }
          break;
      }
      break;

    case (portion = pathname.match(/^\/jetpacks(.*)/i)) ? portion[0] : undefined:
      var jetpackid = (portion = portion[1].match(/^\/(.*)/i)) ? portion[1] : undefined;
      if (jetpackid) {
        var contextNode = document.evaluate(
            "//p/b/text()['Summary:']",
            document.body,
            null,
            XPathResult.FIRST_ORDERED_NODE_TYPE,
            null
        );
        if (contextNode && contextNode.singleNodeValue) {
          contextNode = contextNode.singleNodeValue.parentNode.parentNode;
          addBookmarks(contextNode, "bookmark-");
        }
      }
      break;

    case (portion = pathname.match(/^\/articles(.*)/i)) ? portion[0] : undefined:
      var articleid = (portion = portion[1].match(/^\/(\d+).*/i)) ? portion[1] : undefined;
      if (articleid) {
        var contextNode = document.evaluate(
            "//div[contains(@id, 'comment-body')]",
            document.body,
            null,
            XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
            null
        );
        for (var i = 0; i < contextNode.snapshotLength; ++i) {
          var thisNode = contextNode.snapshotItem(i);

          var commentid = thisNode.getAttribute("id").match(/comment-body-(\d+)/i)[1];
          addBookmarks(thisNode, "bookmark-" + commentid + "-");
        }

        contextNode = document.evaluate(
            "//p[@class='summary']",
            document.body,
            null,
            XPathResult.FIRST_ORDERED_NODE_TYPE,
            null
        );
        if (contextNode && contextNode.singleNodeValue) {
          contextNode = contextNode.singleNodeValue;
          addBookmarks(contextNode, "bookmark-");
        }
      }
      break;

    case (portion = pathname.match(/(.*)\/comments\/?$/i)) ? portion[0] : undefined:
      var contextNode = document.evaluate(
          "//div[contains(@id, 'comment-body')]",
          document.body,
          null,
          XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
          null
      );
      for (var i = 0; i < contextNode.snapshotLength; ++i) {
        var thisNode = contextNode.snapshotItem(i);

        var commentid = thisNode.getAttribute("id").match(/comment-body-(\d+)/i)[1];
        addBookmarks(thisNode, "bookmark-" + commentid + "-");
      }

      contextNode = document.evaluate(
          "//p[@class='summary']",
          document,
          null,
          XPathResult.FIRST_ORDERED_NODE_TYPE,
          null
      );
      if (contextNode && contextNode.singleNodeValue) {
        contextNode = contextNode.singleNodeValue;
        addBookmarks(contextNode, "bookmark-");
      }
      break;

    case (portion = pathname.match(/^\/groups(.*)/i)) ? portion[0] : undefined:
      var groupid = (portion = portion[1].match(/^\/(\d+)\/?$/i)) ? portion[1] : undefined;
      if (groupid) {
        var contextNode = document.evaluate(
            "//div[@class='description']",
            document.body,
            null,
            XPathResult.FIRST_ORDERED_NODE_TYPE,
            null
        );
        if (contextNode && contextNode.singleNodeValue) {
          contextNode = contextNode.singleNodeValue;
          addBookmarks(contextNode, "bookmark-");
        }
      }
      break;

    case (portion = pathname.match(/^\/guides(.*)/i)) ? portion[0] : undefined:
      var guideid = (portion = portion[1].match(/^\/(\d+)\/?$/i)) ? portion[1] : undefined;
      if (guideid) {
        var contextNode = document.evaluate(
            "//div[@id='content']",
            document.body,
            null,
            XPathResult.FIRST_ORDERED_NODE_TYPE,
            null
        );
        if (contextNode && contextNode.singleNodeValue) {
          contextNode = contextNode.singleNodeValue;
          addBookmarks(contextNode, "bookmark-");
        }
      }
      break;

    case (portion = pathname.match(/^\/topics(.*)/i)) ? portion[0] : undefined:
      var topicid = (portion = portion[1].match(/^\/(\d+)\/?$/i)) ? portion[1] : undefined;
      if (topicid) {
        var contextNode = document.evaluate(
            "//td[contains(@id, 'post-body')]",
            document.body,
            null,
            XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
            null
        );
        for (var i = 0; i < contextNode.snapshotLength; ++i) {
          var thisNode = contextNode.snapshotItem(i);

          var postid = thisNode.getAttribute("id").match(/post-body-(\d+)/i)[1];
          addBookmarks(thisNode, "bookmark-" + postid + "-");
        }
      }
      break;

    case (portion = pathname.match(/^\/reviews(.*)/i)) ? portion[0] : undefined:
      var reviewid = (portion = portion[1].match(/^\/(\d+)\/?$/i)) ? portion[1] : undefined;
      if (reviewid) {
        var contextNode = document.evaluate(
            "//div[@class='body']",
            document.body,
            null,
            XPathResult.FIRST_ORDERED_NODE_TYPE,
            null
        );
        if (contextNode && contextNode.singleNodeValue) {
          contextNode = contextNode.singleNodeValue;
          addBookmarks(contextNode, "bookmark-");
        }

        contextNode = document.evaluate(
            "//div[contains(@id, 'comment-body')]",
            document.body,
            null,
            XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
            null
        );
        for (var i = 0; i < contextNode.snapshotLength; ++i) {
          var thisNode = contextNode.snapshotItem(i);

          var commentid = thisNode.getAttribute("id").match(/comment-body-(\d+)/i)[1];
          addBookmarks(thisNode, "bookmark-" + commentid + "-");
        }
      }
      break;
  }


function addBookmarks(contextNode, prefixAttribute) {
  var xpr = document.evaluate(
      "../descendant::a[not(@href)][not(@name)][not(@id)]",
      contextNode,
      null,
      XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
      null
  );

  var thisNode, thatNode;
  if (xpr) {
    var bookmarks = {};

    function checkBookmark(bookmarks, newbookmark, suffix) {
      for (var bookmark in bookmarks) {
        if (newbookmark + ((suffix) ? "-" + suffix : "") == bookmarks[bookmark]) {
          suffix = (suffix) ? suffix + 1 : 1;
          checkBookmark(bookmarks, newbookmark, suffix);
        }
      }
      return suffix;
    }

    for (var i = 0; i < xpr.snapshotLength; ++i) {
      var thatNode = thisNode = xpr.snapshotItem(i);

      while (thatNode) {
        if (thatNode.textContent != "") {
          var newbookmark = thatNode.textContent;

          if (!newbookmark.match(/^about:.*/i)) {
            newbookmark = newbookmark.replace(/^\s*/, "");
            if (newbookmark.match(/(.{1,64})/i)) {
              newbookmark = newbookmark.match(/(.{1,64})/i)[1];
              newbookmark = newbookmark.replace(/\s*$/, "");
              newbookmark = newbookmark.replace(/\s{2,}/g, " ");

              newbookmark = newbookmark.replace(/\.*/g, "");

              newbookmark = encodeURIComponent(newbookmark.toLowerCase());
              newbookmark = newbookmark.replace(/\%20/g, "-");
              newbookmark = newbookmark.replace(/\%/g, ".");
              newbookmark = prefixAttribute + newbookmark;

              var suffix;
              if ((suffix = checkBookmark(bookmarks, newbookmark)))
                newbookmark += "-" + suffix;

              bookmarks[newbookmark] = newbookmark;
            }
            thisNode.setAttribute("name", newbookmark);
            thisNode.setAttribute("id", newbookmark);

            var imgNode = document.createElement("img");
            imgNode.setAttribute("class", "bookmark");
            imgNode.setAttribute("title", "link");
            imgNode.setAttribute("alt", "link");
            imgNode.setAttribute("src", "");

            var anchorNode = document.createElement("a");
            anchorNode.setAttribute("href", "#" + newbookmark);

            anchorNode.appendChild(imgNode);
            thisNode.parentNode.insertBefore(anchorNode, thisNode);
          }
          break;
        }
        thatNode = thatNode.nextSibling;
      }
    }
  }
}
  var hash = window.location.hash.match(/^#(bookmark-.*)/);
  if (hash) {
    var anchorNode = document.evaluate(
        "//a[@id='" + hash[1] + "']",
        document.body,
        null,
        XPathResult.FIRST_ORDERED_NODE_TYPE,
        null
    );
    if (anchorNode && anchorNode.singleNodeValue)
      setTimeout(function () { anchorNode.singleNodeValue.scrollIntoView(); }, 1000);
  }

})();