テキスト変換

提供: MeryWiki
移動先: 案内検索

概要[編集]

開いているドキュメントのテキストを変換します.

特徴[編集]

  • 機能追加が簡単.

更新履歴[編集]

  • 2013/1/23 初版
  • 2013/1/24 グルーピング対応.replaceS, replaceL 対応.
  • 2013/1/25 簡易表記対応.

使用方法[編集]

  1. 基本コードを拡張子 .js で保存します.
  2. 必要な拡張コードを追加します.
    • 「拡張機能の追加」とある場所に,コードをコピペすれば拡張が可能です.
    • 上から順にポップアップに表示されます.
  3. マクロを実行すると登録されている機能の一覧が表示されるので,選択して実行します.
    • テキストを選択している場合は,対象を「選択テキスト」「全体」から選べます.
    • テキストを選択していない場合は「全体」が対象です.

便利な使い方[編集]

  1. セパレータの挿入
    f[f.length] = "--";
    でセパレータを挿入できます(- の数は任意).
  2. グルーピング
    下のコードで,グルーピングができます.
f[f.length] = [ "グループ名", function(f){    // <= グルーピング開始
  f[f.length] = {     // <= 以降はグルーピングされる機能を追加
    title:"大文字に変換",
    replace:function(text){ return text.toUpperCase() }
  };
  f[f.length] = {
    title:"小文字に変換",
    replace:function(text){ return text.toLowerCase() }
  };
]};   // <= グルーピングの終了

コード[編集]

基本コード[編集]

#title = "テキスト変換"
// -----------------------------------------------------------------------------
// 手軽にテキスト変換を行うためのマクロ
// -----------------------------------------------------------------------------
var f = [];
 
// 拡張機能の追加
// ↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓
 
f[f.length] = [ "行頭の空白文字を削除", "", function(a){return a.replace(/^[ \t ]+/mg,"")} ];
f[f.length] = [ "行末の空白文字を削除", "", function(a){return a.replace(/[ \t ]+$/mg,"")} ];
f[f.length] = [ "行頭・行末の空白文字を削除", "", function(a){return a.replace(/^[ \t ]+|[ \t ]+$/mg,"")} ];
f[f.length] = [ "空白行を削除", "L", function(text){ return text==""?null:text} ];
 
// ↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑
 
