You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

800 lines
31 KiB

  1. // ========================== KeySnail Init File =========================== //
  2. // この領域は, GUI により設定ファイルを生成した際にも引き継がれます
  3. // 特殊キー, キーバインド定義, フック, ブラックリスト以外のコードは, この中に書くようにして下さい
  4. // ========================================================================= //
  5. //{{%PRESERVE%
  6. // prompt.rows = 12;
  7. // prompt.useMigemo = false;
  8. // prompt.migemoMinWordLength = 2;
  9. // prompt.displayDelayTime = 300;
  10. // command.kill.killRingMax = 15;
  11. // command.kill.textLengthMax = 8192;
  12. //userscript.addLoadPath(".");
  13. //userscript.require("verticaltab.js");
  14. //////////////////////////////////////
  15. //// sitelocalkeymap
  16. var local = {};
  17. plugins.options["site_local_keymap.local_keymap"] = local;
  18. function fake(k, i) function () { key.feed(k, i); };
  19. function pass(k, i) [k, fake(k, i)];
  20. function ignore(k, i) [k, null];
  21. // ext.add("ext-name", function () {}, "ext description");
  22. // style.register("");
  23. // local["^http://"] = [['a', function(ev, arg){}],];
  24. ///////////////////////////////////////////
  25. //// firefox
  26. // style.register("#bookmarksPanel > hbox,#history-panel > hbox {display: none !important;} //#urlbar-container{max-width: 500px !important;}");
  27. util.setPrefs(
  28. {
  29. "browser.tabs.loadDivertedInBackground": true,
  30. "dom.disable_window_open_feature.location": false,
  31. "dom.max_script_run_time": 30,
  32. "browser.bookmarks.max_backups":0,
  33. "browser.urlbar.autocomplete.enabled":false,
  34. "browser.cache.memory.capacity":16384,
  35. "browser.sessionhistory.max_total_viewers":8,
  36. "browser.download.manager.closeWhenDone":true,
  37. "browser.download.useDownloadDir":false,
  38. "browser.tabs.closeWindowWithLastTab":false,
  39. "network.dns.disableIPv6":true,
  40. "browser.urlbar.trimURLs":false,
  41. "browser.fullscreen.autohide":false,
  42. "keyword.URL":"http://www.bing.com/search?q=",
  43. }
  44. );
  45. ///////////////////////////////////
  46. //検索エンジン
  47. plugins.options["search-url-list"] = [
  48. ["bing","http://bing.com/search?q=%q"],
  49. ["yatwitter search","http://yats-data.com/yats/search?query=%q"],
  50. ["twitter search","http://search.twitter.com/search?q=%q&lang=all"],
  51. ["tospy", "http://topsy.com/s?allow_lang=ja&q=%q"],
  52. ["2ch","http://2ch-ranking.net/search.php?q=%q&imp=and&order=time"],
  53. ["I\'m feelig lucky!","http://www.google.co.jp/search?q=%q&btnI=kudos"],
  54. ["uncyclopedia","http://ja.uncyclopedia.info/wiki/%q"],
  55. ["wikipedia","http://ja.wikipedia.org/wiki/%q"],
  56. ["nicovideo.jp","http://www.nicovideo.jp/search/%q"],
  57. ["alc","http://eow.alc.co.jp/%q/UTF-8/"],
  58. ["google map","http://maps.google.co.jp/maps?hl=ja&q=%q&um=1&ie=UTF-8&sa=N&tab=wl"],
  59. ["weblio","http://www.weblio.jp/content_find?query=%q"],
  60. ["shoutcast","http://www.shoutcast.com/Internet-Radio/%q"],
  61. ["10sr.posterous.com","http://www.google.com/search?q=%q&ie=UTF-8&oe=UTF-8&hl=ja&domains=10sr.posterous.com&sitesearch=10sr.posterous.com"],
  62. ["delicious 10sr","http://delicious.com/10sr?addtag=%q&setcount=50&opennew=1"],
  63. ["open raw","%r"],
  64. ];
  65. plugins.options["my-keysnail-bookmarks"] = [
  66. "twitter.com",
  67. ];
  68. //sitelocal
  69. //////////////////////////////////////////
  70. // 2ch chaika
  71. local["^http://127.0.0.1:8823/thread/"] = [
  72. ['k', function (ev, arg) {
  73. curl = window.content.location.href;
  74. kurl = curl.replace(/http:.*thread\/(.*\/).*/, "chaika://post/$1");
  75. window.content.location.href = kurl;
  76. }
  77. ],
  78. ];
  79. local["^http://w2.p2.2ch.net/p2/read.php"] = [
  80. ['k', function (ev, arg) {
  81. var url = window.content.location.href;
  82. var pt = /host=(.*?)&bbs=(.*?)&key=(.*?)&ls=/ ;
  83. var result = url.match(pt);
  84. var k = format("chaika://post/http://%s/test/read.cgi/%s/%s/", result[1], result[2], result[3]);
  85. window.content.location.href = k;
  86. }
  87. ],
  88. ];
  89. /////////////////////////////////////////
  90. // feedly用マップ
  91. local["^http://www.feedly.com/"] = [
  92. ['d', null],
  93. ['j', null],
  94. ['k', null],
  95. ['n', null],
  96. ['p', null],
  97. ['o', null],
  98. ['b', null],
  99. ['S', null],
  100. ['s', null],
  101. ['?', null],
  102. ['l', function (ev, arg) {window.content.location.href = "http://www.feedly.com/home#latest";}],
  103. ['r', null],
  104. ['x', function (ev, arg) {ev.target.dispatchEvent(key.stringToKeyEvent("g", true));}],
  105. [['t', 'p'], function (ev, arg) {ev.target.dispatchEvent(key.stringToKeyEvent("t", true));}],
  106. [['t', 'w'], function (ev, arg) {ext.exec("twitter-client-tweet", arg, ev);}],
  107. ];
  108. /////////////////////////////////////////
  109. //nicovideo用
  110. local["http://(www|tw|es|de|)\.nicovideo\.jp\/(watch|playlist)/*"] = [
  111. ["i", function (ev, arg) { ext.exec("nicoinfo", arg); }],
  112. ["p", function (ev, arg) { ext.exec("nicopause", arg); }],
  113. ["o", function (ev, arg) { ext.exec("nicommentvisible", arg); }],
  114. ["m", function (ev, arg) { ext.exec("nicomute", arg); }],
  115. [".", function (ev, arg) { ext.exec("nicovolumeIncrement", arg); }],
  116. [",", function (ev, arg) { ext.exec("nicovolumeDecrement", arg); }],
  117. ['f', function (ev, arg) {
  118. curl = window.content.location.href;
  119. kurl = curl.replace(/nicovideo.jp/, "nicovideofire.jp");
  120. window.content.location.href = kurl;
  121. }
  122. ],
  123. ];
  124. /////////////////////////////////////////
  125. // tumblr/dashboard
  126. local["^http://www.tumblr.com/dashboard"] = [
  127. // ["C-<left>", function (ev, arg) {gBrowser.mTabContainer.advanceSelectedTab(-1, true); }],
  128. // ["C-<right>", function (ev, arg) {gBrowser.mTabContainer.advanceSelectedTab(1, true); }],
  129. ["<left>", function (ev, arg) {window.content.location.href = "http://www.tumblr.com/dashboard"; }],
  130. // ["<right>", null],
  131. ["J", function (ev, arg) {
  132. if (window.loadURI) {
  133. loadURI("javascript:(function(){b=20;s=100;t=document.getElementById('next_page_link').href.split('/')[5];max=t.substr(0,t.length-5);min=max-s;i=Math.floor(Math.random()*(max-min)+min);u=(i<b)?'http://www.tumblr.com/dashboard':'http://www.tumblr.com/dashboard/2/'+i+'00000';window.content.location.href=u;}())");
  134. }
  135. }],
  136. ];
  137. ///////////////////////////////////
  138. //tanythhing用
  139. plugins.options["tanything_opt.keymap"] = {
  140. "\\" : "prompt-cancel",
  141. "j" : "prompt-next-completion",
  142. "k" : "prompt-previous-completion",
  143. "o" : "localOpen",
  144. ":" : "localClose",
  145. "L" : "localMovetoend"
  146. };
  147. //////////////////////////////////////////
  148. // yatc
  149. style.register("#keysnail-twitter-client-container{ display:none !important; }");
  150. plugins.options["twitter_client.popup_new_statuses"] = false;
  151. plugins.options["twitter_client.automatically_begin"] = false;
  152. plugins.options["twitter_client.automatically_begin_list"] = false;
  153. plugins.options["twitter_client.timeline_count_beginning"] = 0;
  154. plugins.options["twitter_client.timeline_count_every_updates"] = 0;
  155. plugins.options["twitter_client.tweet_keymap"] = {
  156. "C-RET" : "prompt-decide",
  157. "RET" : ""
  158. };
  159. plugins.options["twitter_client.jmp_id"] = "10sr";
  160. plugins.options["twitter_client.jmp_key"] = "R_c51f889a77cb4b4e993ed868f65083f5";
  161. plugins.options["twitter_client.use_jmp"] = true;
  162. ////////////////////////////////////////////
  163. // エクステ
  164. ext.add('auto-install-plugins', function(ev, arg){
  165. var urls = [
  166. 'https://github.com/mooz/keysnail/raw/master/plugins/yet-another-twitter-client-keysnail.ks.js',
  167. 'https://github.com/mooz/keysnail/raw/master/plugins/site-local-keymap.ks.js',
  168. 'https://github.com/azu/KeySnail-Plugins/raw/master/JSReference/js-referrence.ks.js',
  169. 'https://github.com/tkosaka/keysnail-plugin/raw/master/nicontroller.ks.js',
  170. 'https://raw.github.com/10sr/keysnail-plugin/master/shiitake.ks.js',
  171. ];
  172. function inst(a){
  173. if(a.length == 0){
  174. display.showPopup("auto-install-plugins", "All installation finished.");
  175. }else{
  176. var url = a.shift();
  177. var path = userscript.pluginDir + userscript.directoryDelimiter + url.match(/[^/]+$/)[0];
  178. if(plugins.context[path] === undefined){
  179. userscript.installPluginFromURL(url, function(){inst(a);});
  180. }else{
  181. inst(a);
  182. }
  183. }
  184. }
  185. inst(urls);
  186. }, 'Install plugins automatically if not installed yet.');
  187. ext.add('put-aside-this-page', function (ev, arg) {
  188. var n = gBrowser.mCurrentTab._tPos;
  189. gBrowser.moveTabTo(gBrowser.mCurrentTab, 0);
  190. if (n != 0) {
  191. gBrowser.selectedTab = gBrowser.mTabs[n];
  192. }
  193. }, 'put aside this page');
  194. ext.add('send-escape', function (ev, arg) {
  195. ev.target.dispatchEvent(key.stringToKeyEvent("ESC", true));
  196. }, 'escape');
  197. ext.add("open-hatebu-comment", function (ev, arg) {
  198. if (window.loadURI) {
  199. loadURI("javascript:location.href='http://b.hatena.ne.jp/entry?mode=more&url='+escape(location.href);");
  200. }
  201. }, 'hatebu');
  202. ext.add("fullscreen-page",function (ev) {
  203. getBrowser().selectedTab = getBrowser().addTab("http://home.tiscali.nl/annejan/swf/timeline.swf");
  204. BrowserFullScreen();
  205. }, "fullscreen page");
  206. ext.add("focus-on-content", function(){
  207. let(elem = document.commandDispatcher.focusedElement) elem && elem.blur();
  208. gBrowser.focus();
  209. content.focus();
  210. }, "forcus on content");
  211. ext.add("hide-sidebar", function(){
  212. var sidebarBox = document.getElementById("sidebar-box");
  213. if (!sidebarBox.hidden) {
  214. toggleSidebar(sidebarBox.getAttribute("sidebarcommand"));
  215. }
  216. }, "hide-sidebar");
  217. ext.add("close-and-next-tab", function (ev, arg) {
  218. var n = gBrowser.mCurrentTab._tPos;
  219. BrowserCloseTabOrWindow();
  220. gBrowser.selectedTab = gBrowser.mTabs[n];
  221. }, "close and focus to next tab");
  222. /////////////////////////////////////
  223. // google itranslate
  224. (function(){
  225. let targetLang = "ja"; // target lang to translate into
  226. let alternativeLang = "en"; // if given word is in targetLang, use this instead as a target lang
  227. function translate(word, target, next) {
  228. next("", "", " getting...");
  229. const base = "https://www.googleapis.com/language/translate/v2?key=%s&q=%s&target=%s";
  230. const apikey = "AIzaSyBq48p8NhFgaJ1DfUJ5ltbwLxeXpjEL86A";
  231. let ep = util.format(base, apikey, encodeURIComponent(word), target);
  232. util.httpGet(ep, false, function (res) {
  233. if (res.status === 200) {
  234. let json = decodeJSON(res.responseText);
  235. let srclang = json.data.translations[0].detectedSourceLanguage;
  236. if (target == srclang) {
  237. lookupword(word, alternativeLang);
  238. } else {
  239. let result = json.data.translations[0].translatedText;
  240. next(srclang, target, result);
  241. }
  242. } else {
  243. next("", "", "ERROR!");
  244. }
  245. });
  246. };
  247. function echo(srclang, from, tglang, to){
  248. display.echoStatusBar(srclang + " : " + from + " -> " + tglang + " : " + to);
  249. };
  250. function decodeJSON(json) {
  251. return util.safeEval("(" + json + ")");
  252. };
  253. function lookupword(word, target){
  254. translate(word, target, function (src, tg, translated) {
  255. echo(src, word, tg, translated);
  256. });
  257. };
  258. function read (aInitialInput) {
  259. let prevText = "";
  260. prompt.reader({
  261. message : "word or sentence to translate:",
  262. initialinput : aInitialInput,
  263. onChange: function (arg) {
  264. let word = arg.textbox.value;
  265. if (word !== prevText) {
  266. prevText = word;
  267. lookupword(word, targetLang);
  268. }
  269. },
  270. callback: function (s){},
  271. });
  272. };
  273. ext.add("google-itranslate",function(){read(content.document.getSelection() || "");},"google itranslate");
  274. })();
  275. //////////////////////////////////////
  276. //
  277. ext.add("restart-firefox-add-menu", function(){
  278. const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
  279. var cmdelm = document.createElementNS(XUL_NS, "command");
  280. cmdelm.setAttribute("id", "my_cmd_restartFirefoxKs")
  281. cmdelm.setAttribute("oncommand", "ext.exec('restart-firefox');");
  282. var commandset = document.getElementById("mainCommandSet");
  283. // menu.insertBefore(elm, menu.getElementById("menu_FileQuitItem"));
  284. commandset.appendChild(cmdelm);
  285. var menuelm = document.createElementNS(XUL_NS, "menuitem");
  286. menuelm.setAttribute("label", "Restart Firefox");
  287. menuelm.setAttribute("id", "my_menu_restartFirefoxKs")
  288. menuelm.setAttribute("command", "my_cmd_restartFirefoxKs");
  289. var menu = document.getElementById("menu_FilePopup");
  290. // menu.insertBefore(elm, menu.getElementById("menu_FileQuitItem"));
  291. menu.appendChild(menuelm);
  292. }, "add restart firefox menu");
  293. //////////////////////////////////////
  294. // restart firefox
  295. // http://keysnail.g.hatena.ne.jp/Shinnya/20100723/1279878815
  296. ext.add("restart-firefox",function (ev) {
  297. const nsIAppStartup = Components.interfaces.nsIAppStartup;
  298. // Notify all windows that an application quit has been requested.
  299. var os = Components.classes["@mozilla.org/observer-service;1"]
  300. .getService(Components.interfaces.nsIObserverService);
  301. var cancelQuit = Components.classes["@mozilla.org/supports-PRBool;1"]
  302. .createInstance(Components.interfaces.nsISupportsPRBool);
  303. os.notifyObservers(cancelQuit, "quit-application-requested", null);
  304. // Something aborted the quit process.
  305. if (cancelQuit.data)
  306. return;
  307. // Notify all windows that an application quit has been granted.
  308. os.notifyObservers(null, "quit-application-granted", null);
  309. // Enumerate all windows and call shutdown handlers
  310. var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
  311. .getService(Components.interfaces.nsIWindowMediator);
  312. var windows = wm.getEnumerator(null);
  313. while (windows.hasMoreElements()) {
  314. var win = windows.getNext();
  315. if (("tryToClose" in win) && !win.tryToClose())
  316. return;
  317. }
  318. Components.classes["@mozilla.org/toolkit/app-startup;1"].getService(nsIAppStartup)
  319. .quit(nsIAppStartup.eRestart | nsIAppStartup.eAttemptQuit);
  320. }, "restart firefox");
  321. /////////////////////////////////////////
  322. // copy feed url
  323. ext.add("copy-url", function () {
  324. const doc = content.document;
  325. let feeds = [[e.getAttribute("title"), e.getAttribute("href")]
  326. for ([, e] in Iterator(doc.querySelectorAll(['link[type="application/rss+xml"]',
  327. 'link[type="application/atom+xml"]'])))];
  328. var uh = window.content.location.href.replace(/(.*?\/\/[^/]*)(\/.*)?/,"$1");
  329. for (i = 0; i < feeds.length; i++)
  330. if ( feeds[i][1].substr(0,1) == "/" ) feeds[i][1] = uh + feeds[i][1];
  331. feeds.unshift([window.content.document.title,window.content.location.href]);
  332. prompt.selector(
  333. {
  334. message : "Select Feed",
  335. collection : feeds,
  336. callback : function (i) {
  337. if (i >= 0)
  338. command.setClipboardText(feeds[i][1]);
  339. }
  340. }
  341. );
  342. }, "Copy url or feed url of current page");
  343. ///////////////////////////////////////
  344. // 評価しちゃうっぽい とりあえずこんな感じで
  345. ext.add("keysnail-setting-menu",function(){
  346. var settingmenulist = [["keysnail setting dialogue", function(){return function(){KeySnail.openPreference();};}],
  347. ["firefox addon manager", function(){return function(){BrowserOpenAddonsMgr();};}],
  348. ["reload .keysnail.js", function(){return function() {userscript.reload();};}],
  349. // ["check for plugins update", function(){return function(){ext.exec("check-for-plugins-update");};}],
  350. ["restart firefox", function(){return function(){ext.exec("restart-firefox");};}],
  351. ];
  352. prompt.selector(
  353. {
  354. message : "open setting dialog",
  355. collection : settingmenulist,
  356. callback : function (i) { settingmenulist[i][1]()(); },
  357. });
  358. },"open keysnail setting menu");
  359. // //////////////////////////
  360. // //プラグイン一括アップデート
  361. // ext.add("check-for-plugins-update", function () {
  362. // [p for (p in plugins.context)].forEach(function (p) { try { userscript.updatePlugin(p); } catch(e) {} });
  363. // }, "Check for all plugin's update");
  364. ////////////////////////
  365. //マルチプルタブハンドラ
  366. ext.add("multiple-tab-handler-close-selected-and-current-tabs", function () {
  367. BrowserCloseTabOrWindow();
  368. // if (MultipleTabService) {
  369. // //BrowserCloseTabOrWindow();
  370. // //MultipleTabService.setSelection(gBrowser.mCurrentTab, true);
  371. MultipleTabService.closeTabs(MultipleTabService.getSelectedTabs());
  372. // } else {
  373. // BrowserCloseTabOrWindow();}
  374. }, '選択タブと現在のタブを閉じる');
  375. ext.add("if-mth-exist", function() {
  376. if (MultipleTabService != undefined) display.echoStatusBar("true");
  377. },'if mth exist');
  378. ////////////////////////
  379. // instapaper
  380. ext.add("instapaper-add-this-page-and-close",function(){
  381. var url = window.content.location.href;
  382. var title = window.content.document.title;
  383. var tab = gBrowser.selectedTab;
  384. var username = "8slashes+instapaper@gmail.com";
  385. var password = "";
  386. var passwordManager = (Cc["@mozilla.org/login-manager;1"].getService(Ci.nsILoginManager));
  387. var logins = passwordManager.findLogins({}, "http://www.instapaper.com", "", null);
  388. for (var i = 0; i < logins.length; i++) {
  389. if (logins[i].username == username) {
  390. password = logins[i].password;
  391. break;
  392. }
  393. }
  394. var comment = "";
  395. // prompt.read("Instapaper comment:",function(cm){
  396. // if(cm){comment = cm;}
  397. // });
  398. display.echoStatusBar("Instapaper: adding " + url + "...");
  399. util.httpPost("https://www.instapaper.com/api/add",
  400. {"username" : encodeURIComponent(username),
  401. "password" : password,
  402. "url" : encodeURIComponent(url),
  403. "title" : encodeURIComponent(title),},
  404. function (xhr) {
  405. display.echoStatusBar(xhr.status);
  406. if (xhr.readyState == 4 && xhr.status == 201) {
  407. // var title = decodeURIComponent(xhr.getResponseHeader("X-Instapaper-Title")); //超文字化けする
  408. try {
  409. Components.classes['@mozilla.org/alerts-service;1'].
  410. getService(Components.interfaces.nsIAlertsService).
  411. showAlertNotification(null, "Instapaper", "Page " + title + " added successfully", false, '', null);
  412. } catch(e) {
  413. // prevents runtime error on platforms that don't implement nsIAlertsService
  414. }
  415. display.echoStatusBar("Instapaper: adding " + url + "...done.");
  416. gBrowser.removeTab(tab);
  417. } else{
  418. display.echoStatusBar("Instapaper: Something wrong has happended!");
  419. if (window.loadURI) {
  420. loadURI("javascript:function%20iprl5(){var%20d=document,z=d.createElement('scr'+'ipt'),b=d.body,l=d.location;try{if(!b)throw(0);d.title='(Saving...)%20'+d.title;z.setAttribute('src',l.protocol+'//www.instapaper.com/j/mt8YO6Cuosmf?u='+encodeURIComponent(l.href)+'&t='+(new%20Date().getTime()));b.appendChild(z);}catch(e){alert('Please%20wait%20until%20the%20page%20has%20loaded.');}}iprl5();void(0)");
  421. }
  422. }
  423. });
  424. },'instapaper add page and close tab when done without error.');
  425. ////////////////////////
  426. //検索
  427. ext.add("query-then-engine", function () {
  428. prompt.reader({message : "Search Word?:",
  429. callback : function (q) {
  430. if (q) {
  431. prompt.selector({ message : "search \"" + q + "\" with?",
  432. collection : plugins.options["search-url-list"],
  433. width : [20,80],
  434. callback : function (i) { getBrowser().selectedTab = getBrowser().addTab(plugins.options["search-url-list"][i][1].replace("%r",q).replace("%q",encodeURIComponent(q))); },
  435. });
  436. };
  437. },
  438. initialInput : content.document.getSelection() || "",
  439. });
  440. }, "enter search word and then select engine");
  441. ///////////////////////
  442. //diggler
  443. ext.add("dig-url", function () {
  444. var url = window.content.location.href;
  445. var nsurl = [];
  446. var pname = "";
  447. var ssurl = [];
  448. var durl = [];
  449. nsurl = url.split("#");
  450. var pname = nsurl[1];
  451. ssurl = nsurl[0].split("/");
  452. durl[0] = ssurl[0] + "//" + ssurl[2];
  453. ssurl.splice(0,3);
  454. for (var i = 0; i < ssurl.length; i++){
  455. var durlsaved = durl[0];
  456. durl.unshift(durlsaved + "/" + ssurl[i]);
  457. };
  458. if (pname) {
  459. var durlfull = durl[0] + "#"+ pname;
  460. durl.unshift(durlfull);
  461. };
  462. prompt.selector({ message : "dig " + url,
  463. collection : durl,
  464. callback : function (i) { window.content.location.href = durl[i]; },
  465. });
  466. },"keysnail diggler ");
  467. /////////////////////////////////////
  468. // 閉じたタブリスト
  469. ext.add("list-closed-tabs", function () {
  470. const fav = "chrome://mozapps/skin/places/defaultFavicon.png";
  471. var ss = Cc["@mozilla.org/browser/sessionstore;1"].getService(Ci.nsISessionStore);
  472. var json = Cc["@mozilla.org/dom/json;1"].createInstance(Ci.nsIJSON);
  473. var closedTabs = [[tab.image || fav, tab.title, tab.url] for each (tab in json.decode(ss.getClosedTabData(window)))];
  474. if (!closedTabs.length)
  475. return void display.echoStatusBar("最近閉じたタブが見つかりませんでした", 2000);
  476. prompt.selector(
  477. {
  478. message : "select tab to undo:",
  479. collection : closedTabs,
  480. flags : [ICON | IGNORE, 0, 0],
  481. callback : function (i) { if (i >= 0) window.undoCloseTab(i); }
  482. });
  483. }, "List closed tabs");
  484. ext.add("echo-closed-tabs", function () {
  485. var ss = Cc["@mozilla.org/browser/sessionstore;1"].getService(Ci.nsISessionStore);
  486. var json = Cc["@mozilla.org/dom/json;1"].createInstance(Ci.nsIJSON);
  487. // var closedTabs = [[tab.image || fav, tab.title, tab.url] for each (tab in json.decode(ss.getClosedTabData(window)))];
  488. var lasttab = json.decode(ss.getClosedTabData(window))[0];
  489. dump = ""
  490. for (var i in lasttab) { dump += lasttab[i] + "\n"; }
  491. confirm(dump);
  492. }, "List closed tabs");
  493. //}}%PRESERVE%
  494. // ========================================================================= //
  495. // ========================= Special key settings ========================== //
  496. key.quitKey = "<delete>";
  497. key.helpKey = "C-h";
  498. key.escapeKey = "C-q";
  499. key.macroStartKey = "";
  500. key.macroEndKey = "";
  501. key.universalArgumentKey = "C-u";
  502. key.negativeArgument1Key = "C--";
  503. key.negativeArgument2Key = "C-M--";
  504. key.negativeArgument3Key = "M--";
  505. key.suspendKey = "Not defined";
  506. // ================================= Hooks ================================= //
  507. hook.setHook('KeySnailInitialized', function () {
  508. ext.exec("shiitake-enable-style");
  509. });
  510. hook.setHook('KeyBoardQuit', function (aEvent) {
  511. ext.exec("hide-sidebar");
  512. let(elem = document.commandDispatcher.focusedElement) elem && elem.blur();
  513. gBrowser.focus();
  514. content.focus();
  515. command.closeFindBar();
  516. if (util.isCaretEnabled()) {
  517. command.resetMark(aEvent);
  518. } else {
  519. goDoCommand("cmd_selectNone");
  520. }
  521. key.generateKey(aEvent.originalTarget, KeyEvent.DOM_VK_ESCAPE, true);
  522. });
  523. hook.addToHook('KeyBoardQuit', function (aEvent) {
  524. });
  525. hook.setHook('Unload', function () {
  526. util.getBrowserWindows().some(function (win) {
  527. if (win === window) {
  528. return false;
  529. }
  530. const ks = win.KeySnail;
  531. share.pluginUpdater = ks.getPluginUpdater(share.pluginUpdater.pluginsWithUpdate);
  532. ks.setUpPluginUpdaterDelegator();
  533. return true;
  534. });
  535. });
  536. // ============================= Key bindings ============================== //
  537. key.setGlobalKey('C-<right>', function () {
  538. gBrowser.mTabContainer.advanceSelectedTab(1, true);
  539. }, 'ひとつ右のタブへ');
  540. key.setGlobalKey('C-<left>', function () {
  541. gBrowser.mTabContainer.advanceSelectedTab(-1, true);
  542. }, 'ひとつ左のタブへ');
  543. key.setGlobalKey('C-<up>', function () {
  544. var browser = getBrowser();
  545. if (browser.mCurrentTab.previousSibling) {
  546. browser.moveTabTo(browser.mCurrentTab, browser.mCurrentTab._tPos - 1);
  547. } else {
  548. browser.moveTabTo(browser.mCurrentTab, browser.mTabContainer.childNodes.length - 1);
  549. }
  550. }, '選択中のタブを右へ');
  551. key.setGlobalKey('C-<down>', function () {
  552. var browser = getBrowser();
  553. if (browser.mCurrentTab.nextSibling) {
  554. browser.moveTabTo(browser.mCurrentTab, browser.mCurrentTab._tPos + 1);
  555. } else {
  556. browser.moveTabTo(browser.mCurrentTab, 0);
  557. }
  558. }, '選択中のタブを左へ');
  559. key.setGlobalKey('M-:', function (ev) {
  560. command.interpreter();
  561. }, 'JavaScript のコードを評価');
  562. key.setViewKey('', function (ev, arg) {
  563. var n = gBrowser.mCurrentTab._tPos;
  564. BrowserCloseTabOrWindow();
  565. gBrowser.selectedTab = gBrowser.mTabs[n];
  566. }, '閉じて次のタブ');
  567. key.setViewKey('x', function (aEvent, aArg) {
  568. ext.select(aArg, aEvent);
  569. }, 'エクステ一覧');
  570. key.setViewKey(['t', 'm'], function (ev, arg) {
  571. if (window.loadURI) {
  572. loadURI("javascript:window.location='http://api.tweetmeme.com/visit?url='+window.location;");
  573. }
  574. }, 'open with tweetmeme');
  575. key.setViewKey(['t', 'w'], function (ev, arg) {
  576. ext.exec("twitter-client-tweet", arg, ev);
  577. }, 'つぶやく', true);
  578. key.setViewKey(['t', 'p'], function (ev, arg) {
  579. ext.exec("twitter-client-tweet-this-page", arg, ev);
  580. }, 'このページのタイトルと URL を使ってつぶやく', true);
  581. key.setViewKey('u', function () {
  582. undoCloseTab();
  583. }, '閉じたタブを元に戻す');
  584. key.setViewKey('g', function () {
  585. goDoCommand("cmd_scrollTop");
  586. }, 'ページ先頭へ移動');
  587. key.setViewKey('G', function () {
  588. goDoCommand("cmd_scrollBottom");
  589. }, 'ページ末尾へ移動');
  590. key.setViewKey('r', function (aEvent) {
  591. BrowserReload();
  592. }, '再読み込み');
  593. key.setViewKey('m', function (ev, arg) {
  594. _fi.toogle();
  595. }, 'fetchimiをトグル');
  596. key.setViewKey('d', function (ev, arg) {
  597. if (window.loadURI) {
  598. loadURI("javascript:(function(){f='http://www.delicious.com/save?url='+encodeURIComponent(window.location.href)+'&title='+encodeURIComponent(document.title)+'&notes='+encodeURIComponent(''+(window.getSelection?window.getSelection():document.getSelection?document.getSelection():document.selection.createRange().text))+'&v=6&';a=function(){if(!window.open(f+'noui=1&jump=doclose','deliciousuiv6','location=1,links=0,scrollbars=0,toolbar=0,width=550,height=585'))location.href=f+'jump=yes'};if(/Firefox/.test(navigator.userAgent)){setTimeout(a,0)}else{a()}})()");
  599. }
  600. }, 'deliciousでブックマーク');
  601. key.setViewKey('p', function (ev, arg) {
  602. if (window.loadURI) {
  603. loadURI("javascript:var%20b=document.body;var%20POSTEROUS___bookmarklet_domain='http://posterous.com';if(b&&!document.xmlVersion){void(z=document.createElement('script'));void(z.type='text/javascript');void(z.src='http://posterous.com/javascripts/bookmarklet2.js');void(b.appendChild(z));}else{}");
  604. }
  605. }, 'posterousに投稿');
  606. key.setViewKey('SPC', function (ev, arg) {
  607. MultipleTabService.toggleSelection(gBrowser.mCurrentTab);
  608. gBrowser.mTabContainer.advanceSelectedTab(1, true);
  609. }, 'タブの選択をトグルして次のタブ');
  610. key.setViewKey('S-SPC', function (ev, arg) {
  611. MultipleTabService.toggleSelection(gBrowser.selectedTab);
  612. gBrowser.mTabContainer.advanceSelectedTab(-1, true);
  613. }, 'タブの選択をトグルして前のタブ');
  614. key.setViewKey('z', function (ev, arg) {
  615. ext.exec("keysnail-setting-menu", arg, ev);
  616. }, 'open keysnail setting menu', true);
  617. key.setViewKey('T', function (ev, arg) {
  618. ext.exec("google-itranslate", arg, ev);
  619. }, 'google itranslate', true);
  620. key.setViewKey('C-SPC', function (ev, arg) {
  621. MultipleTabService.toggleSelection(gBrowser.selectedTab);
  622. }, 'タブの選択をトグル');
  623. key.setViewKey('U', function (ev, arg) {
  624. ext.exec("list-closed-tabs", arg, ev);
  625. }, 'List closed tabs', true);
  626. key.setViewKey('e', function () {
  627. command.focusElement(command.elementsRetrieverTextarea, 0);
  628. }, '最初のインプットエリアへフォーカス', true);
  629. key.setViewKey('S', function (ev, arg) {
  630. if (window.loadURI) {
  631. loadURI("javascript:var%20b=document.body;var%20GR________bookmarklet_domain='https://www.google.com';if(b&&!document.xmlVersion){void(z=document.createElement('script'));void(z.src='https://www.google.com/reader/ui/link-bookmarklet.js');void(b.appendChild(z));}else{}");
  632. }
  633. }, 'google reader share');
  634. key.setViewKey('!', function (ev, arg) {
  635. shell.input();
  636. }, 'Command system');
  637. key.setViewKey('b', function (ev, arg) {
  638. BarTap.putOnTap(gBrowser.mCurrentTab, gBrowser);
  639. }, 'bartab put on tab');
  640. key.setViewKey('R', function () {
  641. BrowserReloadSkipCache();
  642. }, '更新(キャッシュを無視)');
  643. key.setViewKey('<backspace>', function () {
  644. BrowserBack();
  645. }, '戻る');
  646. key.setViewKey('S-<backspace>', function () {
  647. BrowserForward();
  648. }, '進む');
  649. key.setViewKey('q', function (ev, arg) {
  650. ext.exec("query-then-engine", arg, ev);
  651. }, 'enter search word and then select engine', true);
  652. key.setViewKey('D', function (ev, arg) {
  653. ext.exec("dig-url", arg, ev);
  654. }, 'keysnail diggler ', true);
  655. key.setViewKey('/', function () {
  656. command.iSearchForward();
  657. }, 'インクリメンタル検索', true);
  658. key.setViewKey('?', function (ev) {
  659. command.iSearchForwardKs(ev);
  660. }, 'Emacs ライクなインクリメンタル検索', true);
  661. key.setViewKey('a', function (ev, arg) {
  662. allTabs.open();
  663. }, 'alltabs.open');
  664. key.setViewKey('C', function (ev, arg) {
  665. ext.exec("copy-url", arg, ev);
  666. }, '選択タブと現在のタブを閉じる', true);
  667. key.setViewKey('I', function (ev, arg) {
  668. ext.exec("instapaper-add-this-page-and-close", arg, ev);
  669. }, 'instapaper add page', true);
  670. key.setViewKey('<left>', function (ev) {
  671. goDoCommand("cmd_scrollPageUp");
  672. }, '一画面分スクロールアップ');
  673. key.setViewKey('<right>', function (ev) {
  674. goDoCommand("cmd_scrollPageDown");
  675. }, '一画面スクロールダウン');
  676. key.setViewKey([['<prior>'], ['<next>']], function (ev, arg) {
  677. return;
  678. }, 'ignore');
  679. key.setViewKey(':', function (ev, arg) {
  680. return !document.getElementById("keysnail-prompt").hidden &&
  681. document.getElementById("keysnail-prompt-textbox").focus();
  682. }, 'KeySnail のプロンプトへフォーカス', true);
  683. key.setViewKey('H', function (ev, arg) {
  684. ext.exec("open-hatebu-comment", arg, ev);
  685. }, 'hatebu', true);
  686. key.setViewKey('l', function (ev) {
  687. command.focusToById("urlbar");
  688. }, 'ロケーションバーへフォーカス', true);
  689. key.setEditKey('C-<tab>', function (ev) {
  690. command.walkInputElement(command.elementsRetrieverTextarea, true, true);
  691. }, '次のテキストエリアへフォーカス');
  692. key.setViewKey('0', function (ev) {
  693. BrowserCloseTabOrWindow();
  694. }, 'タブ / ウィンドウを閉じる');