hardwrap.js 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. // CodeMirror, copyright (c) by Marijn Haverbeke and others
  2. // Distributed under an MIT license: http://codemirror.net/LICENSE
  3. (function(mod) {
  4. if (typeof exports == "object" && typeof module == "object") // CommonJS
  5. mod(require("../../lib/codemirror"));
  6. else if (typeof define == "function" && define.amd) // AMD
  7. define(["../../lib/codemirror"], mod);
  8. else // Plain browser env
  9. mod(CodeMirror);
  10. })(function(CodeMirror) {
  11. "use strict";
  12. var Pos = CodeMirror.Pos;
  13. function findParagraph(cm, pos, options) {
  14. var startRE = options.paragraphStart || cm.getHelper(pos, "paragraphStart");
  15. for (var start = pos.line, first = cm.firstLine(); start > first; --start) {
  16. var line = cm.getLine(start);
  17. if (startRE && startRE.test(line)) break;
  18. if (!/\S/.test(line)) { ++start; break; }
  19. }
  20. var endRE = options.paragraphEnd || cm.getHelper(pos, "paragraphEnd");
  21. for (var end = pos.line + 1, last = cm.lastLine(); end <= last; ++end) {
  22. var line = cm.getLine(end);
  23. if (endRE && endRE.test(line)) { ++end; break; }
  24. if (!/\S/.test(line)) break;
  25. }
  26. return {from: start, to: end};
  27. }
  28. function findBreakPoint(text, column, wrapOn, killTrailingSpace) {
  29. var at = column
  30. while (at < text.length && text.charAt(at) == " ") at++
  31. for (; at > 0; --at)
  32. if (wrapOn.test(text.slice(at - 1, at + 1))) break;
  33. for (var first = true;; first = false) {
  34. var endOfText = at;
  35. if (killTrailingSpace)
  36. while (text.charAt(endOfText - 1) == " ") --endOfText;
  37. if (endOfText == 0 && first) at = column;
  38. else return {from: endOfText, to: at};
  39. }
  40. }
  41. function wrapRange(cm, from, to, options) {
  42. from = cm.clipPos(from); to = cm.clipPos(to);
  43. var column = options.column || 80;
  44. var wrapOn = options.wrapOn || /\s\S|-[^\.\d]/;
  45. var killTrailing = options.killTrailingSpace !== false;
  46. var changes = [], curLine = "", curNo = from.line;
  47. var lines = cm.getRange(from, to, false);
  48. if (!lines.length) return null;
  49. var leadingSpace = lines[0].match(/^[ \t]*/)[0];
  50. for (var i = 0; i < lines.length; ++i) {
  51. var text = lines[i], oldLen = curLine.length, spaceInserted = 0;
  52. if (curLine && text && !wrapOn.test(curLine.charAt(curLine.length - 1) + text.charAt(0))) {
  53. curLine += " ";
  54. spaceInserted = 1;
  55. }
  56. var spaceTrimmed = "";
  57. if (i) {
  58. spaceTrimmed = text.match(/^\s*/)[0];
  59. text = text.slice(spaceTrimmed.length);
  60. }
  61. curLine += text;
  62. if (i) {
  63. var firstBreak = curLine.length > column && leadingSpace == spaceTrimmed &&
  64. findBreakPoint(curLine, column, wrapOn, killTrailing);
  65. // If this isn't broken, or is broken at a different point, remove old break
  66. if (!firstBreak || firstBreak.from != oldLen || firstBreak.to != oldLen + spaceInserted) {
  67. changes.push({text: [spaceInserted ? " " : ""],
  68. from: Pos(curNo, oldLen),
  69. to: Pos(curNo + 1, spaceTrimmed.length)});
  70. } else {
  71. curLine = leadingSpace + text;
  72. ++curNo;
  73. }
  74. }
  75. while (curLine.length > column) {
  76. var bp = findBreakPoint(curLine, column, wrapOn, killTrailing);
  77. changes.push({text: ["", leadingSpace],
  78. from: Pos(curNo, bp.from),
  79. to: Pos(curNo, bp.to)});
  80. curLine = leadingSpace + curLine.slice(bp.to);
  81. ++curNo;
  82. }
  83. }
  84. if (changes.length) cm.operation(function() {
  85. for (var i = 0; i < changes.length; ++i) {
  86. var change = changes[i];
  87. if (change.text || CodeMirror.cmpPos(change.from, change.to))
  88. cm.replaceRange(change.text, change.from, change.to);
  89. }
  90. });
  91. return changes.length ? {from: changes[0].from, to: CodeMirror.changeEnd(changes[changes.length - 1])} : null;
  92. }
  93. CodeMirror.defineExtension("wrapParagraph", function(pos, options) {
  94. options = options || {};
  95. if (!pos) pos = this.getCursor();
  96. var para = findParagraph(this, pos, options);
  97. return wrapRange(this, Pos(para.from, 0), Pos(para.to - 1), options);
  98. });
  99. CodeMirror.commands.wrapLines = function(cm) {
  100. cm.operation(function() {
  101. var ranges = cm.listSelections(), at = cm.lastLine() + 1;
  102. for (var i = ranges.length - 1; i >= 0; i--) {
  103. var range = ranges[i], span;
  104. if (range.empty()) {
  105. var para = findParagraph(cm, range.head, {});
  106. span = {from: Pos(para.from, 0), to: Pos(para.to - 1)};
  107. } else {
  108. span = {from: range.from(), to: range.to()};
  109. }
  110. if (span.to.line >= at) continue;
  111. at = span.from.line;
  112. wrapRange(cm, span.from, span.to, {});
  113. }
  114. });
  115. };
  116. CodeMirror.defineExtension("wrapRange", function(from, to, options) {
  117. return wrapRange(this, from, to, options || {});
  118. });
  119. CodeMirror.defineExtension("wrapParagraphsInRange", function(from, to, options) {
  120. options = options || {};
  121. var cm = this, paras = [];
  122. for (var line = from.line; line <= to.line;) {
  123. var para = findParagraph(cm, Pos(line, 0), options);
  124. paras.push(para);
  125. line = para.to;
  126. }
  127. var madeChange = false;
  128. if (paras.length) cm.operation(function() {
  129. for (var i = paras.length - 1; i >= 0; --i)
  130. madeChange = madeChange || wrapRange(cm, Pos(paras[i].from, 0), Pos(paras[i].to - 1), options);
  131. });
  132. return madeChange;
  133. });
  134. });