closebrackets.js 3.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182
  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()) 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. return CodeMirror.Pass;
  51. if (cm.somethingSelected()) return surround(cm);
  52. if (left == right && maybeOverwrite(cm) != CodeMirror.Pass) return;
  53. var cur = cm.getCursor(), ahead = CodeMirror.Pos(cur.line, cur.ch + 1);
  54. var line = cm.getLine(cur.line), nextChar = line.charAt(cur.ch), curChar = cur.ch > 0 ? line.charAt(cur.ch - 1) : "";
  55. if (left == right && CodeMirror.isWordChar(curChar))
  56. return CodeMirror.Pass;
  57. if (line.length == cur.ch || closingBrackets.indexOf(nextChar) >= 0 || SPACE_CHAR_REGEX.test(nextChar))
  58. cm.replaceSelection(left + right, {head: ahead, anchor: ahead});
  59. else
  60. return CodeMirror.Pass;
  61. };
  62. if (left != right) map["'" + right + "'"] = maybeOverwrite;
  63. })(pairs.charAt(i), pairs.charAt(i + 1));
  64. return map;
  65. }
  66. function buildExplodeHandler(pairs) {
  67. return function(cm) {
  68. var cur = cm.getCursor(), around = charsAround(cm, cur);
  69. if (!around || pairs.indexOf(around) % 2 != 0) return CodeMirror.Pass;
  70. cm.operation(function() {
  71. var newPos = CodeMirror.Pos(cur.line + 1, 0);
  72. cm.replaceSelection("\n\n", {anchor: newPos, head: newPos}, "+input");
  73. cm.indentLine(cur.line + 1, null, true);
  74. cm.indentLine(cur.line + 2, null, true);
  75. });
  76. };
  77. }
  78. })();