Feedly Search

Add search box on Feedly

  1. // ==UserScript==
  2. // @name Feedly Search
  3. // @namespace http://nodaguti.usamimi.info/
  4. // @description Add search box on Feedly
  5. // @include http://feedly.com/*
  6. // @include https://feedly.com/*
  7. // @version 0.1
  8. // @author nodaguti
  9. // @license MIT License
  10. // @grant GM_log
  11. // @grant GM_addStyle
  12. // @grant unsafeWindow
  13. // ==/UserScript==
  14.  
  15. (function(window, document){
  16.  
  17. var DB_NAME = 'feedly-search-entries';
  18. var DB_VERSION = 1;
  19. var DB_STORE_NAME = 'entries';
  20. var DB = null;
  21.  
  22. var timeline = document.getElementById('box');
  23.  
  24. var SEARCH_ICON = "";
  25.  
  26. var STYLE = "\
  27. .hidden{\
  28. display: none !important;\
  29. }\
  30. \
  31. .invisible{\
  32. visibility: hidden !important;\
  33. }\
  34. \
  35. #feedlySearchBoxContainer{\
  36. position: absolute;\
  37. top: 0;\
  38. right: 0;\
  39. z-index: 99999;\
  40. color: rgb(102, 102, 102);\
  41. background-color: rgb(245, 245, 245);\
  42. box-shadow: 0px 2px 4px rgba(0, 0, 0, 0.2);\
  43. border: 1px solid rgb(190, 190, 190);\
  44. padding: 1em;\
  45. }\
  46. \
  47. #feedlySearchBoxContainer input[type='text']{\
  48. border: 1px #bfbfbf solid;\
  49. border-radius: 3px;\
  50. color: #444;\
  51. padding: 3px;\
  52. }\
  53. #feedlySearchBoxContainer button,\
  54. #feedlySearchBoxContainer input[type='checkbox'],\
  55. #feedlySearchBoxContainer select{\
  56. background-image: linear-gradient(to bottom, #ededed, #ededed 38%, #dedede);\
  57. border: 1px #ccc solid;\
  58. border-radius: 3px;\
  59. box-shadow: 0 1px 0 rgba(0, 0, 0, 0.1);\
  60. color: #444;\
  61. text-shadow: 0 1px 0 #f0f0f0;\
  62. padding: 2px 10px;\
  63. margin: 0 5px;\
  64. }\
  65. #feedlySearchBoxContainer button::-moz-focus-inner{\
  66. border: 0 !important;\
  67. padding: 0 !important;\
  68. }\
  69. #feedlySearchBoxContainer form{\
  70. margin: 0;\
  71. }\
  72. \
  73. #feedlySearchOptions{\
  74. margin-top: 5px;\
  75. font-size: 90%;\
  76. color: #333;\
  77. }\
  78. \
  79. #feedlySearchLoading{\
  80. width: 12px;\
  81. height: 12px;\
  82. border-radius: 50%;\
  83. border: 3px solid #333;\
  84. border-right-color: transparent;\
  85. animation: spin 1s linear infinite;\
  86. display: inline-block;\
  87. vertical-align: middle;\
  88. margin-left: 0.5em;\
  89. }\
  90. \
  91. @keyframes spin{\
  92. 0% { transform: rotate(0deg); opacity: 0.2; }\
  93. 50% { transform: rotate(180deg); opacity: 1.0; }\
  94. 100% { transform: rotate(360deg); opacity: 0.2; }\
  95. }\
  96. \
  97. #feedlySearchHitList{\
  98. list-style: none;\
  99. margin-top: 3em;\
  100. }\
  101. #feedlySearchHitList > li{\
  102. margin: 1em 0;\
  103. list-style: none;\
  104. }\
  105. #feedlySearchHitList a{\
  106. color: #1122CC !important;\
  107. }\
  108. .feedlySearchResultTitle{\
  109. font-size: 130%;\
  110. }\
  111. .feedlySearchResultSource{\
  112. display: inline-block;\
  113. margin-left: 0.5em;\
  114. font-size: 80%;\
  115. color: #888;\
  116. }\
  117. .feedlySearchResultSource::before{\
  118. content: '(';\
  119. }\
  120. .feedlySearchResultSource::after{\
  121. content: ')';\
  122. }\
  123. .feedlySearchResultBody{\
  124. padding: 1em 0 0 2.5em;\
  125. width: 80%;\
  126. font-size: 110%;\
  127. max-height: 200px;\
  128. overflow: hidden;\
  129. }\
  130. .feedlySearchResultBody:hover{\
  131. max-height: auto;\
  132. }";
  133.  
  134.  
  135.  
  136. var FeedlySearch = {
  137.  
  138. init: function(){
  139. //Add observer
  140. var observer = new MutationObserver(function(mutations){
  141. this.addEntries();
  142. }.bind(this));
  143.  
  144. observer.observe(timeline, { childList: true, subtree: true });
  145.  
  146. //Open database
  147. this.openDatabase();
  148.  
  149. //Add search button after waiting for building the page
  150. setTimeout(this.addSearchButton, 3000);
  151.  
  152. GM_addStyle(STYLE);
  153. },
  154.  
  155.  
  156. addSearchButton: function(){
  157. //Search Button
  158. var img = document.createElement("img");
  159. img.id = "pageActionSearch";
  160. img.className = "pageAction";
  161. img.width = "20";
  162. img.height = "20";
  163. img.border = "0";
  164. img.src = SEARCH_ICON;
  165. img.dataset.appAction = "search";
  166. img.title = "Search";
  167.  
  168. var parent = document.querySelector("#feedlyPageHeader > .pageActionBar");
  169. if(!parent) return;
  170.  
  171. parent.appendChild(img);
  172.  
  173. img.addEventListener('click', function(){
  174. var rect = document.getElementById('feedlyPageHeader').getBoundingClientRect();
  175. var searchbox = document.getElementById('feedlySearchBoxContainer');
  176.  
  177. //Adjust position of search box
  178. searchbox.style.top = (rect.bottom + 5) + 'px';
  179. searchbox.style.right = (document.documentElement.clientWidth - rect.right) + 'px';
  180.  
  181. //Show search box
  182. searchbox.classList.toggle('hidden');
  183.  
  184. //Focus search box
  185. document.getElementById('feedlySearchBox').focus();
  186. }, false);
  187.  
  188.  
  189. //Search Box
  190. var searchBoxTag = '' +
  191. '<form action="" onsubmit="FeedlySearch.search(document.getElementById(\'feedlySearchBox\').value);return false;">'+
  192. '<input type="text" id="feedlySearchBox" title="Split by whitepace to AND search" />'+
  193. '<button type="submit">Search</button>'+
  194. '<div id="feedlySearchLoading" class="invisible" title="Click to abort searching"></div>'+
  195. '<div id="feedlySearchOptions">'+
  196. '<label><input type="checkbox" id="feedlySearchTitle" checked="checked" /> Title</label> '+
  197. '<label><input type="checkbox" id="feedlySearchURL" checked="checked" /> URL</label> '+
  198. '<label><input type="checkbox" id="feedlySearchBody" checked="checked" /> Body</label> '+
  199. '<label><input type="checkbox" id="feedlySearchRegExp" /> RegExp</label>'+
  200. '</div>'+
  201. '</form>';
  202.  
  203. var container = document.createElement('div');
  204. container.id = 'feedlySearchBoxContainer';
  205. container.classList.add('hidden');
  206. document.body.appendChild(container);
  207. container.innerHTML = searchBoxTag;
  208.  
  209. setTimeout(function(){
  210. document.getElementById('feedlySearchLoading').addEventListener('click', function(){
  211. FeedlySearch.abortSearch();
  212. }, false);
  213. });
  214. },
  215.  
  216.  
  217. openDatabase: function(){
  218. var req = window.indexedDB.open(DB_NAME, DB_VERSION);
  219.  
  220. req.onerror = this.onError;
  221. req.onupgradeneeded = this.createDatabase;
  222. req.onsuccess = function(event){
  223. GM_log('Success: Opening the database.');
  224. DB = event.target.result;
  225. };
  226. },
  227.  
  228.  
  229. createDatabase: function(event){
  230. var objectStore = event.target.result.createObjectStore(DB_STORE_NAME, { keyPath: "id" });
  231.  
  232. GM_log("Success: Creating objectStore.");
  233. },
  234.  
  235.  
  236. resetDatabase: function(){
  237. DB.transaction([DB_STORE_NAME], "readwrite").objectStore(DB_STORE_NAME).clear();
  238. },
  239.  
  240.  
  241. addEntries: function(){
  242. GM_log("Adding new entries...");
  243.  
  244. var transaction = DB.transaction([DB_STORE_NAME], "readwrite");
  245. transaction.onerror = this.onError;
  246. transaction.oncomplete = this.onAllEntriesAdded;
  247.  
  248. var objectStore = transaction.objectStore(DB_STORE_NAME);
  249.  
  250. //unread articles
  251. var unreadEntries = Array.slice(timeline.getElementsByClassName('u0Entry')).filter(function(item){
  252. return item.getElementsByClassName('unread').length > 0;
  253. });
  254.  
  255. unreadEntries.forEach(function(entry){
  256. var id = entry.dataset.inlineentryid;
  257. var title = entry.dataset.title;
  258. var url = entry.dataset.alternateLink;
  259. var sourceTitle = entry.querySelector('.sourceTitle > a');
  260. var summary = entry.getElementsByClassName('u0Summary')[0].innerHTML;
  261.  
  262. var request = objectStore.put({
  263. id: id,
  264. title: title,
  265. url: url,
  266. sourceTitle: sourceTitle.firstChild.nodeValue,
  267. sourceURL: sourceTitle.href,
  268. body: summary,
  269. });
  270. request.onsuccess = FeedlySearch.onEntryAdded;
  271. request.onerror = FeedlySearch.onError;
  272. });
  273.  
  274.  
  275. //opened articles
  276. var selectedEntry = timeline.querySelector('.inlineFrame[data-uninlineentryid] .u100Entry');
  277. if(selectedEntry){
  278. var id = selectedEntry.dataset.selectentryid;
  279. var title = selectedEntry.dataset.title;
  280. var url = selectedEntry.dataset.alternateLink;
  281. var sourceTitle = selectedEntry.getElementsByClassName('sourceTitle')[0];
  282. var body = selectedEntry.getElementsByClassName('entryBody')[0];
  283.  
  284. var fullFeedLoaded = body.classList.contains('gm_fullfeed_loaded');
  285.  
  286. var content = fullFeedLoaded ? body : body.querySelector('.content');
  287.  
  288. var request = objectStore.put({
  289. id: id,
  290. title: title,
  291. url: url,
  292. sourceTitle: sourceTitle.firstChild.nodeValue,
  293. sourceURL: sourceTitle.href,
  294. body: content.innerHTML,
  295. });
  296. request.onsuccess = this.onEntryAdded;
  297. request.onerror = this.onError;
  298. }
  299.  
  300. //If remove the following code, this script doesn't work well. (I don't know why)
  301. if(objectStore.mozGetAll)
  302. objectStore.mozGetAll().onsuceess = function(event){};
  303. },
  304.  
  305.  
  306. onEntryAdded: function(event){
  307. GM_log("Entry Saved: " + event.target.result);
  308. },
  309.  
  310. onAllEntriesAdded: function(event){
  311. GM_log("Finish Saving All Entries.");
  312. },
  313.  
  314.  
  315. search: function(key){
  316. GM_log("Searching...");
  317. this._abortSearch = false;
  318.  
  319. var count = 0;
  320. var objectStore = DB.transaction([DB_STORE_NAME]).objectStore(DB_STORE_NAME);
  321.  
  322. //Get search options
  323. var optionTags = Array.slice(document.getElementById('feedlySearchOptions').querySelectorAll('input[type="checkbox"]'));
  324. var options = {};
  325. var keys;
  326.  
  327. optionTags.forEach(function(optionTag){
  328. options[optionTag.id.replace('feedlySearch', '').toLowerCase()] = optionTag.checked;
  329. });
  330.  
  331.  
  332. //Create RegExp Object if RegExp option selected
  333. if(options.regexp){
  334. keys = [new RegExp(key)];
  335. }else{
  336. keys = key.split(/[\s ]+/);
  337. }
  338.  
  339.  
  340. //Create Search Display
  341. var titleBar = document.getElementById('feedlyTitleBar');
  342. var hhint = titleBar.getElementsByClassName('hhint')[0];
  343.  
  344. //Change Title to "Search"
  345. titleBar.firstChild.nodeValue = 'Search';
  346. hhint.innerHTML = '';
  347.  
  348. //Show Loading icon
  349. var loadingIcon = document.getElementById('feedlySearchLoading');
  350. loadingIcon.classList.remove('invisible');
  351.  
  352. //Clear timeline
  353. var entriesArea = document.getElementById('mainArea');
  354. while(entriesArea.hasChildNodes()){
  355. entriesArea.removeChild(entriesArea.firstChild);
  356. }
  357.  
  358. //Create List
  359. var hitEntriesList = document.createElement('ul');
  360. hitEntriesList.id = "feedlySearchHitList";
  361. entriesArea.appendChild(hitEntriesList);
  362.  
  363. var startTime = Date.now();
  364.  
  365.  
  366. //Emphasize every hit term
  367. function emphasizeTerm(str, keys){
  368. var _str = str;
  369.  
  370. keys.forEach(function(key){
  371. _str = _str.replace(key, "<strong>$&</strong>", "g");
  372. });
  373.  
  374. return _str;
  375. }
  376.  
  377.  
  378. //Search
  379. objectStore.openCursor().onsuccess = function(event){
  380. var cursor = event.target.result;
  381.  
  382. if(cursor){
  383. var entry = cursor.value;
  384.  
  385. if(
  386. keys.every(function(key){
  387. return options.regexp ?
  388. (options.title && key.test(entry.title)) ||
  389. (options.url && key.test(entry.url)) ||
  390. (options.body && key.test(entry.body))
  391. :
  392. (options.title && entry.title.indexOf(key) > -1) ||
  393. (options.url && entry.url.indexOf(key) > -1) ||
  394. (options.body && entry.body.indexOf(key) > -1)
  395. })
  396. ){
  397. count++;
  398. hitEntriesList.insertAdjacentHTML("beforeend", "" +
  399. "<li>"+
  400. '<div class="feedlySearchResultTitle">' +
  401. '<a href="' + entry.url + '" target="_blank">' + emphasizeTerm(entry.title, keys) + "</a>" +
  402. '<div class="feedlySearchResultSource">' +
  403. '<a href="' + entry.sourceURL + '">' + entry.sourceTitle + "</a>" +
  404. "</div>" +
  405. "</div>" +
  406. '<div class="feedlySearchResultBody">' +
  407. emphasizeTerm(entry.body, keys) +
  408. '</div>' +
  409. "</li>");
  410. }
  411.  
  412. if(!FeedlySearch._abortSearch) return cursor.continue();
  413. }
  414.  
  415. loadingIcon.classList.add('invisible');
  416. GM_log("Search finished.");
  417.  
  418. if(count == 0){
  419. entriesArea.innerHTML = "No Entries Found.";
  420. }else{
  421. hhint.innerHTML = count + ' results (' + ((Date.now() - startTime) / 1000) + ' seconds)';
  422. }
  423. }
  424. },
  425.  
  426.  
  427. abortSearch: function(){
  428. this._abortSearch = true;
  429. },
  430.  
  431.  
  432.  
  433. onError: function(event){
  434. GM_log('Error has occurred.\n\nType: ' + event.type + '\nValue: ' + event.value);
  435. }
  436.  
  437. };
  438.  
  439.  
  440. window.FeedlySearch = FeedlySearch;
  441. FeedlySearch.init();
  442.  
  443. })(unsafeWindow, unsafeWindow.document);

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址