/* NEW BSD LICENSE {{{
Copyright (c) 2010, 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 =
feedSomeKeys 3
feedSomeKeys 3
feed some defined key events into the Web content
キーイベントをWebコンテンツ側に送る
1.7.2
anekos
new BSD License (Please read the source code comments of this plugin)
修正BSDライセンス (ソースコードのコメントを参照してください)
http://svn.coderepos.org/share/lang/javascript/vimperator-plugins/trunk/feedSomeKeys_3.js
2.3
2.3
_libly.js
||
command! -nargs=+ lazy autocmd VimperatorEnter .*
lazy fmaps -u='mail\.google\.com/mail' c / j k n p o u e x s r a # [ ] ? gi gs gt gd ga gc
lazy fmaps -u='mail\.google\.com/mail/.*/[0-9a-f]+$' c / j,n k,p n,j p,k o u e x s r a # [ ] ? gi gs gt gd ga gc
lazy fmaps -u='www\.google\.co\.jp/reader' -events=vkeypress j k n p m s v A r S N P X O gh ga gs gt gu u / ? J K
lazy fmaps -u='(fastladder|livedoor)\.com/reader' j k s a p o v c i,p z b < > q w e,g
lazy fmaps -u='https?://www\.rememberthemilk\.com/home/' j k m i c t ? d F,f G,g S,s L,l Y,y H,h M,m
lazy fmaps -u='http://code.google.com/p/vimperator-labs/issues/list' o j k
lazy fmaps -u='http://code.google.com/p/vimperator-labs/issues/detail' u
||<
]]>
||
command! -nargs=+ lazy autocmd VimperatorEnter .*
lazy fmaps -u='mail\.google\.com/mail' c / j k n p o u e x s r a # [ ] ? gi gs gt gd ga gc
lazy fmaps -u='mail\.google\.com/mail/.*/[0-9a-f]+$' c / j,n k,p n,j p,k o u e x s r a # [ ] ? gi gs gt gd ga gc
lazy fmaps -u='www\.google\.co\.jp/reader' -events=vkeypress j k n p m s v A r S N P X O gh ga gs gt gu u / ? J K
lazy fmaps -u='(fastladder|livedoor)\.com/reader' j k s a p o v c i,p z b < > q w e,g
lazy fmaps -u='https?://www\.rememberthemilk\.com/home/' j k m i c t ? d F,f G,g S,s L,l Y,y H,h M,m
lazy fmaps -u='http://code.google.com/p/vimperator-labs/issues/list' o j k
lazy fmaps -u='http://code.google.com/p/vimperator-labs/issues/detail' u
||<
]]>
;
// }}}
// INFO {{{
let INFO = <>
anekos
New BSD License
Feed key events directly into web contents.
-
:fmap
:fmap -events=eventnamelist -urls=urlpattern lhs rhs
Define one mapping.
If -urls=urlpattern is given,
the mappings becomes effective mappings only on the page specifed by urlpattern.
-
:fmaps
:fmaps -events=eventnamelist -urls=urlpattern mappingpair ....
Two or more mappings are defined at once.
mappingpair is a pair of key names separated by ",".
e.g. "<Leader><S-j>,j"
If -urls=urlpattern is given,
the mappings becomes effective mappings only on the page specifed by urlpattern.
-
:fmapc
:fmapc! urlpattern
Remove the mappings matched with urlpattern.
If "!" is given, remove all mappings.
-
:funmap
:funmap -urls=urlpattern lhs
Remove the mappings.
If you wish to remove url-local mappings, give -urls=urlpattern.
urlpattern
The value of urlpattern should be regular expression.
xpath
The XPath for a target element.
framenumber
The number of a target frame.
Refer the completion for this number.
eventnamelist
eventnamelist is a list of below values.
- keypress
- keydown
- keyup
- vkeypress
- vkeydown
- vkeyup
"v-" values use virtual key code.
The default value of this option is "keypress".
fmaps examples for .vimperatorrc
If you input directly these commands in vimperator commandline, remove the ":lazy".
:command! -nargs=+ lazy autocmd VimperatorEnter .* <args>
:lazy fmaps -u='mail\.google\.com/mail' c / j k n p o u e x s r a # [ ] ? gi gs gt gd ga gc
:lazy fmaps -u='mail\.google\.com/mail/.*/[0-9a-f]+$' c / j,n k,p n,j p,k o u e x s r a # [ ] ? gi gs gt gd ga gc
:lazy fmaps -u='www\.google\.co\.jp/reader' -events=vkeypress j k n p m s v A r S N P X O gh ga gs gt gu u / ? J K
:lazy fmaps -u='(fastladder|livedoor)\.com/reader' j k s a p o v c i,p <Space> <S-Space> z b < > q w e,g
:lazy fmaps -u='https?://www\.rememberthemilk\.com/home/' j k m i c t ? d F,f G,g S,s L,l Y,y H,h M,m <Del> <C-S-Left> <C-S-Right>
:lazy fmaps -u='http://code.google.com/p/vimperator-labs/issues/list' o j k
:lazy fmaps -u='http://code.google.com/p/vimperator-labs/issues/detail' u
anekos
New BSD License
Web コンテンツに直接キーイベントを送ります。
-
:fmap
:fmap -events=eventnamelist -xpath=xpath -frame=framenumber -urls=urlpattern lhs rhs
マッピングを一つ定義します。
-urls=urlpattern が与えられたとき、
そのマッピングは urlpattern で指定されたページのみで有効になります。
-
:fmaps
:fmaps -events=eventnamelist -xpath=xpath -frame=framenumber -urls=urlpattern mappingpair ....
一度に複数のマッピングを定義できます。
mappingpair は、"," で区切られたキー名の組です。
例: "<Leader><S-j>,j"
-urls=urlpattern が与えられたとき、
そのマッピングは urlpattern で指定されたページのみで有効になります。
-
:fmapc
:fmapc! urlpattern
urlpattern のマッピングを削除します。
"!" が与えられたときは、全てのマッピングが削除されます。
-
:funmap
:funmap -urls=urlpattern lhs
マッピングを削除します。
urlpattern 付きのマッピングを削除するときは、-urls を指定する必要があります。
urlpattern
urlpattern の値は正規表現でなければいけません。
eventnamelist
eventnamelist は以下の値のリストです。
- keypress
- keydown
- keyup
- vkeypress
- vkeydown
- vkeyup
"v-" のものは、仮想キーコードでイベントを発行します。
このオプションのデフォルト値は "keypress" です。
xpath
キーイベントを送るべき要素を指定するための XPath。
framenumber
キーイベントを送るべきフレームの番号。
番号は、補完を参考にしてください。
.vimperatorrc 用の fmaps サンプル
コマンドラインで直接に入力するときは、":lazy" を除いてください。
:command! -nargs=+ lazy autocmd VimperatorEnter .* <args>
:lazy fmaps -u='mail\.google\.com/mail' c / j k n p o u e x s r a # [ ] ? gi gs gt gd ga gc
:lazy fmaps -u='mail\.google\.com/mail/.*/[0-9a-f]+$' c / j,n k,p n,j p,k o u e x s r a # [ ] ? gi gs gt gd ga gc
:lazy fmaps -u='www\.google\.co\.jp/reader' -events=vkeypress j k n p m s v A r S N P X O gh ga gs gt gu u / ? J K
:lazy fmaps -u='(fastladder|livedoor)\.com/reader' j k s a p o v c i,p <Space> <S-Space> z b < > q w e,g
:lazy fmaps -u='https?://www\.rememberthemilk\.com/home/' j k m i c t ? d F,f G,g S,s L,l Y,y H,h M,m <Del> <C-S-Left> <C-S-Right>
:lazy fmaps -u='http://code.google.com/p/vimperator-labs/issues/list' o j k
:lazy fmaps -u='http://code.google.com/p/vimperator-labs/issues/detail' u
>;
// }}}
(function () {
const EVENTS = 'keypress keydown keyup'.split(/\s+/);
const EVENTS_WITH_V = EVENTS.concat(['v' + n for each (n in EVENTS)]);
const IGNORE_URLS = //;
const VKeys = {
'0': KeyEvent.DOM_VK_0,
'1': KeyEvent.DOM_VK_1,
'2': KeyEvent.DOM_VK_2,
'3': KeyEvent.DOM_VK_3,
'4': KeyEvent.DOM_VK_4,
'5': KeyEvent.DOM_VK_5,
'6': KeyEvent.DOM_VK_6,
'7': KeyEvent.DOM_VK_7,
'8': KeyEvent.DOM_VK_8,
'9': KeyEvent.DOM_VK_9,
';': KeyEvent.DOM_VK_SEMICOLON,
'=': KeyEvent.DOM_VK_EQUALS,
'a': KeyEvent.DOM_VK_A,
'b': KeyEvent.DOM_VK_B,
'c': KeyEvent.DOM_VK_C,
'd': KeyEvent.DOM_VK_D,
'e': KeyEvent.DOM_VK_E,
'f': KeyEvent.DOM_VK_F,
'g': KeyEvent.DOM_VK_G,
'h': KeyEvent.DOM_VK_H,
'i': KeyEvent.DOM_VK_I,
'j': KeyEvent.DOM_VK_J,
'k': KeyEvent.DOM_VK_K,
'l': KeyEvent.DOM_VK_L,
'm': KeyEvent.DOM_VK_M,
'n': KeyEvent.DOM_VK_N,
'o': KeyEvent.DOM_VK_O,
'p': KeyEvent.DOM_VK_P,
'q': KeyEvent.DOM_VK_Q,
'r': KeyEvent.DOM_VK_R,
's': KeyEvent.DOM_VK_S,
't': KeyEvent.DOM_VK_T,
'u': KeyEvent.DOM_VK_U,
'v': KeyEvent.DOM_VK_V,
'w': KeyEvent.DOM_VK_W,
'x': KeyEvent.DOM_VK_X,
'y': KeyEvent.DOM_VK_Y,
'z': KeyEvent.DOM_VK_Z,
'*': KeyEvent.DOM_VK_MULTIPLY,
'+': KeyEvent.DOM_VK_ADD,
'-': KeyEvent.DOM_VK_SUBTRACT,
',': KeyEvent.DOM_VK_COMMA,
'.': KeyEvent.DOM_VK_PERIOD,
'/': KeyEvent.DOM_VK_SLASH,
'?': KeyEvent.DOM_VK_SLASH,
'`': KeyEvent.DOM_VK_BACK_QUOTE,
'{': KeyEvent.DOM_VK_OPEN_BRACKET,
'\\': KeyEvent.DOM_VK_BACK_SLASH,
'}': KeyEvent.DOM_VK_CLOSE_BRACKET,
'\'': KeyEvent.DOM_VK_QUOTE
};
function id (v)
v;
function or (list, func)
(list.length && let ([head,] = list) (func(head) || or(list.slice(1), func)));
function getFrames () {
function bodyCheck (content)
(content.document.body.localName.toLowerCase() === 'body');
function get (content)
(bodyCheck(content) && result.push(content), Array.slice(content.frames).forEach(get));
let result = [];
get(content);
return result;
}
function fromXPath (doc, xpath) {
let result = util.evaluateXPath(xpath, doc);
return result.snapshotLength && result.snapshotItem(0);
}
function createEvent (eventName, event) {
let result = content.document.createEvent('KeyEvents');
result.initKeyEvent(
eventName,
true,
true,
content,
event.ctrlKey,
event.altKey,
event.shiftKey,
event.metaKey,
event.keyCode,
event.charCode
);
return result;
}
function virtualize (event) {
let cc = event.charCode;
if (/^[A-Z]$/.test(String.fromCharCode(cc)))
event.shiftKey = true;
event.keyCode = VKeys[String.fromCharCode(cc).toLowerCase()];
event.charCode = 0;
return event;
}
function feed (keys, eventNames, target) {
let _passAllKeys = modes.passAllKeys;
modes.passAllKeys = true;
modes.passNextKey = false;
for (let [, keyEvent] in Iterator(events.fromString(keys))) {
eventNames.forEach(function (eventName) {
let [, vkey, name] = eventName.match(/^(v)?(.+)$/);
if (vkey)
virtualize(keyEvent);
let event = createEvent(name, keyEvent);
target.dispatchEvent(event);
});
}
modes.passAllKeys = _passAllKeys;
}
function regexpValidator (expr) {
try {
RegExp(expr);
return true;
} catch (e) {}
return false;
}
function xpathValidator (expr) {
try {
document.evaluate(expr, document, null, null, null);
return true;
} catch (e) {}
return false;
}
function makeListValidator (list)
function (values)
(values && !values.some(function (value) !list.some(function (event) event === value)));
function findMappings ({all, filter, urls, ignoreUrls, not, result}) {
function match (map) {
let r = (
map.feedSomeKeys &&
(all ||
(!filter || filter === map.names[0]) &&
(ignoreUrls || urls === IGNORE_URLS || mappings._matchingUrlsTest(map, urls)))
);
if (result && r) {
if (typeof result.matched === 'number')
result.matched++;
else
result.matched = 1;
}
return !!r ^ !!not;
}
if (filter)
filter = mappings._expandLeader(filter);
if (urls)
urls = RegExp(urls);
return mappings._user[modes.NORMAL].filter(match);
}
function unmap (condition) {
condition.not = true;
mappings._user[modes.NORMAL] = findMappings(condition);
}
function list (condition) {
let maps = findMappings(condition);
let template = modules.template;
let list =
{
template.map(maps, function (map)
template.map(map.names, function (name)
{name} |
{map.feedSomeKeys.rhs} |
{map.matchingUrls ? map.matchingUrls : '[Global]'} |
))
}
;
if (list.*.length() == list.text().length()) {
liberator.echomsg("No mapping found");
return;
}
commandline.echo(list, commandline.HL_NORMAL, commandline.FORCE_MULTILINE);
}
function fmapCompleter (context, args) {
context.title = ['name', 'rhs & url'];
context.completions = [
[
{map.names[0]},
{map.feedSomeKeys.rhs}
{
args['-ignoreurls']
? <> for {map.matchingUrls ? map.matchingUrls : 'Global'}>
: ''
}
]
for each (map in findMappings({urls: args['-urls'], ignoreUrls: args['-ignoreurls']}))
];
}
function urlCompleter ({currentURL}) {
return function (context, args) {
let maps = findMappings({all: true});
let uniq = {};
let result = [
(uniq[map.matchingUrls] = 1, [map.matchingUrls.source, map.names])
for each (map in maps)
if (map.matchingUrls && !uniq[map.matchingUrls])
];
if (currentURL) {
result.unshift([util.escapeRegex(buffer.URL), 'Current URL']);
result.unshift([util.escapeRegex(content.document.domain), 'Current domain']);
}
return result;
};
}
function frameCompleter (context, args) {
return [
[i, frame.document.location]
for each ([i, frame] in Iterator(getFrames()))
];
}
'fmap fmaps'.split(/\s+/).forEach(function (cmd) {
let multi = cmd === 'fmaps';
function action (multi) {
return function (args) {
function add ([lhs, rhs]) {
if (!lhs)
return;
rhs = rhs || lhs;
mappings.addUserMap(
[modes.NORMAL],
[lhs],
args['description'] || 'by feedSomeKeys_3.js',
function () {
function body (win)
(win.document.body || win.document);
let win = document.commandDispatcher.focusedWindow;
let frames = getFrames();
let elem = body(win);
if (typeof args['-frame'] !== 'undefined') {
frames = [frames[args['-frame']]];
elem = body(frames[0]);
}
if (args['-xpath']) {
elem = or(frames, function (f) fromXPath(f.document, args['-xpath'])) || elem;
}
feed(rhs, args['-events'] || ['keypress'], elem);
},
{
matchingUrls: args['-urls'],
feedSomeKeys: {
rhs: rhs,
}
},
true
);
}
if (multi) {
let sep = let (s = args['-separator'] || ',') function (v) v.split(s);
args.literalArg.split(/\s+/).map(String.trim).map(sep).forEach(add);
} else {
let [, lhs, rhs] = args.literalArg.match(/^(\S+)\s+(.*)$/) || args.literalArg;
if (!rhs) {
list({
filter: args.literalArg.trim(),
urls: args['-urls'],
ignoreUrls: !args['-urls']
});
} else {
add([lhs, rhs]);
}
}
};
}
commands.addUserCommand(
[cmd],
'Feed map a key sequence',
action(multi),
{
literal: 0,
options: [
[['-urls', '-u'], commands.OPTION_STRING, regexpValidator, urlCompleter({currentURL: true})],
[['-desc', '-description'], commands.OPTION_STRING],
[['-frame', '-f'], commands.OPTION_INT, null, frameCompleter],
[['-xpath', '-x'], commands.OPTION_STRING, xpathValidator],
[
['-events', '-e'],
commands.OPTION_LIST,
makeListValidator(EVENTS_WITH_V),
EVENTS_WITH_V.map(function (v) [v, v])
]
].concat(
multi ? [[['-separator', '-s'], commands.OPTION_STRING]]
: []
),
completer: multi ? null : fmapCompleter
},
true
);
});
commands.addUserCommand(
['fmapc'],
'Clear fmappings',
function (args) {
if (args.bang) {
unmap({ignoreUrls: true});
liberator.log('All fmappings were removed.');
} else {
let result = {};
unmap({urls: args.literalArg, result: result});
liberator.echo(result.matched ? 'Some fmappings were removed.' : 'Not found specifed fmappings.');
}
},
{
literal: 0,
bang: true,
completer: function (context) {
context.title = ['URL Pattern'];
context.completions = urlCompleter({})(context);
}
},
true
);
commands.addUserCommand(
['funmap'],
'Remove fmappings',
function (args) {
let urls = args['-urls'];
let name = args.literalArg;
if (!name)
return liberator.echoerr('E471: Argument required');
let result = {};
unmap({filter: name, urls: urls, ignoreUrls: args['-ignoreurls'], result: result});
liberator.echo(result.matched ? 'Some fmappings were removed.' : 'Not found specifed fmappings.');
},
{
literal: 0,
options: [
[['-urls', '-u'], commands.OPTION_STRING, regexpValidator, urlCompleter({})],
[['-ignoreurls', '-iu'], commands.OPTION_NOARG]
],
completer: fmapCompleter
},
true
);
plugins.libly.$U.around(
mappings,
'getCandidates',
function (next, [mode, prefix, patternOrUrl]) {
let map = mappings.get(mode, prefix, patternOrUrl);
if (map && map.matchingUrls)
return [];
return next();
}
);
__context__.API =
'VKeys feed getFrames fromXPath virtualize unmap findMappings list'.split(/\s+/).reduce(
function (result, name)
(result[name] = eval(name), result),
{}
);
})();
// vim:sw=2 ts=2 et si fdm=marker: