Add search box on Feedly
// ==UserScript== // @name Feedly Search // @namespace http://nodaguti.usamimi.info/ // @description Add search box on Feedly // @include http://feedly.com/* // @include https://feedly.com/* // @version 0.1 // @author nodaguti // @license MIT License // @grant GM_log // @grant GM_addStyle // @grant unsafeWindow // ==/UserScript== (function(window, document){ var DB_NAME = 'feedly-search-entries'; var DB_VERSION = 1; var DB_STORE_NAME = 'entries'; var DB = null; var timeline = document.getElementById('box'); var SEARCH_ICON = ""; var STYLE = "\ .hidden{\ display: none !important;\ }\ \ .invisible{\ visibility: hidden !important;\ }\ \ #feedlySearchBoxContainer{\ position: absolute;\ top: 0;\ right: 0;\ z-index: 99999;\ color: rgb(102, 102, 102);\ background-color: rgb(245, 245, 245);\ box-shadow: 0px 2px 4px rgba(0, 0, 0, 0.2);\ border: 1px solid rgb(190, 190, 190);\ padding: 1em;\ }\ \ #feedlySearchBoxContainer input[type='text']{\ border: 1px #bfbfbf solid;\ border-radius: 3px;\ color: #444;\ padding: 3px;\ }\ #feedlySearchBoxContainer button,\ #feedlySearchBoxContainer input[type='checkbox'],\ #feedlySearchBoxContainer select{\ background-image: linear-gradient(to bottom, #ededed, #ededed 38%, #dedede);\ border: 1px #ccc solid;\ border-radius: 3px;\ box-shadow: 0 1px 0 rgba(0, 0, 0, 0.1);\ color: #444;\ text-shadow: 0 1px 0 #f0f0f0;\ padding: 2px 10px;\ margin: 0 5px;\ }\ #feedlySearchBoxContainer button::-moz-focus-inner{\ border: 0 !important;\ padding: 0 !important;\ }\ #feedlySearchBoxContainer form{\ margin: 0;\ }\ \ #feedlySearchOptions{\ margin-top: 5px;\ font-size: 90%;\ color: #333;\ }\ \ #feedlySearchLoading{\ width: 12px;\ height: 12px;\ border-radius: 50%;\ border: 3px solid #333;\ border-right-color: transparent;\ animation: spin 1s linear infinite;\ display: inline-block;\ vertical-align: middle;\ margin-left: 0.5em;\ }\ \ @keyframes spin{\ 0% { transform: rotate(0deg); opacity: 0.2; }\ 50% { transform: rotate(180deg); opacity: 1.0; }\ 100% { transform: rotate(360deg); opacity: 0.2; }\ }\ \ #feedlySearchHitList{\ list-style: none;\ margin-top: 3em;\ }\ #feedlySearchHitList > li{\ margin: 1em 0;\ list-style: none;\ }\ #feedlySearchHitList a{\ color: #1122CC !important;\ }\ .feedlySearchR###ltTitle{\ font-size: 130%;\ }\ .feedlySearchR###ltSource{\ display: inline-block;\ margin-left: 0.5em;\ font-size: 80%;\ color: #888;\ }\ .feedlySearchR###ltSource::before{\ content: '(';\ }\ .feedlySearchR###ltSource::after{\ content: ')';\ }\ .feedlySearchR###ltBody{\ padding: 1em 0 0 2.5em;\ width: 80%;\ font-size: 110%;\ max-height: 200px;\ overflow: hidden;\ }\ .feedlySearchR###ltBody:hover{\ max-height: auto;\ }"; var FeedlySearch = { init: function(){ //Add observer var observer = new MutationObserver(function(mutations){ this.addEntries(); }.bind(this)); observer.observe(timeline, { childList: true, subtree: true }); //Open database this.openDatabase(); //Add search button after waiting for building the page setTimeout(this.addSearchButton, 3000); GM_addStyle(STYLE); }, addSearchButton: function(){ //Search Button var img = document.createElement("img"); img.id = "pageActionSearch"; img.className = "pageAction"; img.width = "20"; img.height = "20"; img.border = "0"; img.src = SEARCH_ICON; img.dataset.appAction = "search"; img.title = "Search"; var parent = document.querySelector("#feedlyPageHeader > .pageActionBar"); if(!parent) return; parent.appendChild(img); img.addEventListener('click', function(){ var rect = document.getElementById('feedlyPageHeader').getBoundingClientRect(); var searchbox = document.getElementById('feedlySearchBoxContainer'); //Adjust position of search box searchbox.style.top = (rect.bottom + 5) + 'px'; searchbox.style.right = (document.documentElement.clientWidth - rect.right) + 'px'; //Show search box searchbox.classList.toggle('hidden'); //Focus search box document.getElementById('feedlySearchBox').focus(); }, false); //Search Box var searchBoxTag = '' + '<form action="" onsubmit="FeedlySearch.search(document.getElementById(\'feedlySearchBox\').value);return false;">'+ '<input type="text" id="feedlySearchBox" title="Split by whitepace to AND search" />'+ '<button type="submit">Search</button>'+ '<div id="feedlySearchLoading" class="invisible" title="Click to abort searching"></div>'+ '<div id="feedlySearchOptions">'+ '<label><input type="checkbox" id="feedlySearchTitle" checked="checked" /> Title</label> '+ '<label><input type="checkbox" id="feedlySearchURL" checked="checked" /> URL</label> '+ '<label><input type="checkbox" id="feedlySearchBody" checked="checked" /> Body</label> '+ '<label><input type="checkbox" id="feedlySearchRegExp" /> RegExp</label>'+ '</div>'+ '</form>'; var container = document.createElement('div'); container.id = 'feedlySearchBoxContainer'; container.classList.add('hidden'); document.body.appendChild(container); container.innerHTML = searchBoxTag; setTimeout(function(){ document.getElementById('feedlySearchLoading').addEventListener('click', function(){ FeedlySearch.abortSearch(); }, false); }); }, openDatabase: function(){ var req = window.indexedDB.open(DB_NAME, DB_VERSION); req.onerror = this.onError; req.onupgradeneeded = this.createDatabase; req.onsuccess = function(event){ GM_log('Success: Opening the database.'); DB = event.target.r###lt; }; }, createDatabase: function(event){ var objectStore = event.target.r###lt.createObjectStore(DB_STORE_NAME, { keyPath: "id" }); GM_log("Success: Creating objectStore."); }, resetDatabase: function(){ DB.transaction([DB_STORE_NAME], "readwrite").objectStore(DB_STORE_NAME).clear(); }, addEntries: function(){ GM_log("Adding new entries..."); var transaction = DB.transaction([DB_STORE_NAME], "readwrite"); transaction.onerror = this.onError; transaction.oncomplete = this.onAllEntriesAdded; var objectStore = transaction.objectStore(DB_STORE_NAME); //unread articles var unreadEntries = Array.slice(timeline.getElementsByClassName('u0Entry')).filter(function(item){ return item.getElementsByClassName('unread').length > 0; }); unreadEntries.forEach(function(entry){ var id = entry.dataset.inlineentryid; var title = entry.dataset.title; var url = entry.dataset.alternateLink; var sourceTitle = entry.querySelector('.sourceTitle > a'); var summary = entry.getElementsByClassName('u0Summary')[0].innerHTML; var request = objectStore.put({ id: id, title: title, url: url, sourceTitle: sourceTitle.firstChild.nodeValue, sourceURL: sourceTitle.href, body: summary, }); request.onsuccess = FeedlySearch.onEntryAdded; request.onerror = FeedlySearch.onError; }); //opened articles var selectedEntry = timeline.querySelector('.inlineFrame[data-uninlineentryid] .u100Entry'); if(selectedEntry){ var id = selectedEntry.dataset.selectentryid; var title = selectedEntry.dataset.title; var url = selectedEntry.dataset.alternateLink; var sourceTitle = selectedEntry.getElementsByClassName('sourceTitle')[0]; var body = selectedEntry.getElementsByClassName('entryBody')[0]; var fullFeedLoaded = body.classList.contains('gm_fullfeed_loaded'); var content = fullFeedLoaded ? body : body.querySelector('.content'); var request = objectStore.put({ id: id, title: title, url: url, sourceTitle: sourceTitle.firstChild.nodeValue, sourceURL: sourceTitle.href, body: content.innerHTML, }); request.onsuccess = this.onEntryAdded; request.onerror = this.onError; } //If remove the following code, this script doesn't work well. (I don't know why) if(objectStore.mozGetAll) objectStore.mozGetAll().onsuceess = function(event){}; }, onEntryAdded: function(event){ GM_log("Entry Saved: " + event.target.r###lt); }, onAllEntriesAdded: function(event){ GM_log("Finish Saving All Entries."); }, search: function(key){ GM_log("Searching..."); this._abortSearch = false; var count = 0; var objectStore = DB.transaction([DB_STORE_NAME]).objectStore(DB_STORE_NAME); //Get search options var optionTags = Array.slice(document.getElementById('feedlySearchOptions').querySelectorAll('input[type="checkbox"]')); var options = {}; var keys; optionTags.forEach(function(optionTag){ options[optionTag.id.replace('feedlySearch', '').toLowerCase()] = optionTag.checked; }); //Create RegExp Object if RegExp option selected if(options.regexp){ keys = [new RegExp(key)]; }else{ keys = key.split(/[\s ]+/); } //Create Search Display var titleBar = document.getElementById('feedlyTitleBar'); var hhint = titleBar.getElementsByClassName('hhint')[0]; //Change Title to "Search" titleBar.firstChild.nodeValue = 'Search'; hhint.innerHTML = ''; //Show Loading icon var loadingIcon = document.getElementById('feedlySearchLoading'); loadingIcon.classList.remove('invisible'); //Clear timeline var entriesArea = document.getElementById('mainArea'); while(entriesArea.hasChildNodes()){ entriesArea.removeChild(entriesArea.firstChild); } //Create List var hitEntriesList = document.createElement('ul'); hitEntriesList.id = "feedlySearchHitList"; entriesArea.appendChild(hitEntriesList); var startTime = Date.now(); //Emphasize every hit term function emphasizeTerm(str, keys){ var _str = str; keys.forEach(function(key){ _str = _str.replace(key, "<strong>$&</strong>", "g"); }); return _str; } //Search objectStore.openCursor().onsuccess = function(event){ var cursor = event.target.r###lt; if(cursor){ var entry = cursor.value; if( keys.every(function(key){ return options.regexp ? (options.title && key.test(entry.title)) || (options.url && key.test(entry.url)) || (options.body && key.test(entry.body)) : (options.title && entry.title.indexOf(key) > -1) || (options.url && entry.url.indexOf(key) > -1) || (options.body && entry.body.indexOf(key) > -1) }) ){ count++; hitEntriesList.insertAdjacentHTML("beforeend", "" + "<li>"+ '<div class="feedlySearchR###ltTitle">' + '<a href="' + entry.url + '" target="_blank">' + emphasizeTerm(entry.title, keys) + "</a>" + '<div class="feedlySearchR###ltSource">' + '<a href="' + entry.sourceURL + '">' + entry.sourceTitle + "</a>" + "</div>" + "</div>" + '<div class="feedlySearchR###ltBody">' + emphasizeTerm(entry.body, keys) + '</div>' + "</li>"); } if(!FeedlySearch._abortSearch) return cursor.continue(); } loadingIcon.classList.add('invisible'); GM_log("Search finished."); if(count == 0){ entriesArea.innerHTML = "No Entries Found."; }else{ hhint.innerHTML = count + ' r###lts (' + ((Date.now() - startTime) / 1000) + ' seconds)'; } } }, abortSearch: function(){ this._abortSearch = true; }, onError: function(event){ GM_log('Error has occurred.\n\nType: ' + event.type + '\nValue: ' + event.value); } }; window.FeedlySearch = FeedlySearch; FeedlySearch.init(); })(unsafeWindow, unsafeWindow.document);