closebrackets.js 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. (function() {
  2. var DEFAULT_BRACKETS = "()[]{}''\"\"";
  3. var DEFAULT_EXPLODE_ON_ENTER = "[]{}";
  4. var SPACE_CHAR_REGEX = /\s/;
  5. CodeMirror.defineOption("autoCloseBrackets", false, function(cm, val, old) {
  6. if (old != CodeMirror.Init && old)
  7. cm.removeKeyMap("autoCloseBrackets");
  8. if (!val) return;
  9. var pairs = DEFAULT_BRACKETS, explode = DEFAULT_EXPLODE_ON_ENTER;
  10. if (typeof val == "string") pairs = val;
  11. else if (typeof val == "object") {
  12. if (val.pairs != null) pairs = val.pairs;
  13. if (val.explode != null) explode = val.explode;
  14. }
  15. var map = buildKeymap(pairs);
  16. if (explode) map.Enter = buildExplodeHandler(explode);
  17. cm.addKeyMap(map);
  18. });
  19. function charsAround(cm, pos) {
  20. var str = cm.getRange(CodeMirror.Pos(pos.line, pos.ch - 1),
  21. CodeMirror.Pos(pos.line, pos.ch + 1));
  22. return str.length == 2 ? str : null;
  23. }
  24. function buildKeymap(pairs) {
  25. var map = {
  26. name : "autoCloseBrackets",
  27. Backspace: function(cm) {
  28. if (cm.somethingSelected() || cm.getOption("disableInput")) return CodeMirror.Pass;
  29. var cur = cm.getCursor(), around = charsAround(cm, cur);
  30. if (around && pairs.indexOf(around) % 2 == 0)
  31. cm.replaceRange("", CodeMirror.Pos(cur.line, cur.ch - 1), CodeMirror.Pos(cur.line, cur.ch + 1));
  32. else
  33. return CodeMirror.Pass;
  34. }
  35. };
  36. var closingBrackets = "";
  37. for (var i = 0; i < pairs.length; i += 2) (function(left, right) {
  38. if (left != right) closingBrackets += right;
  39. function surround(cm) {
  40. var selection = cm.getSelection();
  41. cm.replaceSelection(left + selection + right);
  42. }
  43. function maybeOverwrite(cm) {
  44. var cur = cm.getCursor(), ahead = cm.getRange(cur, CodeMirror.Pos(cur.line, cur.ch + 1));
  45. if (ahead != right || cm.somethingSelected()) return CodeMirror.Pass;
  46. else cm.execCommand("goCharRight");
  47. }
  48. map["'" + left + "'"] = function(cm) {
  49. if (left == "'" && cm.getTokenAt(cm.getCursor()).type == "comment" ||
  50. cm.getOption("disableInput"))
  51. return CodeMirror.Pass;
  52. if (cm.somethingSelected()) return surround(cm);
  53. if (left == right && maybeOverwrite(cm) != CodeMirror.Pass) return;
  54. var cur = cm.getCursor(), ahead = CodeMirror.Pos(cur.line, cur.ch + 1);
  55. var line = cm.getLine(cur.line), nextChar = line.charAt(cur.ch), curChar = cur.ch > 0 ? line.charAt(cur.ch - 1) : "";
  56. if (left == right && CodeMirror.isWordChar(curChar))
  57. return CodeMirror.Pass;
  58. if (line.length == cur.ch || closingBrackets.indexOf(nextChar) >= 0 || SPACE_CHAR_REGEX.test(nextChar))
  59. cm.replaceSelection(left + right, {head: ahead, anchor: ahead});
  60. else
  61. return CodeMirror.Pass;
  62. };
  63. if (left != right) map["'" + right + "'"] = maybeOverwrite;
  64. })(pairs.charAt(i), pairs.charAt(i + 1));
  65. return map;
  66. }
  67. function buildExplodeHandler(pairs) {
  68. return function(cm) {
  69. var cur = cm.getCursor(), around = charsAround(cm, cur);
  70. if (!around || pairs.indexOf(around) % 2 != 0 || cm.getOption("disableInput"))
  71. return CodeMirror.Pass;
  72. cm.operation(function() {
  73. var newPos = CodeMirror.Pos(cur.line + 1, 0);
  74. cm.replaceSelection("\n\n", {anchor: newPos, head: newPos}, "+input");
  75. cm.indentLine(cur.line + 1, null, true);
  76. cm.indentLine(cur.line + 2, null, true);
  77. });
  78. };
  79. }
  80. })();