/** * ==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) + "
" + ""; for (var i = 0; i < marks.length; i++) { list += "" + "" + "" + "" + "" + ""; } list += "
marklinecolfile
" + marks[i][0] + "" + Math.round(marks[i][2] * 100) + "%" + Math.round(marks[i][3] * 100) + "%" + liberator.modules.util.escapeHTML(marks[i][1]) + "
"; return list; } else { var list = ":" + liberator.modules.util.escapeHTML(liberator.modules.commandline.command) + "
" + ""; for (var i = 0; i < marks.length; i++) { list += ""; } list += "
QuickMarkURL
" + marks[i][0] + "" + liberator.modules.util.escapeHTML(marks[i][1]) + "
"; 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) {
/* {{{
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>Garbage Finder</name>
  <name lang="ja">ゴミ探し</name>
  <description>Find the taints in global(window object)</description>
  <description lang="ja">グローバル(window オブジェクト)の汚染を調べる</description>
  <version>1.0.2</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>
  <minVersion>2.0pre</minVersion>
  <maxVersion>2.0pre</maxVersion>
  <updateURL>http://svn.coderepos.org/share/lang/javascript/vimperator-plugins/trunk/garbage_finder.js</updateURL>
  <detail><![CDATA[
    == Commands ==
      :garbages:
        Displays (removed|appended) variables.
  ]]></detail>
  <detail lang="ja"><![CDATA[
    前回(Firefox)起動時の window オブジェクトにおける変数と現在のそれの差分を取ります
    インストールした拡張がグローバルを汚していないか調べるのに便利かもしれません
    == Commands ==
      :garbages:
        (追加|削除)された変数を表示
  ]]></detail>
</VimperatorPlugin>;
// }}}

(function () {

  const STORAGE_NAME = 'plugin-garbage-finder';
  const IGNORES = (let (gv = liberator.globalVariables.garbage_finder_ignore)
                      gv === undefined ? 'DownloadUtils PluralForm' : gv).split(/\s+/);

  function Somali (n)
    let(V,[l,s,j,t]=liberator.eval('[loadPref,savePref,json,Object]',storage.newObject))
      ({load:function(d)let(v=l(n,true,t))(V=v?v.value:d),
        save:function(v)s({store:true,name:n,serial:j.encode({value:v===undefined?V:v})})});

  function vars () {
    let result = [];
    for (let name in window)
      result.push(name);
    return result.sort();
  }

  function has (ary, v)
    ary.some(function (it) v == it);

  function diff (oldList, newList) {
    function sub (n, o)
      n.filter(function (it) !has(o, it));
    return {
      appended: ignore(sub(newList, oldList)),
      removed: ignore(sub(oldList, newList))
    };
  }

  function id (value)
    value;

  function ignore (ary)
    ary.filter(function (it) !has(IGNORES, it));


  let store = new Somali(STORAGE_NAME);
  let prevVars = store.load(vars());

  function save ()
    store.save(vars());

  autocommands.add(
    'VimperatorEnter',
    /.*/,
    function () setTimeout(save, 2000)
  );

  commands.addUserCommand(
    ['garbages'],
    'Display garbages',
    function (args) {
      function makeLI (list) {
        if (list.length) {
          let result = <></>;
          list.forEach(function (it) (result += <li>{it}</li>));
          return <ol>{result}</ol>;
        }
        // XXX 駄目くさいけどめんどくさいので…
        return <ol>Nothing</ol>;
      }

      if (args.bang) {
        save();
        liberator.echo('Current variables was saved.');
      } else {
        let gs = diff(prevVars, vars());
        let as = makeLI(gs.appended), rs = makeLI(gs.removed);
        let output = <div><h1>Appended</h1><div>{as}</div><h1>Removed</h1><div>{rs}</div></div>;
        if (args['-clipboard']) {
          let cbOut = '';
          function pushLine (v, i)
            cbOut += '  ' + i + '.' + v + '\n';
          cbOut += 'Appended';
          gs.appended.forEach(pushLine);
          cbOut += 'Removed';
          gs.removed.forEach(pushLine);
          util.copyToClipboard(cbOut);
        }
        liberator.echo(output);
      }
    },
    {
      bang: true,
      options: [ [['-clipboard', '-c'], commands.OPTION_NOARG] ]
    },
    true
  );

})();

// vim:sw=2 ts=2 et si fdm=marker: