Feedly Search

Add search box on Feedly

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==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;\
    }\
    .feedlySearchResultTitle{\
        font-size: 130%;\
    }\
    .feedlySearchResultSource{\
        display: inline-block;\
        margin-left: 0.5em;\
        font-size: 80%;\
        color: #888;\
    }\
    .feedlySearchResultSource::before{\
        content: '(';\
    }\
    .feedlySearchResultSource::after{\
        content: ')';\
    }\
    .feedlySearchResultBody{\
        padding: 1em 0 0 2.5em;\
        width: 80%;\
        font-size: 110%;\
        max-height: 200px;\
        overflow: hidden;\
    }\
    .feedlySearchResultBody: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.result;
        };
    },


    createDatabase: function(event){
        var objectStore = event.target.result.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.result);
    },

    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.result;

            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="feedlySearchResultTitle">' +
                                '<a href="' + entry.url + '" target="_blank">' + emphasizeTerm(entry.title, keys) + "</a>" +
                                '<div class="feedlySearchResultSource">' +
                                    '<a href="' + entry.sourceURL + '">' + entry.sourceTitle + "</a>" +
                                "</div>" +
                            "</div>" +
                            '<div class="feedlySearchResultBody">' +
                                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 + ' results (' + ((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);