picMF

show MetaFilter profile pictures next to names in posts and comments

  1. // picMF.user.js
  2. //
  3. // Written by: Michael Devore
  4. // Released to the public domain
  5. //
  6. // This is a Greasemonkey script.
  7. // See http://www.greasespot.net/ for more information on Greasemonkey.
  8. //
  9. // ==UserScript==
  10. // @name picMF
  11. // @namespace http://www.devoresoftware.com/gm/picMF
  12. // @description show MetaFilter profile pictures next to names in posts and comments
  13. // @match https://*.metafilter.com/*
  14. // @match http://*.metafilter.com/*
  15. // @grant GM_xmlhttpRequest
  16. // @run-at document-end
  17. // @version 2.0
  18. // ==/UserScript==
  19. //
  20.  
  21. "use strict";
  22.  
  23. var theWidth = "64px";
  24. var theHeight = "64px";
  25. var zoomWidth = "256px";
  26. var zoomHeight = "256px";
  27. var secondsToLoad = 1.4;
  28. var serverPrefix = "//s3-us-west-2.amazonaws.com/mefi.profile/";
  29. var noPicLink = "";
  30.  
  31. function onLoaded()
  32. {
  33. // var xpath = "//DIV/SPAN[starts-with(text(),'posted by') and (@class='smallcopy' or @class='smallcopy postbyline' or @class='smallcopy byline')]";
  34. var xpath = "//DIV/SPAN[starts-with(text(),'posted by') and contains(@class, 'smallcopy')]";
  35. var postNodes = document.evaluate(
  36. xpath,
  37. document,
  38. null,
  39. XPathResult.ORDERED_NODE_SNAPSHOT_TYPE,
  40. null
  41. );
  42. var total = postNodes.snapshotLength;
  43. for (var i = 0; i < total; i++)
  44. {
  45. // not much validation here, cuts performance overhead by avoiding extra tests against the nodes
  46. // tighten it down later if it conflicts with other add-ons or Metafilter bling
  47. var userSpan = postNodes.snapshotItem(i);
  48. var currentNode = userSpan.firstChild;
  49. var found = false;
  50. var userLink;
  51. var linkNode;
  52. while (currentNode && !found)
  53. {
  54. if (currentNode.nodeName === "A")
  55. {
  56. var href_value = currentNode.getAttribute('href');
  57. var result = href_value.match(/\/user\/(\d+)/);
  58. if (result && result[1])
  59. {
  60. var userNumber = result[1];
  61. var serverLink = serverPrefix+userNumber+".jpg";
  62. var img = document.createElement("img");
  63. img.setAttribute("src", serverLink);
  64. setTimeout(function(x, y)
  65. {
  66. return function()
  67. {
  68. if (x.height <= 0)
  69. {
  70. x.setAttribute("src", noPicLink);
  71. }
  72.  
  73. x.setAttribute("height", theHeight);
  74. x.setAttribute("width", theWidth);
  75. // y.insertBefore(document.createTextNode(" "), y.firstChild);
  76. // y.insertBefore(x, y.firstChild);
  77. y.parentNode.insertBefore(document.createTextNode(" "), y.parentNode.firstChild);
  78. y.parentNode.insertBefore(x, y.parentNode.firstChild);
  79. x.addEventListener('mouseover', picHover, false);
  80. x.addEventListener('mouseout', picRestore, false);
  81. };
  82. }(img, currentNode), secondsToLoad * 1000);
  83. break;
  84. }
  85. }
  86. currentNode = currentNode.nextSibling;
  87. }
  88. }
  89. }
  90.  
  91. function picHover(evt)
  92. {
  93. var picNode = evt['target'];
  94. picNode.setAttribute("height", zoomHeight);
  95. picNode.setAttribute("width", zoomWidth);
  96. }
  97.  
  98. function picRestore(evt)
  99. {
  100. var picNode = evt['target'];
  101. picNode.setAttribute("height", theHeight);
  102. picNode.setAttribute("width", theWidth);
  103. }
  104.  
  105. document.addEventListener('DOMContentLoaded',onLoaded,true);