Feedly slimFeed

Adds favicons to feeds and quick mark as read buttons.

  1. // ==UserScript==
  2. // @id feedlyslimfeed@labm0nkey
  3. // @name Feedly slimFeed
  4. // @version 1.3
  5. // @namespace
  6. // @author labm0nkey
  7. // @description Adds favicons to feeds and quick mark as read buttons.
  8. // @include http://feedly.com/*
  9. // @include https://feedly.com/*
  10. // @include http://www.feedly.com/*
  11. // @include https://www.feedly.com/*
  12. // @include http://cloud.feedly.com/*
  13. // @include https://cloud.feedly.com/*
  14. // @include http://www.cloud.feedly.com/*
  15. // @include https://www.cloud.feedly.com/*
  16. // @match http://feedly.com/*
  17. // @match https://feedly.com/*
  18. // @match http://www.feedly.com/*
  19. // @match https://www.feedly.com/*
  20. // @match http://cloud.feedly.com/*
  21. // @match https://cloud.feedly.com/*
  22. // @match http://www.cloud.feedly.com/*
  23. // @match https://www.cloud.feedly.com/*
  24. // @require http://code.jquery.com/jquery-latest.min.js
  25. // @run-at document-end
  26. // ==/UserScript==
  27.  
  28. function deserialize(name, def) {
  29. return eval(GM_getValue(name, (def || '({})')));
  30. }
  31.  
  32. function serialize(name, val) {
  33. GM_setValue(name, uneval(val));
  34. }
  35.  
  36. var arrow_down = '<img class="arrow_down" title="Mark below as read" src="">';
  37.  
  38. var arrow_up = '<img class="arrow_up" title="Mark above as read" src="">';
  39.  
  40. var cache_refresh = '<img id="cacheRefresh" width="24" height="24" border="0" src="" class="pageAction" title="Refresh cache">';
  41.  
  42. var h1read = '<h1 class="readWait">Marking as read...</h1>';
  43. var entryClass = '';
  44.  
  45. (function ($) {
  46. var favicons = {};
  47.  
  48. var marking_as_read = false;
  49.  
  50. //check settings
  51. if (GM_getValue("ffa_cache") == undefined) {
  52. favicons = generateFavicons();
  53. } else {
  54. favicons = deserialize("ffa_cache");
  55. }
  56.  
  57. var cssTxt;
  58. cssTxt = " .arrow_up, .arrow_down {cursor: pointer;} .readWait { position: absolute; top: 50%; left: 50%; background: none repeat scroll 0 0 rgba(255, 255, 255, 0.2); border-radius: 3px; color: #82BD1A;} .u4Entry { padding-left: 24px; }";
  59. cssTxt += ".u0Entry:nth-child(even) { background-color:#f6fceb; } .u0Entry:nth-child(odd) { background-color:#fdfef9; }";
  60. cssTxt += ".u4Entry:nth-child(even) { background-color:#f6fceb; } .u4Entry:nth-child(odd) { background-color:#fdfef9; }";
  61. cssTxt += ".u5Entry:nth-child(even) { background-color:#f6fceb; } .u5Entry:nth-child(odd) { background-color:#fdfef9; }";
  62. cssTxt += ".u100Entry:nth-child(even) { background-color:#f6fceb; } .u100Entry:nth-child(odd) { background-color:#fdfef9; }";
  63. cssTxt += ".slim-favicon { opacity: 0.9; height: 16px, width: 16px; float: left;}";
  64. cssTxt += ".metadata { overflow: visible !important; }";
  65. cssTxt += ".u0Entry .slim-favicon {margin: 8px 7px 0px 0px;}";
  66. cssTxt += ".u4Entry .slim-favicon {margin: 1px 7px 0px 0px;} .u4Entry .metadata { max-height: 20px !important; } .u4Entry .slim-favicon { opacity: 0.9; width: 16px; height: 16px; display: inline; } ";
  67. cssTxt += ".u5Entry .slim-favicon {margin: 1px 7px 0px 0px;} .u5Entry .metadata { max-height: 20px !important; } .u5Entry .slim-favicon { opacity: 0.9; width: 16px; height: 16px; display: inline; } ";
  68. cssTxt += ".u100Entry .slim-favicon {margin: 1px 7px 0px 0px;} .u100Entry .metadata { max-height: 20px !important; } .u100Entry .slim-favicon { opacity: 0.9; width: 16px; height: 16px; display: inline; } ";
  69.  
  70. GM_addStyle(cssTxt);
  71.  
  72. function viewToClass() {
  73. return "." + checkView() + "Entry";
  74. }
  75.  
  76. /*
  77. $('#feedlyPart').on('DOMAttrModified', function (event) {
  78. var element = event.target;
  79.  
  80. if (element.attributes["id"].nodeValue == 'loadingEntries') {
  81.  
  82. var entry = viewToClass();
  83. $(entry).each(function () {
  84. $(this).slimStyle();
  85. });
  86.  
  87. $('#section0_column0').on('DOMNodeInserted', function (event) {
  88. var element = event.target;
  89. var entry = viewToClass();
  90. if (element.attributes["class"].nodeValue == (entry + ' ')) {
  91. addFavicon($(element));
  92. addArrows($(this));
  93. }
  94. });
  95. }
  96. });
  97. */
  98. $.fn.slimStyle = function () {
  99. return this.each(function () {
  100. addFavicon($(this));
  101. addArrows($(this));
  102. });
  103. };
  104.  
  105. var init = function () {
  106.  
  107. };
  108.  
  109. function checkView() {
  110. var timeline = $('#timeline');
  111. if (timeline.hasClass('u0EntryList')) {
  112. entryClass = 'u0Entry';
  113. return 'u0';
  114. }
  115. if (timeline.hasClass('u4EntryList')) {
  116. entryClass = 'u4Entry';
  117. return 'u4';
  118. }
  119. if (timeline.hasClass('u5EntryList')) {
  120. entryClass = 'u5Entry';
  121. return 'u5';
  122. }
  123. if (timeline.hasClass('u100EntryList')) {
  124. entryClass = 'u100Frame';
  125. return 'u100';
  126. }
  127. return 'error';
  128. }
  129.  
  130. function addFavicon(e) {
  131. var fav = e.find('.slim-favicon');
  132. if (fav.length == 0) {
  133. switch (checkView()) {
  134. case 'u0':
  135. var info = e.find('.sourceInfo');
  136. var uri = info.find('a').attr('data-uri');
  137. info.before(favicons[uri]);
  138. break;
  139. case 'u4':
  140. var info = e.find('.sourceTitle');
  141. var uri = info.attr('data-uri');
  142. info.before(favicons[uri]);
  143. break;
  144. case 'u5':
  145. var info = e.find('.sourceTitle');
  146. var uri = info.attr('data-uri');
  147. info.before(favicons[uri]);
  148. break;
  149. case 'u100':
  150. var info = e.find('.sourceTitle');
  151. var uri = info.attr('data-uri');
  152. info.before(favicons[uri]);
  153. break;
  154. }
  155. }
  156. }
  157.  
  158. function addArrows(e) {
  159. var tools = e.find('.condensedTools');
  160.  
  161. if (tools.find('.arrow_down').length == 0) {
  162. var a = $(tools).prepend(arrow_up);
  163. $(a).find('.arrow_up').on('click', function () {
  164. if (!marking_as_read) {
  165. marking_as_read = true;
  166. $("#feedlyPart0").css("opacity", "0.1");
  167. $("#feedlyPart0").parent().prepend(h1read);
  168.  
  169. var entry = viewToClass();
  170. $(this).closest(entry).prevAll(entry).each(function () {
  171. if ($(this).find('.unread').length == 1) {
  172. $(this).click();
  173. }
  174. });
  175. $(this).closest(entry).click();
  176. $("#feedlyPart0").css("opacity", "1");
  177. $("#feedlyPart0").parent().find(".readWait").remove();
  178. marking_as_read = false;
  179. }
  180. });
  181. var b = tools.prepend(arrow_down);
  182. $(b).find('.arrow_down').on('click', function () {
  183. if (!marking_as_read) {
  184. marking_as_read = true;
  185. $("#feedlyPart0").css("opacity", "0.1");
  186. $("#feedlyPart0").parent().prepend(h1read);
  187.  
  188. var entry = viewToClass();
  189. $(this).closest(entry).nextAll(entry).each(function () {
  190. if ($(this).find('.unread').length == 1) {
  191. $(this).click();
  192. }
  193. });
  194. $(this).closest(entry).click();
  195. $("#feedlyPart0").css("opacity", "1");
  196. $("#feedlyPart0").parent().find(".readWait").remove();
  197. marking_as_read = false;
  198. }
  199. });
  200. }
  201. }
  202.  
  203. function generateFavicons() {
  204. var array = {};
  205. $('.tab').each(function () {
  206. $(this).find('.icon.handle').not('.expanded').each(function () {
  207. $(this).click();
  208. });
  209.  
  210. $(this).find('.favicon').each(function () {
  211. var element = $(this).clone();
  212. element.addClass('slim-favicon');
  213. element.removeClass('favicon');
  214. var image = $('<div/>').append(element).html();
  215.  
  216. var index = $(this).parent().attr('data-uri');
  217.  
  218. array[index] = image;
  219. });
  220.  
  221. $(this).find('.icon.handle.expanded').each(function () {
  222. $(this).click();
  223. });
  224. });
  225. return array;
  226. }
  227.  
  228. $(document).bind('DOMNodeInserted', function (event) {
  229. var element = event.target;
  230. if (element.attributes["id"].nodeValue == 'timeline') {
  231. $('#timeline > div').bind("DOMNodeInserted", function (event) {
  232. checkView();
  233. if ($(event.target).hasClass(entryClass)) {
  234. $(event.target).slimStyle();
  235. }
  236. });
  237.  
  238. }
  239. if (element.attributes["id"].nodeValue == 'feedlyTitleBar') {
  240. if ($('#cacheRefresh').length == 0) {
  241. var cache = $('.pageActionBar').prepend(cache_refresh);
  242. $(cache).find('#cacheRefresh').on('click', function () {
  243. favicons = generateFavicons();
  244. serialize("ffa_cache", favicons);
  245. alert('Cache refreshed. Please refresh page.');
  246. });
  247. }
  248. }
  249. });
  250. })(jQuery);