// 基本機能
(function(){
  var topMenu = CreatePopupMenu();
  var menus = null;
  var selMenu = null;
  var docMenu = topMenu;
 
  var DIFF_ID      = 0x10000;   // 適用対象を判別するための ID の最低値
  var SELECTION_ID = 0x10000;   // Selection を対象
  var DOCUMENT_ID  = 0x20000;   // Document を対象
  var EDITOR_ID    = 0x40000;   // Editor を対象(未対応)
 
  // サブメニュー登録
  var AddSubMenu = function(menu, text) {
    var subMenu = null;
    if (menu){
      subMenu = CreatePopupMenu();
      menu.AddPopup(text, subMenu);
    }
    return subMenu;
  };
  // コマンド登録
  var commands = {}, id = 0;
  var AddCommand = function(menus, text, cmd, flag) {
    for (var i=0; i<menus.length; i++) {
      if (menus[i]) {
        menus[i].Add(text, (DIFF_ID << i) + id, flag);
        commands[id] = cmd;
        id++;
      }
    }
  };
  // 選択領域を行全体に変更してその文字列を返す
  SelectLine = function(selection, isPosView) {
    var posFlag = isPosView ? mePosView : mePosLogical;
    var ty = selection.GetTopPointY(posFlag);
    var x = ScrollX, y = ScrollY;
    selection.SetActivePoint(posFlag, 1, selection.GetBottomPointY(posFlag));
    selection.EndOfLine(false, posFlag);
    selection.SetAnchorPoint(posFlag, 1, ty);
    ScrollX = x; ScrollY = y;
    return selection.Text;
  };
  // 行単位処理
  EnumLine = function(selection, isPosView, replace) {
    var lines = SelectLine(selection, isPosView).split("\n");
    var result = [];
    for (var i=0; i<lines.length; i++) {
      var text = replace(lines[i]);
      if (text != null) {
        result.push(text);
      }
    }
    return result.join("\n");
  }
 
  // 選択文字列があるときは選択肢追加
  var sel = document.Selection;
  if (sel.Text) {
    selMenu = AddSubMenu(topMenu, "選択文字列");
    docMenu = AddSubMenu(topMenu, "全体");
  }
 
  // 選択肢追加
  // 定義リストからメニューを作成
  var ListToMenu = function(menus, f) {
    for (var i=0; i<f.length; i++) {
      // グループ簡易設定の変換
      if (f[i] instanceof Array && f[i].length >= 2 && typeof f[i][0] == "string" && typeof f[i][1] == "function") {
        f[i] = { title:f[i][0], group:f[i][1] };
      }
      // 関数簡易設定の変換
      if (f[i] instanceof Array && f[i].length >= 3 && typeof f[i][0] == "string" && typeof f[i][1] == "string" && typeof f[i][2] == "function") {
        var ar = f[i];
        f[i] = { title:ar[0] };
        switch (ar[1]) {
          case "": f[i]["replace"] = ar[2]; break;
          case "S": f[i]["replaceS"] = ar[2]; break;
          case "L": f[i]["replaceL"] = ar[2]; break;
        }
        if (ar.length >= 4 && typeof ar[3] == "boolean") {
          f[i]["view"] = ar[3];
        }
      }
      if (typeof f[i]["title"] == "string") {
        // グループ登録またはコマンド登録
        if (typeof f[i]["group"] == "function") {
          var group = [];
          f[i].group(group);
          var subMenus = [];
          for (var j=0; j<menus.length; j++) {
            subMenus.push(AddSubMenu(menus[j], f[i].title));
          }
          ListToMenu(subMenus, group);
        } else if (typeof f[i]["replace"] == "function") {
          // 通常コマンド
          var exec = function(replace, text, isViewMode, selection){ return replace(text, selection) };
          AddCommand(menus, f[i].title, {replace:f[i].replace, exec:exec});
        } else if (typeof f[i]["replaceS"] == "function") {
          // 行全体選択
          var exec = function(replace, text, isViewMode, selection){ return replace(SelectLine(selection, isViewMode), selection) };
          AddCommand(menus, f[i].title, {replace:f[i].replaceS, isViewMode:f[i]["view"], exec:exec});
        } else if (typeof f[i]["replaceL"] == "function") {
          // 行単位処理
          var exec = function(replace, text, isViewMode, selection){ return EnumLine(selection, isViewMode, replace) };
          AddCommand(menus, f[i].title, {replace:f[i].replaceL, isViewMode:f[i]["view"], exec:exec});
        }
      } else if (typeof f[i] == "string" && /^-+$/.test(f[i])) {
        // セパレータ処理
        AddCommand(menus, "", null, meMenuSeparator);
      }
    }
  };
  ListToMenu([selMenu, docMenu], f);
 
  // ポップアップを表示して選択されたコマンドを実行
  var mePosCaret = 0;
  var select = topMenu.Track(mePosMouse);
  if (select > 0) {
    var command = commands[select&(DIFF_ID-1)];
    if (select & SELECTION_ID) {
      sel.Text = command.exec(command.replace, sel.Text, command["isViewMode"], sel);
    } else if (select & DOCUMENT_ID) {
      var x = ScrollX, y = ScrollY;
      sel.SelectAll();
      ScrollX = x; ScrollY = y;
      Document.Text = command.exec(command.replace, Document.Text, command["isViewMode"], sel);
    }
  }
})();

拡張コード[編集]

ご自由に追加してください!
2つ並んでいますが,機能は全く同じです.
管理のしやすさから,1 行にまとめている方をお勧めします.

空白削除[編集]

