/* 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: