// // pino.js - Open livedoor Reader (and clone server) pinned items - // // LICENSE: {{{ // // This software distributable under the terms of an MIT-style license. // // Copyright (c) 2009 snaka // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. // // OSI page : http://opensource.org/licenses/mit-license.php // Japanese : http://sourceforge.jp/projects/opensource/wiki/licenses%2FMIT_license // // }}} // PLUGIN INFO: {{{ var PLUGIN_INFO = xml` {NAME} Open livedoor Reader pinned items livedoor Reader でピンを立てたページを開く 3.0 https://github.com/vimpr/vimperator-plugins/raw/master/pino.js _libly.js snaka MIT style license 1.4.1 || { created_on : (create date), link : (url), title : (page title) } ||< plugins.pino.shift(): Return first item and remove pin. plugins.pino.remove(link): Remove pin from item that matched by 'link'. ]]> で先頭のn件(デフォルト5件、グローバル変数で調整可能) をバックグラウンドのタブで開きます。 で補完候補の一覧にピンを立てた記事の一覧から選択することもできます。 count を指定すると、その件数だけ開きます。 以下のオプションが指定可能です。 -list: ピンの一覧を表示します。 == グローバル変数 == g:pinoOpenItemsCount: 一度に開くピンの数 default: 5 g:pinoOpenBehavior: ピンを開くときの挙動、liberator.open()の第2引数として使用する値 参考)http://wiki.livedoor.jp/shin_yan/d/liberator%282%2e0%29#content_34 default: liberator.NEW_BACKGROUND_TAB g:pinoAscendingOrder: ピンの一覧の表示順を昇順(古い順)とするかどうか default: "false" (新しい順) g:pinoBaseURL: fastladder を使う場合は、この変数を "http://fastladder.com" とする。 default: "http://reader.livedoor.com" g:pinoOpenInterval: タブを開く間隔(ミリ秒) == API == plugins.pino.items(): ピンの一覧を配列で取得する。 ピンのデータ構造は以下のとおりとなっている。 >|| { created_on : (create date), link : (url), title : (page title) } ||< plugins.pino.shift(): 先頭のピンを取得して、そのピンを一覧から削除する。 plugins.pino.remove(link): linkに該当するピンを一覧から削除する。 ]]> `; // }}} let self = liberator.plugins.pino = (function() { // COMMAND /////////////////////////////////////////////////////// {{{ commands.addUserCommand( ["pinneditemopen", "pino"], "Open livedoor Reader(and clone server) pinned item", function(args) { let pins = new Pins(); let items = pins.items(); if (!items || items.length == 0) { liberator.echo("Pinned item doesn't exists."); return; } if (args["-list"]) { //let items = pins.items(); let list = xml`
{items.length} items.
`; liberator.echo(list, commandline.FORCE_MULTILINE); return; } if (args.string == "") { let pin; let max = (args.count >= 1) ? args.count : openItemsCount(); for(let i = 0; i < max; i++) { if (!(pin = pins.shift())) break; setTimeout(function(link) liberator.open(link, openBehavior()), openInterval() * i, pin.link); } } else { liberator.open(args.string, openBehavior()); pins.remove(args.string); } }, { literal: 0, count: true, completer: function(context) { var pins = new Pins(); context.title = ["url", "title"]; context.filters = [CompletionContext.Filter.textDescription]; context.anchored = false; context.completions = [ [i.link, i.title] for each (i in pins.items()) ]; }, options: [ [["-list", "-l"], commands.OPTION_NOARG] ] }, true // for Debug ); // }}} // GLOBAL VARIABLES ////////////////////////////////////////////// {{{ var gv = liberator.globalVariables; function openItemsCount() gv.pinoOpenItemsCount || 5; function ascending() window.eval(gv.pinoAscendingOrder) == true; // default: false function openBehavior() window.eval(gv.pinoOpenBehavior) || liberator.NEW_BACKGROUND_TAB; function baseURL() gv.pinoBaseURL || "http://reader.livedoor.com"; function openInterval() gv.pinoOpenInterval || 200; // }}} // CLASS ///////////////////////////////////////////////////////// {{{ function Pins() { this.cache = null; this.apiKey = getLDRApiKey(); this.sortOrder = ascending() ? function(a, b) (a.created_on < b.created_on ? -1 : 1) : function(a, b) (a.created_on > b.created_on ? -1 : 1); } Pins.prototype = { items : function() { let result = this.cache ? this.cache : this.cache = this._getPinnedItems(); return (result || []).sort(this.sortOrder); }, shift : function() { if (this.items().length == 0) return null; var pin = this.items().shift(); this.remove(pin.link); return pin; }, remove : function(link) { var unescapedLink = unescapeHTML(link); var request = new libly.Request( baseURL() + "/api/pin/remove", { //Cookie: "reader_sid=" + this.apiKey, //Referer: "http://reader.livedoor.com/reader/" }, { postBody: toQuery({link: unescapedLink, ApiKey: this.apiKey}) } ); request.addEventListener("success", function(data) { liberator.log("Removed pin from '" + link + "' was succeeded."); }); request.addEventListener("failure", function(data) { liberator.echoerr("Cannot remove pin"); }); request.post(); }, _getPinnedItems : function() { var result = null; var request = new libly.Request( baseURL() + "/api/pin/all", null, { asynchronous: false, postBody: toQuery({ApiKey: this.apiKey}) } ); request.addEventListener("success", function(data) { if (isLoginPage(data)) { liberator.echoerr("Can't get pinned list. Maybe you should login to livedoor."); return; } result = unentifyObjectValues(liberator.eval(data.responseText)); }); request.addEventListener("failure", function(data) { liberator.echoerr("Can't get pinned list!!!"); }); request.post(); return result; }, } // }}} // FUNCTIONS ///////////////////////////////////////////////////// {{{ var libly = plugins.libly; function getLDRApiKey() { var ioService = Cc["@mozilla.org/network/io-service;1"] .getService(Ci.nsIIOService); var uri = ioService.newURI(baseURL(), null, null); var channel = ioService.newChannelFromURI(uri); var cookie = Cc["@mozilla.org/cookieService;1"] .getService(Ci.nsICookieService) .getCookieString(uri, channel); var apiKey = cookie.match(/reader_sid=([^;]+)/); return apiKey ? apiKey[1]: null; } function unescapeHTML(source) { var result = source; [ [/</g, "<"], [/>/g, ">"], [/&/g, "&"] ].forEach( function(rule) { result = result.replace(rule[0], rule[1]); }); return result; } function toQuery(source) [encodeURIComponent(i) + "=" + encodeURIComponent(source[i]) for (i in source) ].join('&'); function isLoginPage(response) response.responseText.substr(0, 5) == '