f[f.length] = [ "空白削除", function(f){
  f[f.length] = [ "行頭の空白文字を削除", "", function(a){return a.replace(/^[ \t ]+/mg,"")} ];
  f[f.length] = [ "行末の空白文字を削除", "", function(a){return a.replace(/[ \t ]+$/mg,"")} ];
  f[f.length] = [ "行頭・行末の空白文字を削除", "", function(a){return a.replace(/^[ \t ]+|[ \t ]+$/mg,"")} ];
  f[f.length] = [ "空白行を削除", "L", function(text){ return text==""?null:text} ];
  f[f.length] = [ "カンマ前後の空白文字を削除", "", function(a){return a.replace(/^[ \t ]+|[ \t ]+$/mg,"").replace(/[ \t ]*,[ \t ]*/mg,",")} ];
}];
f[f.length] = {
  title:"空白削除",
  group:function(f){
    f[f.length] = {
      title:"行頭の空白文字を削除",
      replace:function(text){ return text.replace(/^[ \t ]+/mg, "") }
    };
    f[f.length] = {
      title:"行末の空白文字を削除",
      replace:function(text){ return text.replace(/[ \t ]+$/mg, "") }
    };
    f[f.length] = {
      title:"行頭・行末の空白文字を削除",
      replace:function(text){ return text.replace(/^[ \t ]+|[ \t ]+$/mg, "") }
    };
    f[f.length] = {
      title:"空白行を削除",
      replaceL:function(text){ return text=="" ? null : text }
    };
    f[f.length] = {
      title:"カンマ前後の空白文字を削除",
      replace:function(text){ return text.replace(/^[ \t ]+|[ \t ]+$/mg, "").replace(/[ \t ]*,[ \t ]*/mg, ","); }
    };
  }
};

文字変換[編集]

f[f.length] = [ "文字変換", function(f){
  f[f.length] = [ "大文字に変換", "", function(a){return a.toUpperCase()} ];
  f[f.length] = [ "小文字に変換", "", function(a){return a.toLowerCase()} ];
  f[f.length] = [ "全角⇒半角", function(f){
    f[f.length] = [ "英字", "", function(a){return a.replace(/[A-Za-z]/mg,function(s){return String.fromCharCode(s.charCodeAt(0)-0xFEE0)})} ];
    f[f.length] = [ "数字", "", function(a){return a.replace(/[0-9]/mg,function(s){return String.fromCharCode(s.charCodeAt(0)-0xFEE0)})} ];
    f[f.length] = [ "英数字", "", function(a){return a.replace(/[A-Za-z0-9]/mg,function(s){return String.fromCharCode(s.charCodeAt(0)-0xFEE0)})} ];
  }];
  f[f.length] = [ "半角⇒全角", function(f){
    f[f.length] = [ "英字", "", function(a){return a.replace(/[A-Za-z]/mg,function(s){return String.fromCharCode(s.charCodeAt(0)+0xFEE0)})} ];
    f[f.length] = [ "数字", "", function(a){return a.replace(/[0-9]/mg,function(s){return String.fromCharCode(s.charCodeAt(0)+0xFEE0)})} ];
    f[f.length] = [ "英数字", "", function(a){return a.replace(/[A-Za-z0-9]/mg,function(s){return String.fromCharCode(s.charCodeAt(0)+0xFEE0)})} ];
  }];
  f[f.length] = [ "アラビア数字⇒漢数字", "", function(a){return a.replace(/[0-9]/mg,function(s){return String.fromCharCode(s.charCodeAt(0)-0xFEE0)}).replace(/[0-9]/mg,function(s){return"〇一二三四五六七八九".substr(Number(s),1)})} ];
}];
f[f.length] = {
  title:"文字変換",
  group:function(f){
    f[f.length] = {
      title:"大文字に変換",
      replace:function(text){ return text.toUpperCase() }
    };
    f[f.length] = {
      title:"小文字に変換",
      replace:function(text){ return text.toLowerCase() }
    };
    f[f.length] = {
      title:"全角⇒半角",
      group:function(f){
        f[f.length] = {
          title:"英字",
          replace:function(text){ return text.replace(/[A-Za-z]/mg, function(s){ return String.fromCharCode(s.charCodeAt(0) - 0xFEE0) }) }
        };
        f[f.length] = {
          title:"数字",
          replace:function(text){ return text.replace(/[0-9]/mg, function(s){ return String.fromCharCode(s.charCodeAt(0) - 0xFEE0) }) }
        };
        f[f.length] = {
          title:"英数字",
          replace:function(text){ return text.replace(/[A-Za-z0-9]/mg, function(s){ return String.fromCharCode(s.charCodeAt(0) - 0xFEE0) }) }
        };
      }
    };
    f[f.length] = {
      title:"半角⇒全角",
      group:function(f){
        f[f.length] = {
          title:"英字",
          replace:function(text){ return text.replace(/[A-Za-z]/mg, function(s){ return String.fromCharCode(s.charCodeAt(0) + 0xFEE0) }) }
        };
        f[f.length] = {
          title:"数字",
          replace:function(text){ return text.replace(/[0-9]/mg, function(s){ return String.fromCharCode(s.charCodeAt(0) + 0xFEE0) }) }
        };
        f[f.length] = {
          title:"英数字",
          replace:function(text){ return text.replace(/[A-Za-z0-9]/mg, function(s){ return String.fromCharCode(s.charCodeAt(0) + 0xFEE0) }) }
        };
      }
    };
    f[f.length] = {
      title:"アラビア数字⇒漢数字",
      replace:function(text){
        return text.replace(/[0-9]/mg, function(s){ return String.fromCharCode(s.charCodeAt(0) - 0xFEE0) }).replace(/[0-9]/mg, function(s){ return "〇一二三四五六七八九".substr(Number(s), 1); });
      }
    };
  }
};

