aboutsummaryrefslogtreecommitdiffstats
path: root/appendAnchor.js
blob: 8ca02a260c8408719638e18607d08005a73879d4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
let PLUGIN_INFO =
<VimperatorPlugin>
  <name>appendAnchor</name>
  <description>append anchors to texts look like url.</description>
  <description lang="ja">リンク中の URL っぽいテキストにアンカーをつける</description>
  <version>0.4.7</version>
  <author>SAKAI, Kazuaki</author>
  <minVersion>2.0pre</minVersion>
  <maxVersion>2.4</maxVersion>
  <detail><![CDATA[
    == Commands ==
      :anc:
        Append anchors.
    == GlobalVariables ==
      g:auto_append_anchor:
        Execute ":anc" automatically when Vimperator shows the hints.
      g:auto_append_anchor_once:
        Just first once.
  ]]></detail>
  <detail lang="ja"><![CDATA[
    == Commands ==
      :anc:
        アンカーを付加する
    == GlobalVariables ==
      g:auto_append_anchor:
        Vimperator がヒントを表示するときに自動的に ":anc" する
      g:auto_append_anchor_once:
        最初の一回だけ
  ]]></detail>
</VimperatorPlugin>;

(function(){

  // settings ---
  // "ACEILMPRSTXY" is result of below code.
  //   Array.prototype.uniq = function() this.reduceRight( function (a, b) (a[0] === b || a.unshift(b), a), []);
  //   [ 'TITLE', 'STYLE', 'SCRIPT', 'TEXTAREA', 'XMP', 'A', ].join('').split('').sort().uniq().join('');
  const xpathQueryPlainText = '/descendant::*[not(contains(" TITLE STYLE SCRIPT TEXTAREA XMP A ", concat(" ", translate(local-name(), "aceilmprstxy", "ACEILMPRSTXY"), " ")))]/child::text()';
  const regexpLikeURL = new RegExp("(h?ttps?|ftp):/+([a-zA-Z0-9][-_.!~*'()a-zA-Z0-9;/?:@&=+$,%#]+[-_~*(a-zA-Z0-9;/?@&=+$%#])");

  // process global variable
  if (stringToBoolean(liberator.globalVariables.auto_append_anchor, false)) {
    let originalHintsShow = liberator.modules.hints.show;
    let once = stringToBoolean(liberator.globalVariables.auto_append_anchor_once, true);
    hints.show = function () {
      if (!(once && content.document.anchor_appended)) {
        content.document.anchor_appended = true;
        liberator.execute('anc');
      }
      originalHintsShow.apply(this, arguments);
    };
  }

  // register command
  liberator.modules.commands.addUserCommand(['anc'], 'append anchors to texts look like url',
    function(arg) {
      function append() {
        let result = 0;
        const doc = window.content.document;
        const range = doc.createRange();

        let nodes = util.evaluateXPath(xpathQueryPlainText);
        for (let node in nodes) {
          while (node) {
            range.selectNode(node)

            // search string like URL
            let start = range.toString().search(regexpLikeURL);
            // go to next node when there is nothing look like URL in current node
            if (!(start > -1)) break;

            result++;

            // build URL
             let scheme = RegExp.$1, host = RegExp.$2, lastMatch = RegExp.lastMatch;
             if (/^ttps?$/.test(scheme)) scheme = 'h' + scheme;
             let href = scheme + '://' + host;

            // reset range
            range.setStart(node, start);
            range.setEnd(node, start + lastMatch.length);

            // build anchor element
            let anchor = doc.createElement('a');
            anchor.setAttribute('href', href);
            range.surroundContents(anchor);

            // insert
            range.insertNode(anchor);

            // iterate
            node = node.nextSibling.nextSibling.nextSibling;
          }
        }
        range.detach();
        return result;
      }
      for (let i = 0; i < 20 && append(); i++)
        ;
    },
    {},
    true
  );

  // stuff function
  function stringToBoolean(str, defaultValue) {
    if (typeof str === 'number')
      str = str.toString();
    return !str                          ? (defaultValue ? true : false)
         : str.toLowerCase() === 'false' ? false
         : /^\d+$/.test(str)             ? (parseInt(str) ? true : false)
         :                                 true;
  }

})();
pan> the pathname of the current location</dd> <dt>%HOST%</dt> <dd>to the host of the current location</dd> <dt>%PORT%</dt> <dd>to the port of the current location</dd> <dt>%PROTOCOL%</dt> <dd>to the protocol of the current location</dd> <dt>%SERCH%</dt> <dd>to the search(?...) of the curernt location</dd> <dt>%HASH%</dt> <dd>to the hash(anchor #..) of the current location</dd> </dl> </description> </item> <item> <tags>copy-template</tags> <spec>copy-template</spec> <description> <p>you can set your own template using inline JavaScript</p> <code><![CDATA[ javascript <<EOM liberator.globalVariables.copy_templates = [ { label: 'titleAndURL', value: '%TITLE%\n%URL%' }, { label: 'title', value: '%TITLE%', map: ',y' }, { label: 'anchor', value: '<a href="%URL%">%TITLE%</a>' }, { label: 'selanchor', value: '<a href="%URL%" title="%TITLE%">%SEL%</a>' }, { label: 'htmlblockquote', value: '<blockquote cite="%URL%" title="%TITLE%">%HTMLSEL%</blockquote>' } { label: 'ASIN', value: 'copy ASIN code from Amazon', custom: function(){return content.document.getElementById('ASIN').value;} }, ]; EOM ]]></code> <dl> <dt>label</dt> <dd>template name which is command argument</dd> <dt>value</dt> <dd>copy string. <a>copy-keyword</a> is replaced</dd> <dt>map</dt> <dd>key map <a>lhs</a> (optional)</dd> <dt>custom</dt> <dd> <a>function</a> or <a>Array</a> (optional) <dl> <dt><a>function</a></dt> <dd>execute the function and copy return value, if specified</dd> <dt><a>Array</a></dt> <dd> replace to the <a>value</a> by normal way at first. then replace words matched <a>Array</a>[0] in the repalced string to <a>Array</a>[1]. <dl> <dt><a>Array</a>[0]</dt> <dd>String or RegExp</dd> <dt><a>Array</a>[1]</dt> <dd>String or Function</dd> </dl> see: <link topic="http://developer.mozilla.org/en/docs/Core_JavaScript_1.5_Reference:Global_Objects:String:replace">http://developer.mozilla.org/en/docs/Core_JavaScript_1.5_Reference:Global_Objects:String:replace</link> </dd> </dl> </dd> </dl> </description> </item> <item> <tags>copy-option</tags> <spec>copy-option</spec> <description> <code><ex>liberator.globalVariables.copy_use_wedata = false; // false by default</ex></code> <p>true に設定すると wedata からテンプレートを読込みます</p> <code><ex>liberator.globalVariables.copy_wedata_include_custom = true; // false by default</ex></code> <p>custom が設定された wedata を読込みます SandBox でなくwindow.eval を利用してオブジェクトする為 セキュリティ上の理由で初期設定は false になっています true に設定する場合は動作を理解したうえ自己責任でご利用ください</p> <code><ex>liberator.globalVariables.copy_wedata_exclude_labels = ['pathtraqnormalize', ];</ex></code> <p>wedata から読込まない label のリストを定義します</p> </description> </item> </plugin>; var PLUGIN_INFO = <VimperatorPlugin> <name>{NAME}</name> <description>enable to copy strings from a template (like CopyURL+)</description> <description lang="ja">テンプレートから文字列のコピーを可能にしますCopyURL+みたいなもの</description> <minVersion>2.0pre</minVersion> <maxVersion>2.0pre</maxVersion> <updateURL>https://github.com/vimpr/vimperator-plugins/raw/master/copy.js</updateURL> <author mail="teramako@gmail.com" homepage="http://vimperator.g.hatena.ne.jp/teramako/">teramako</author> <license>MPL 1.1/GPL 2.0/LGPL 2.1</license> <version>0.7.5</version> </VimperatorPlugin>; liberator.plugins.exCopy = (function(){ var excludeLabelsMap = {}; var copy_templates = []; if (!liberator.globalVariables.copy_templates){ liberator.globalVariables.copy_templates = [ { label: 'titleAndURL', value: '%TITLE%\n%URL%' }, { label: 'title', value: '%TITLE%' }, { label: 'anchor', value: '<a href="%URL%">%TITLE%</a>' }, { label: 'selanchor', value: '<a href="%URL%" title="%TITLE%">%SEL%</a>' }, { label: 'htmlblockquote', value: '<blockquote cite="%URL%" title="%TITLE%">%HTMLSEL%</blockquote>' } ]; } copy_templates = liberator.globalVariables.copy_templates.map(function(t){ return { label: t.label, value: t.value, custom: t.custom, map: t.map } }); copy_templates.forEach(function(template){ if (typeof template.map == 'string') addUserMap(template.label, [template.map]); else if (template.map instanceof Array) addUserMap(template.label, template.map); }); const REPLACE_TABLE = { get TITLE () buffer.title, get URL () buffer.URL, get SEL () { var sel = ''; var win = new XPCNativeWrapper(window.content.window); var selection = win.getSelection(); if (selection.rangeCount < 1) return ''; for (var i=0, c=selection.rangeCount; i<c; i++){ sel += selection.getRangeAt(i).toString(); } return sel; }, get HTMLSEL () { var htmlsel = ''; var win = new XPCNativeWrapper(window.content.window); var selection = win.getSelection(); if (selection.rangeCount < 1) return ''; var serializer = new XMLSerializer(); for (var i=0, c=selection.rangeCount; i<c; i++){ htmlsel += serializer.serializeToString(selection.getRangeAt(i).cloneContents()); } return htmlsel.replace(/<(\/)?(\w+)([\s\S]*?)>/g, function(all, close, tag, attr){ return "<" + close + tag.toLowerCase() + attr + ">"; }); }, get CLIP () { return util.readFromClipboard(); } }; 'hostname pathname host port protocol search hash'.split(' ').forEach(function (name){ REPLACE_TABLE[name.toUpperCase()] = function () content.location && content.location[name]; }); // used when argument is none //const defaultValue = templates[0].label; commands.addUserCommand(['copy'],'Copy to clipboard', function(args){ liberator.plugins.exCopy.copy(args.literalArg, args.bang, !!args["-append"]); },{ completer: function(context, args){ if (args.bang){ completion.javascript(context); return; } context.title = ['Template','Value']; var templates = copy_templates.map(function(template) [template.label, liberator.modules.util.escapeString(template.value, '"')] ); if (!context.filter){ context.completions = templates; return; } var candidates = []; var filter = context.filter.toLowerCase(); context.completions = templates.filter(function(template) template[0].toLowerCase().indexOf(filter) == 0); }, literal: 0, bang: true, options: [ [["-append","-a"], commands.OPTION_NOARG] ] }, true ); function addUserMap(label, map){ mappings.addUserMap([modes.NORMAL,modes.VISUAL], map, label, function(){ liberator.plugins.exCopy.copy(label); }, { rhs: label } ); } function getCopyTemplate(label){ var ret = null; copy_templates.some(function(template) template.label == label ? (ret = template) && true : false); return ret; } function replaceVariable(str){ if (!str) return ''; function replacer(orig, name){ //{{{ if (name == '') return '%'; if (!REPLACE_TABLE.hasOwnProperty(name)) return orig; let value = REPLACE_TABLE[name]; if (typeof value == 'function') return value(); else return value.toString(); return orig; } //}}} return str.replace(/%([A-Z]*)%/g, replacer); } function wedataRegister(item){ var libly = liberator.plugins.libly; var logger = libly.$U.getLogger("copy"); item = item.data; if (excludeLabelsMap[item.label]) return; if (item.custom && item.custom.toLowerCase().indexOf('function') != -1) { if (!liberator.globalVariables.copy_wedata_include_custom || item.label == 'test') { logger.log('skip: ' + item.label); return; } let custom = (function(item){ return function(value, value2){ var STORE_KEY = 'plugins-copy-ok-func'; var store = storage.newMap(STORE_KEY, {store: true}); var check = store.get(item.label); var ans; if (!check){ ans = window.confirm( 'warning!!!: execute "' + item.label + '" ok ?\n' + '(this function is working with unsafe sandbox.)\n\n' + '----- execute code -----\n\n' + 'value: ' + item.value + '\n' + 'function: ' + item.custom ); } else { if (item.value == check.value && item.custom == check.custom && item.map == check.map){ ans = true; } else { ans = window.confirm( 'warning!!!: "' + item.label + '" was changed when you registered the function.\n' + '(this function is working with unsafe sandbox.)\n\n' + '----- execute code -----\n\n' + 'value: ' + item.value + '\n' + 'function: ' + item.custom ); } } if (!ans) return; store.set(item.label, item); store.save(); var func; try{ func = window.eval('(' + item.custom + ')'); } catch (e){ logger.echoerr(e); logger.log(item.custom); return; } return func(value, value2); }; })(item); exCopyManager.add(item.label, item.value, custom, item.map); } else { exCopyManager.add(item.label, item.value, null, item.map); } } var exCopyManager = { add: function(label, value, custom, map){ var template = {label: label, value: value, custom: custom, map: map}; copy_templates.unshift(template); if (map) addUserMap(label, map); return template; }, get: function(label){ return getCopyTemplate(label); }, copy: function(arg, special, appendMode){ var copyString = ''; var isError = false; if (special && arg){ try { copyString = liberator.eval(arg); switch (typeof copyString){ case 'object': copyString = copyString === null ? 'null' : copyString.toSource(); break; case 'function': copyString = copyString.toString(); break; case 'number': case 'boolean': copyString = '' + copyString; break; case 'undefined': copyString = 'undefined'; break; } } catch (e){ isError = true; copyString = e.toString(); } } else { if (!arg) arg = copy_templates[0].label; var template = getCopyTemplate(arg) || {value: arg}; if (typeof template.custom == 'function'){ copyString = template.custom.call(this, template.value, replaceVariable(template.value)); } else if (template.custom instanceof Array){ copyString = replaceVariable(template.value).replace(template.custom[0], template.custom[1]); } else { copyString = replaceVariable(template.value); } } if (appendMode){ copyString = util.readFromClipboard() + copyString; } if (copyString) util.copyToClipboard(copyString); if (isError){ liberator.echoerr('CopiedErrorString: `' + copyString + "'"); } else { liberator.echo('CopiedString: `' + util.escapeHTML(copyString || '') + "'"); } } }; if (liberator.globalVariables.copy_use_wedata){ function loadWedata(){ if (!liberator.plugins.libly){ liberator.echomsg("need a _libly.js when use wedata."); return; } var libly = liberator.plugins.libly; copy_templates.forEach(function(item) excludeLabelsMap[item.label] = item.value); if (liberator.globalVariables.copy_wedata_exclude_labels) liberator.globalVariables.copy_wedata_exclude_labels.forEach(function(item) excludeLabelsMap[item] = 1); var wedata = new libly.Wedata("vimp%20copy"); wedata.getItems(24 * 60 * 60 * 1000, wedataRegister); } loadWedata(); } return exCopyManager; })(); // vim: set fdm=marker sw=4 ts=4 et: