Greasy Fork is available in English.

Feedly Search

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;\
    }\
    .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);