ソート[編集]

昇順で並び替え」「降順で並び替え」 の移植です.

f[f.length] = [ "ソート", function(f){
  f[f.length] = [ "昇順ソート(文字列)", "S", function(a){return a.split("\n").sort().join("\n")} ];
  f[f.length] = [ "降順ソート(文字列)", "S", function(a){return a.split("\n").sort().reverse().join("\n")} ];
}];
f[f.length] = {
  title:"ソート",
  group:function(f){
    f[f.length] = {
      title:"昇順ソート(文字列)",
      replaceS:function(text){ return text.split("\n").sort().join("\n") }
    };
    f[f.length] = {
      title:"降順ソート(文字列)",
      replaceS:function(text){ return text.split("\n").sort().reverse().join("\n") }
    };
  }
};

行操作[編集]

連続する重複行を削除」の移植を含みます.

f[f.length] = [ "行操作", function(f){
  f[f.length] = [ "文字列追加", function(f){
    f[f.length] = [ "先頭", "L", (function(b){var c=null;return function(a){if(!c){if(!(c=Prompt("追加する文字列を入力してください.",""))){Quit()}}return c+a}})() ];
    f[f.length] = [ "末尾", "L", (function(b){var c=null;return function(a){if(!c){if(!(c=Prompt("追加する文字列を入力してください.",""))){Quit()}}return a+c}})() ];
  }];
  f[f.length] = [ "行削除", function(f){
    f[f.length] = [ "含む行(文字列)", "L", (function(b){var c=null;return function(a){if(!c){if(!(c=Prompt("検索文字列",""))){Quit()}}return a.indexOf(c)>=0?null:a}})() ];
    f[f.length] = [ "含む行(正規表現)", "L", (function(b){var c=null;return function(a){if(!c){if(!(c=Prompt("正規表現",""))){Quit()}else{c=new RegExp(c)}}return c.test(a)?null:a}})() ];
    f[f.length] = [ "含まない行(文字列)", "L", (function(b){var c=null;return function(a){if(!c){if(!(c=Prompt("検索文字列",""))){Quit()}}return a.indexOf(c)>=0?a:null}})() ];
    f[f.length] = [ "含まない行(正規表現)", "L", (function(b){var c=null;return function(a){if(!c){if(!(c=Prompt("正規表現",""))){Quit()}else{c=new RegExp(c)}}return c.test(a)?a:null}})() ];
    f[f.length] = [ "連続する重複を削除", "S", function(a){a=a.split("\n");var b=[a[0]];for(var i=1;i<a.length;i++){if(a[i]!==a[i-1]){b.push(a[i])}}return b.join("\n")} ];
    f[f.length] = [ "重複行を削除", "S", function(a){a=a.split("\n");var b=[],map={};for(var i=0;i<a.length;i++){if(!map[a[i]]){map[a[i]]=true;b.push(a[i])}}return b.join("\n")} ];
  }];
}];
f[f.length] = {
  title:"行操作",
  group:function(f){
    f[f.length] = {
      title:"文字列追加",
      group:function(f){
        f[f.length] = {
          title:"先頭",
          replaceL:(function(text){
            var input = null;
            return function(text){
              if (!input){ if(!(input=Prompt("追加する文字列を入力してください.",""))){Quit()} }
              return input + text;
            }
          })()
        };
        f[f.length] = {
          title:"末尾",
          replaceL:(function(text){
            var input = null;
            return function(text){
              if (!input){ if(!(input=Prompt("追加する文字列を入力してください.",""))){Quit()} }
              return text + input;
            }
          })()
        };
      }
    };
    f[f.length] = {
      title:"行削除",
      group:function(f){
        f[f.length] = {
          title:"含む行(文字列)",
          replaceL:(function(text){
            var input = null;
            return function(text){
              if (!input){ if(!(input=Prompt("検索文字列", ""))){Quit()} }
              return text.indexOf(input) >= 0 ? null : text;
            }
          })()
        };
        f[f.length] = {
          title:"含む行(正規表現)",
          replaceL:(function(text){
            var input = null;
            return function(text){
              if (!input){ if(!(input=Prompt("正規表現", ""))){Quit()}else{input=new RegExp(input)} }
              return input.test(text) ? null : text;
            }
          })()
        };
        f[f.length] = {
          title:"含まない行(文字列)",
          replaceL:(function(text){
            var input = null;
            return function(text){
              if (!input){ if(!(input=Prompt("検索文字列", ""))){Quit()} }
              return text.indexOf(input) >= 0 ? text : null;
            }
          })()
        };
        f[f.length] = {
          title:"含まない行(正規表現)",
          replaceL:(function(text){
            var input = null;
            return function(text){
              if (!input){ if(!(input=Prompt("正規表現", ""))){Quit()}else{input=new RegExp(input)} }
              return input.test(text) ? text : null;
            }
          })()
        };
        f[f.length] = {
          title:"連続する重複を削除",
          replaceS:function(text){
            text = text.split("\n");
            var result = [text[0]];
            for (var i=1; i<text.length; i++) {
              if (text[i] !== text[i-1]) {
                result.push(text[i]);
              }
            }
            return result.join("\n");
          }
        };
        f[f.length] = {
          title:"重複行を削除",
          replaceS:function(text){
            text = text.split("\n");
            var result = [], map = {};
            for (var i=0; i<text.length; i++) {
              if (!map[text[i]]) {
                map[text[i]] = true;
                result.push(text[i]);
              }
            }
            return result.join("\n");
          }
        };
      }
    };
  }
};

グルーピング[編集]

選択範囲を簡易的にグルーピングします.

f[f.length] = [ "機能のグルーピング", "S", function(a){var b=Prompt("グループ名","");if(!b){Quit()}var c=a.match(/^\s*/);var d="\t";return c+'f[f.length] = [ "'+b+'", function(f){\n'+d+a.split("\n").join("\n"+d)+"\n"+c+"}];"} ];
f[f.length] = {
  title:"機能のグルーピング",
  replaceS:function(text){
    var group = Prompt("グループ名", "");
    if (!group) { Quit(); }
    var indent = text.match(/^\s*/);
    var tab = "\t";   // インデント方式.タブでない場合はその分の空白を代入しておく.
    return indent + 'f[f.length] = [ "' + group + '", function(f){\n'+ tab + text.split("\n").join("\n" + tab) + "\n" + indent + "}];";
  }
};

BASE64エンコード(Unicode)[編集]

f[f.length] = {
  title:"BASE64エンコード(Unicode)",
  replace:function(text){
    var map = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".split("");
    var Ch = function(n){ return n<text.length ? text.charCodeAt(n) : 0 };
    var len = text.length, pad = len % 3, s = "";
    for (var i=0; i<len; i+=3) {
      var tmp = (Ch(i)<<16) | (Ch(i+1)<<8) | Ch(i+2);
      s += map[(tmp>>>18)&0x3F] + map[(tmp>>>12)&0x3F] + map[(tmp>>>6)&0x3F] + map[tmp&0x3F];
    }
    if (pad) { s = s.substring(0, s.length-(3-pad)) + "===".substring(pad) }
    return s;
  }
};

