/**
* ==VimperatorPlugin==
* @name moreqmarks.js
* @description add feature(record position, stack, queue) to QuickMarks
* @description-ja QuickMarksに機能追加(位置の記憶、qmarksとは別のスタックとキュー追加)
* @author hogelog
* @version 0.05
* ==/VimperatorPlugin==
*
* MAPPINGS:
* "gj" -> Jump to QuickMark for current URL
* "gd" -> Delete QuickMark for current URL
* "gs" -> Push QuickMarkStack for current URL
* "gS" -> Pop QuickMarkStack and Jump
* "gq" -> Queue QuickMarkQueue for current URL
* "gQ" -> Dequeue QuickMarkStack and Jump
*
* COMMANDS:
* :qmarkpu[sh], :qmpu[sh] -> Push QuickMarkStack for current URL
* :qmarkpo[p], :qmpo[p] -> Pop QuickMarkStack and Jump
* :stackli[st], :stli[st] -> List QuickMarkStack
* :qmarkqu[eue], :qmqu[eue] -> Queue QuickMarkQueue for current URL
* :qmarkde[que], :qmde[que] -> Dequeue QuickMarkStack and Jump
* :queueli[st], :quli[st] -> List QuickMarkQueue
*
*/
(function(){
var use_position = true;
var use_default_data = true;
var qmarks = {};
var qmark_stack = [];
var qmark_queue = [];
// TODO: move to a storage module
var savedMarks = liberator.modules.options.getPref("extensions.vimperator.moreqmarks", "").split("\n");
var savedMarkStack = liberator.modules.options.getPref("extensions.vimperator.moreqmarkstack", "").split("\n");
var savedMarkQueue = liberator.modules.options.getPref("extensions.vimperator.moreqmarkqueue", "").split("\n");
// load the saved quickmarks -- TODO: change to sqlite
if(use_default_data) {
var defaultMarks = liberator.modules.options.getPref("extensions.vimperator.quickmarks", "").split("\n");
for (var i = 0; i < defaultMarks.length - 1; i += 2) {
var url = defaultMarks[i+1];
qmarks[defaultMarks[i]] = {url: url, x: 0, y: 0};
}
}
for (var i = 0; i < savedMarks.length - 1; i += 4) {
var url = savedMarks[i+1];
var x = savedMarks[i+2];
var y = savedMarks[i+3];
qmarks[savedMarks[i]] = {url: url, x: x, y:y};
}
for (var i = 0; i < savedMarkStack.length - 1; i += 3) {
var url = savedMarkStack[i];
var x = savedMarkStack[i+1];
var y = savedMarkStack[i+2];
qmark_stack.push({url: url, x: x, y:y});
}
for (var i = 0; i < savedMarkQueue.length - 1; i += 3) {
var url = savedMarkQueue[i];
var x = savedMarkQueue[i+1];
var y = savedMarkQueue[i+2];
qmark_queue.unshift({url: url, x: x, y:y});
}
function add_qmark(qmark, item, target) {
switch(target) {
case "stack":
qmark_stack.push(item);
break;
case "queue":
qmark_queue.unshift(item);
break;
case "mark":
default:
qmarks[qmark] = item;
break;
}
}
function get_qmark(qmark, target) {
switch(target) {
case "stack":
return qmark_stack[qmark_stack.length-1];
case "queue":
return item = qmark_queue[qmark_queue.length-1];
case "mark":
default:
return qmarks[qmark];
}
}
function get_qmarks(target) {
var marks = [];
// TODO: should we sort these in a-zA-Z0-9 order?
var count = 0;
var list;
switch(target) {
case "stack":
list = qmark_stack;
break;
case "queue":
list = qmark_queue;
break;
case "mark":
default:
list = qmarks;
break;
}
for (var mark in list) {
marks.push([mark, list[mark].url, list[mark].x, list[mark].y]);
}
marks.sort();
return marks;
}
function list_qmarks(marks) {
if(use_position) {
var list = ":" + liberator.modules.util.escapeHTML(liberator.modules.commandline.command) + "
" +
"
mark | line | col | file |
";
for (var i = 0; i < marks.length; i++)
{
list += "" +
" " + marks[i][0] + " | " +
"" + Math.round(marks[i][2] * 100) + "% | " +
"" + Math.round(marks[i][3] * 100) + "% | " +
"" + liberator.modules.util.escapeHTML(marks[i][1]) + " | " +
"
";
}
list += "
";
return list;
} else {
var list = ":" + liberator.modules.util.escapeHTML(liberator.modules.commandline.command) + "
" +
"QuickMark | URL |
";
for (var i = 0; i < marks.length; i++)
{
list += " " + marks[i][0] +
" | " + liberator.modules.util.escapeHTML(marks[i][1]) + " |
";
}
list += "
";
return list;
}
}
function jump_item(item, where, find) {
var url = item.url;
var x = item.x;
var y = item.y;
if (url) {
if(find) {
for (let [number, browser] in Iterator(liberator.modules.tabs.browsers)) {
var marked_url = browser.contentDocument.location.href;
if(marked_url == url) {
liberator.modules.tabs.select(number, false);
var win = getBrowser().selectedTab.linkedBrowser.contentWindow;
if(use_position) {
if(x!=0 || y!=0) {
win.scrollTo(x*win.scrollMaxX, y*win.scrollMaxY);
}
}
return true;
}
}
}
var tab = open_url(url, where);
if(tab) {
var win = tab.linkedBrowser.contentWindow;
if(use_position) {
if(x!=0 || y!=0) {
// scrollMaxX and scrollMaxY are 0 before construct content
// But small document(scrollMaxX = 0 and scrollMaxY = 0) is marked,
// maybe f is run forever.
var f = function() {
if(win.scrollMaxX==0 && win.scrollMaxY==0) {
setTimeout(f, 100);
} else {
win.scrollTo(x*win.scrollMaxX, y*win.scrollMaxY);
}
}
f();
}
}
}
return true;
}
return false;
}
function save_qmarks(target) {
switch(target) {
case "stack":
var savedQuickMarkStack = "";
for (var mark in qmark_stack) {
savedQuickMarkStack += qmark_stack[mark].url + "\n";
savedQuickMarkStack += qmark_stack[mark].x + "\n";
savedQuickMarkStack += qmark_stack[mark].y + "\n";
}
liberator.modules.options.setPref("extensions.vimperator.moreqmarkstack", savedQuickMarkStack);
break;
case "queue":
var savedQuickMarkQueue = "";
for (var mark in qmark_queue) {
savedQuickMarkQueue += qmark_queue[mark].url + "\n";
savedQuickMarkQueue += qmark_queue[mark].x + "\n";
savedQuickMarkQueue += qmark_queue[mark].y + "\n";
}
liberator.modules.options.setPref("extensions.vimperator.moreqmarkqueue", savedQuickMarkQueue);
break;
case "mark":
default:
var savedQuickMarks = "";
for (var mark in qmarks) {
savedQuickMarks += mark + "\n";
savedQuickMarks += qmarks[mark].url + "\n";
savedQuickMarks += qmarks[mark].x + "\n";
savedQuickMarks += qmarks[mark].y + "\n";
}
liberator.modules.options.setPref("extensions.vimperator.moreqmarks", savedQuickMarks);
if(use_default_data) {
var savedQuickMarks = "";
for (var mark in qmarks) {
savedQuickMarks += mark + "\n";
savedQuickMarks += qmarks[mark].url + "\n";
}
liberator.modules.options.setPref("extensions.vimperator.quickmarks", savedQuickMarks);
}
break;
}
}
function open_url(url, where) {
if (liberator.forceNewTab && liberator.has("tabs"))
where = liberator.NEW_TAB;
else if (!where || !liberator.has("tabs"))
where = liberator.CURRENT_TAB;
var whichwindow = window;
// decide where to load the first url
switch (where)
{
case liberator.CURRENT_TAB:
getBrowser().loadURIWithFlags(url, null, null, null, null);
return getBrowser().selectedTab;
case liberator.NEW_TAB:
return getBrowser().selectedTab = getBrowser().addTab(url, null, null, null);
case liberator.NEW_BACKGROUND_TAB:
return getBrowser().addTab(url, null, null, null);
case liberator.NEW_WINDOW:
window.open();
var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
.getService(Components.interfaces.nsIWindowMediator);
whichwindow = wm.getMostRecentWindow("navigator:browser");
whichwindow.loadURI(url, null, null);
return witchWindow.getBrowser().selectedTab;
case "mark":
default:
liberator.echoerr("Exxx: Invalid 'where' directive in liberator.plugins.moreqmarks openurls(...)");
return false;
}
}
//// MAPPINGS
var modes = [liberator.modules.modes.NORMAL];
liberator.modules.mappings.addUserMap(modes,
["gj"], "Jump to QuickMark for current URL",
function (arg)
{
var where = /\bquickmark\b/.test(liberator.modules.options["activate"]) ? liberator.NEW_TAB : liberator.NEW_BACKGROUND_TAB;
liberator.modules.quickmarks.jumpTo(arg, where, true);
},
{arg: true});
liberator.modules.mappings.addUserMap(modes,
["gd"], "Delete QuickMark for current URL",
function ()
{
liberator.plugins.moreqmarks.remove('', liberator.modules.buffer.URL);
});
liberator.modules.mappings.addUserMap(modes,
["gs"], "Push QuickMarkStack for current URL",
function ()
{
liberator.plugins.moreqmarks.add("", liberator.modules.buffer.URL, "stack");
});
liberator.modules.mappings.addUserMap(modes,
["gS"], "Pop QuickMarkStack and Jump",
function ()
{
var where = /\bquickmark\b/.test(liberator.modules.options["activate"]) ? liberator.NEW_TAB : liberator.NEW_BACKGROUND_TAB;
if(liberator.modules.quickmarks.jumpTo("", where, true, "stack")) {
liberator.modules.quickmarks.remove("", "", "stack");
}
});
liberator.modules.mappings.addUserMap(modes,
["gq"], "Queue QuickMarkQueue for current URL",
function ()
{
liberator.plugins.moreqmarks.add("", liberator.modules.buffer.URL, "queue");
});
liberator.modules.mappings.addUserMap(modes,
["gQ"], "Dequeue QuickMarkStack and Jump",
function ()
{
var where = /\bquickmark\b/.test(liberator.modules.options["activate"]) ? liberator.NEW_TAB : liberator.NEW_BACKGROUND_TAB;
if(liberator.modules.quickmarks.jumpTo("", where, true, "queue")) {
liberator.modules.quickmarks.remove("", "", "queue");
}
});
//// COMMANDS
liberator.modules.commands.add(["qmarkpu[sh]", "qmpu[sh]"], "Push QuickMarkStack for current URL",
function ()
{
liberator.plugins.moreqmarks.add("", liberator.modules.buffer.URL, "stack");
});
liberator.modules.commands.add(["qmarkpo[p]", "qmpo[p]"], "Pop QuickMarkStack and Jump",
function ()
{
var where = /\bquickmark\b/.test(liberator.modules.options["activate"]) ? liberator.NEW_TAB : liberator.NEW_BACKGROUND_TAB;
liberator.modules.quickmarks.jumpTo("", where, true, "stack");
});
liberator.modules.commands.add(["stackli[st]", "stls"], "List QuickMarkStack",
function ()
{
liberator.plugins.moreqmarks.list("", "stack");
});
liberator.modules.commands.add(["qmarkqu[eue]", "qmqu[eue]"], "Queue QuickMarkQueue for current URL",
function ()
{
liberator.plugins.moreqmarks.add("", liberator.modules.buffer.URL, "queue");
});
liberator.modules.commands.add(["qmarkde[que]", "qmde[que]"], "Dequeue QuickMarkStack and Jump",
function ()
{
var where = /\bquickmark\b/.test(liberator.modules.options["activate"]) ? liberator.NEW_TAB : liberator.NEW_BACKGROUND_TAB;
liberator.modules.quickmarks.jumpTo("", where, true, "queue");
});
liberator.modules.commands.add(["queueli[st]", "quli[st]"], "List QuickMarkQueue",
function ()
{
liberator.plugins.moreqmarks.list("", "queue");
});
//// PUBLIC SECTION
liberator.plugins.moreqmarks = {
add: function (qmark, url, target)
{
if(use_position) {
var win = window.content;
var x, y;
if (win.document.body.localName.toLowerCase() == "frameset") {
x = 0;
y = 0;
} else {
x = win.scrollMaxX ? win.pageXO/* {{{
Copyright (c) 2008-2009, anekos.
All rights reserved.
Redistribution and use in source and binary forms, with or without modification,
are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
3. The names of the authors may not be used to endorse or promote products
derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
THE POSSIBILITY OF SUCH DAMAGE.
###################################################################################
# http://sourceforge.jp/projects/opensource/wiki/licenses%2Fnew_BSD_license #
# に参考になる日本語訳がありますが、有効なのは上記英文となります。 #
###################################################################################
}}} */
// PLUGIN_INFO {{{
let PLUGIN_INFO =
<VimperatorPlugin>
<name>Read Cat Later</name>
<description>Read it later</description>
<description lang="ja">後で読む</description>
<version>1.1.1</version>
<author mail="anekos@snca.net" homepage="http://d.hatena.ne.jp/nokturnalmortum/">anekos</author>
<license>new BSD License (Please read the source code comments of this plugin)</license>
<license lang="ja">修正BSDライセンス (ソースコードのコメントを参照してください)</license>
<updateURL>http://svn.coderepos.org/share/lang/javascript/vimperator-plugins/trunk/readcatlater.js</updateURL>
<minVersion>2.0pre</minVersion>
<maxVersion>2.2pre</maxVersion>
<detail><![CDATA[
== Usage ==
:readcatlater [TAG]:
:rcl:
Add current URL to read-it-later list.
:readcatnow URL:
:rcn URL:
Open the URL and delete from bookmarks.
When used with "!", delete the URL from bookmarks.
== Link ==
http://d.hatena.ne.jp/nokturnalmortum/20080918#1221729188
]]></detail>
<detail lang="ja"><![CDATA[
== Usage ==
:readcatlater [タグ]:
:rcl:
現在のURLを後で読むリストに追加する。
:readcatnow [-n=開く数] URL:
:rcn URL:
そのURLを開いて、ブックマークから削除する。
"!" 付きの場合は、ブックマークから削除する。
== Link ==
http://d.hatena.ne.jp/nokturnalmortum/20080918#1221729188
]]></detail>
</VimperatorPlugin>;
// }}}
(function () {
liberator.log('readcatlater.js loading');
// このプラグインでブックマークしたときに必ずつくタグ
const RCL_TAG = 'readcatlater';
// このプラグインが保存するブックマークのフォルダ名
// 変更しても良いし、場所を移動しても平気である。
const FOLDER_NAME = 'L';
// 逆順表示
const REVERSE = eval(liberator.globalVariables.readcatlater_reverse || 'false');
let prefs = {
prefix: 'extensions.vimperator.plugins.readcatlater.',
prefs: Components.classes["@mozilla.org/preferences-service;1"].
getService(Components.interfaces.nsIPrefBranch),
get: function (name, def) {
var name, type;
try {
const nsIPrefBranch = Components.interfaces.nsIPrefBranch;
name = this.prefix + name;
type = this.prefs.getPrefType(name);
switch (type) {
case nsIPrefBranch.PREF_STRING:
try {
return this.prefs.getComplexValue(name, Components.interfaces.nsISupportsString).data;
} catch (e) {
this.prefs.getCharPref(name);
}
break;
case nsIPrefBranch.PREF_INT:
return this.prefs.getIntPref(name);
break;
case nsIPrefBranch.PREF_BOOL:
return this.prefs.getBoolPref(name);
default:
return def;
}
} catch (e) {
return def;
}
},
set: function (name, value, type) {
var name = this.prefix + name;
switch (type || typeof value) {
case 'string':
let str = Cc['@mozilla.org/supports-string;1'].
createInstance(Ci.nsISupportsString);
str.data = value;
return this.prefs.setComplexValue(name, Components.interfaces.nsISupportsString, str);
case 'boolean':
return this.prefs.setBoolPref(name, value);
case 'number':
return this.prefs.setIntPref(name, value);
default:
alert('unknown pref type');
}
},
};
const migemo = Cc['@piro.sakura.ne.jp/xmigemo/factory;1'].
getService(Components.interfaces.pIXMigemoFactory).
getService("ja");
const tagssvc = Cc["@mozilla.org/browser/tagging-service;1"].
getService(Ci.nsITaggingService);
const IOService = Cc["@mozilla.org/network/io-service;1"].
getService(Ci.nsIIOService);
const myURI = IOService.newURI("http://www.mozilla.com", null, null);
const bookmarks = Cc["@mozilla.org/browser/nav-bookmarks-service;1"].
getService(Ci.nsINavBookmarksService);
const history = Cc["@mozilla.org/browser/nav-history-service;1"].
getService(Ci.nsINavHistoryService);
let [folderId, folderGUID] = [prefs.get('folderId', false), prefs.get('folderGUID', false)];
if (!(folderGUID && folderId && (folderId == bookmarks.getItemIdForGUID(folderGUID)))) {
folderId = bookmarks.createFolder(bookmarks.toolbarFolder, FOLDER_NAME, bookmarks.DEFAULT_INDEX);
prefs.set('folderId', folderId);
prefs.set('folderGUID', bookmarks.getItemGUID(folderId));
}
/*
// Get an array of tags for a URI
let myTags = tagssvc.getTagsForURI(myURI, {});
// Get an array of URIs for a tag
let taggedURIs = tagssvc.getURIsForTag("mozilla");
// Get an array of all tags
let arrayOfAllTags = tagssvc.allTags;
// Remove tags from a URI
tagssvc.untagURI(myURI, ["mozilla", "firefox"]);
*/
function openURI (uri) {
if (window.content.document.location.href == 'about:blank') {
window.loadURI(uri);
} else {
if ('delayedOpenTab' in window)
window.delayedOpenTab(uri, null);
else
window.getBrowser().addTab(uri, null);
}
}
function makeURI (uri) {
return IOService.newURI(uri, null, null);
}
function addEntry (doc, tags) {
var loc = doc.location.href;
var title = doc.title || loc;
var uri = makeURI(loc);
var bkmk = bookmarks.insertBookmark(folderId, uri, bookmarks.DEFAULT_INDEX, title);
tags.push(RCL_TAG);
tagssvc.tagURI(uri, tags);
return title;
}
function splitBySpaces (str) {
return [it for each (it in str.split(/\s+/)) if (/\w/.test(it))];
}
function RCL_Bookmarks (terms) { try {
let query = history.getNewQuery();
query.setFolders([folderId], 1);
let m;
if (terms) {
let ts = splitBySpaces(terms).map(function (it) {
var re = new RegExp(migemo.getRegExp(it), 'i');
var f = function (it) re.exec(it);
return {s: it, r: f};
});
m = function (it)
ts.some(function (t)
(it.URI.indexOf(t.s) >= 0 || it.title.indexOf(t.s) >= 0 ||
t.r(it.URI) || t.r(it.title))
? true : false);
} else {
m = function () true;
}
let result = [];
let qres = history.executeQueries([query], 1, history.getNewQueryOptions());
let folderNode = qres.root;
let closeOriginally = !folderNode.containerOpen;
if (closeOriginally)
folderNode.containerOpen = true;
for (let i = 0; i < folderNode.childCount; ++i) {
let node = folderNode.getChild(i);
if (PlacesUtils.nodeIsBookmark(node)) {
let it = {id: node.itemId,
title: node.title,
URI: node.uri };
if (m(it))
result.push(it);
}
}
if (closeOriginally)
folderNode.containerOpen = false;
return liberator.globalVariables.readcatlater_reverse ? result.reverse() : result;
} catch (e) { liberator.log(e); } }
function completer (context, arg) {
context.title = ['URL', 'Title'];
context.completions = RCL_Bookmarks(context.filter).
filter(function (it) it.id).
map(function (it) [it.URI, bookmarks.getItemTitle(it.id)]);
}
function removeItems (uri) {
var removed = false;
for each (let id in bookmarks.getBookmarkIdsForURI(makeURI(uri), {}))
if (folderId == bookmarks.getFolderIdForItem(id)) {
removed = true;
bookmarks.removeItem(id);
}
return removed;
}
/********************************************************************************
* Add Command
********************************************************************************/
commands.addUserCommand(
['readcatlater', 'rcl'],
'read cat later',
function (args) {
var res = addEntry(window.content.document, splitBySpaces(args.string));
if (res)
liberator.echo('"' + buffer.title + '" was added');
else
liberator.echo("this URI already exists!?");
},
{},
true
);
commands.addUserCommand(
['readcatnow', 'rcn'],
'read cat now',
function (arg) {
let opennum = arg['-number'];
if (opennum) {
liberator.log(typeof opennum)
let us = RCL_Bookmarks(arg.literalArg).reverse().splice(0, opennum).map(function (it) it.URI);
liberator.open(us, liberator.NEW_BACKGROUND_TAB);
if (!arg.bang) {
us.forEach(removeItems);
liberator.echo('[' + us + '] was removed.');
}
} else {
let uri = arg.string;
openURI(uri);
if (!arg.bang && removeItems(uri))
liberator.echo('"' + uri + '" was removed.');
}
},
{
literal: 0,
options: [ [['-number', '-n'], commands.OPTION_INT] ],
completer: completer
},
true
);
commands.addUserCommand(
['deletecatnow', 'dcn'],
'delete cat now',
function (arg) {
let uri = arg.string;
if (removeItems(uri))
liberator.echo('"' + uri + '" was removed.');
},
{
completer: completer
},
true
);
liberator.log('readcatlater.js loaded');
})();