開発者向け資料[編集]

これは拡張コードを開発される方向けの仕様です.
拡張コードは次のいずれかの関数を実装します.

  1. replace
  2. replaceS
  3. replaceL

replace[編集]

function replace(
    text : string,
    selection : Selection
) : string

最も基本的な変換関数です.
選択されている文字列,またはドキュメント全体の文字列を引数として, 変換結果を返します.
selection 引数は通常必要有りません. Selection の持つ関数を利用したい場合などに利用します.

replaceS[編集]

function replaceS(
    text : string,
    selection : Selection
) : string

行途中の選択がされている場合に,行全体を選択し直してから実行される変換関数です.
行は論理行で処理しますが,下記のように view:true を定義時に登録することで, 表示行で処理することもできます.

f[f.length] = {
  title:"replaceL",
  view:true,
  replaceS:function(text){ return text }
};

replaceL[編集]

function replaceL(
    text : string,
    selection : Selection
) : string

行単位で処理する場合に利用する変換関数です.
渡される text は,他の関数と違い「1 行分の文字列」です.
String.replaceString.split("\n").join("\n") を利用することで同様の処理が行えますが,より簡易的に処理するために用意されています.

次の 2 つの定義は同じ処理(空白行削除)をします.

replaceS:function(text){ return text.replace(/^\n/mg, "") }
replaceL:function(text){ if (text != "") { return text } }

replaceL 関数の場合,null を返すかあるいは return しない場合に行を削除します.

replaceL 関数は行毎に呼ばれるため,全体で 1 回のみの処理を記述するのが面倒です.
もし必要な場合は,拡張コードの「文字列追加」のようにクロージャを利用する必要があります.

関数の簡易表記[編集]

関数を 1 行で書きやすくするため,簡易表記を追加しました.
配列の形式で記述し,

f[f.length] = [ "機能名", "関数タイプ", function(text){ return text; }, isView

で記述します.

関数タイプは文字で記述し,それぞれ次のように対応します.

  • "" ⇒ replace
  • "S" ⇒ replaceS
  • "L" ⇒ replaceL

従来通りの記述から,簡易形式に変換するためのマクロを用意しました.
関数部分はクリップボードにコピーされるため,そのまま貼り付けるか,
packer などを利用して圧縮して貼り付けてください.

// 関数の簡易表現化
// 関数部分はクリップボードにコピーする
// http://dean.edwards.name/packer/ などを用いて圧縮して貼り付ける.(別のそのまま貼り付けてもOK)
// view 属性には非対応
f[f.length] = {
  title:"関数の簡易表現化",
  replaceS:function(text){
    var ar = text.split("\n");
    if (!/^\s*f\[f\.length\]\s*=\s*\{\s*$/.test(ar[0]) || !/^\s*\};\s*/.test(ar[ar.length-1])) {
      Alert("対応していない表記です.選択範囲のミスはありませんか?");
      Quit();
    }
    var indent = text.match(/^\s*/);
    var result = indent + 'f[f.length] = [';
    var func = "";
    for (var i=1; i<ar.length-1; i++) {
      if (ar[i].match(/^\s*title:(.*)$/)) {
        result += " " + RegExp.$1;
      } else if (ar[i].match(/^\s*replace([A-Z]?):(.*)$/)) {
        result += ' "' + RegExp.$1 + '", ';
        func += RegExp.$2 + "\n";
      } else {
        //func += ar[i].replace(/^\s*/, "");
        func += ar[i] + "\n";
      }
    }
    ClipboardData.SetData(func);
    //result += " " + func + "];"
    result += " ];";
    return result;
  }
};


ご意見・ご要望[編集]

何かありましたら,右の [編集] から追記してください.対応するもかしれません.