codemirror-rails 2.3 → 2.21

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (71) hide show
  1. data/README.md +0 -16
  2. data/codemirror-rails.gemspec +1 -1
  3. data/lib/codemirror/rails/version.rb +2 -2
  4. data/vendor/assets/javascripts/codemirror.js +323 -687
  5. data/vendor/assets/javascripts/codemirror/modes/clike.js +3 -40
  6. data/vendor/assets/javascripts/codemirror/modes/clojure.js +14 -14
  7. data/vendor/assets/javascripts/codemirror/modes/coffeescript.js +0 -6
  8. data/vendor/assets/javascripts/codemirror/modes/css.js +1 -1
  9. data/vendor/assets/javascripts/codemirror/modes/diff.js +5 -24
  10. data/vendor/assets/javascripts/codemirror/modes/gfm.js +4 -40
  11. data/vendor/assets/javascripts/codemirror/modes/go.js +22 -20
  12. data/vendor/assets/javascripts/codemirror/modes/htmlembedded.js +1 -1
  13. data/vendor/assets/javascripts/codemirror/modes/htmlmixed.js +2 -4
  14. data/vendor/assets/javascripts/codemirror/modes/javascript.js +7 -8
  15. data/vendor/assets/javascripts/codemirror/modes/less.js +54 -100
  16. data/vendor/assets/javascripts/codemirror/modes/markdown.js +49 -52
  17. data/vendor/assets/javascripts/codemirror/modes/pascal.js +46 -2
  18. data/vendor/assets/javascripts/codemirror/modes/perl.js +1 -1
  19. data/vendor/assets/javascripts/codemirror/modes/php.js +25 -54
  20. data/vendor/assets/javascripts/codemirror/modes/python.js +16 -14
  21. data/vendor/assets/javascripts/codemirror/modes/rpm-spec.js +1 -1
  22. data/vendor/assets/javascripts/codemirror/modes/rst.js +1 -1
  23. data/vendor/assets/javascripts/codemirror/modes/ruby.js +9 -4
  24. data/vendor/assets/javascripts/codemirror/modes/scheme.js +46 -74
  25. data/vendor/assets/javascripts/codemirror/modes/smalltalk.js +16 -16
  26. data/vendor/assets/javascripts/codemirror/modes/stex.js +6 -21
  27. data/vendor/assets/javascripts/codemirror/modes/tiddlywiki.js +45 -55
  28. data/vendor/assets/javascripts/codemirror/modes/xml.js +14 -79
  29. data/vendor/assets/javascripts/codemirror/{utils/overlay.js → overlay.js} +2 -3
  30. data/vendor/assets/javascripts/codemirror/runmode.js +27 -0
  31. data/vendor/assets/stylesheets/codemirror.css +2 -63
  32. data/vendor/assets/stylesheets/codemirror/modes/tiddlywiki.css +21 -14
  33. data/vendor/assets/stylesheets/codemirror/themes/eclipse.css +1 -1
  34. data/vendor/assets/stylesheets/codemirror/themes/elegant.css +2 -2
  35. data/vendor/assets/stylesheets/codemirror/themes/neat.css +3 -3
  36. data/vendor/assets/stylesheets/codemirror/themes/night.css +1 -1
  37. data/vendor/assets/stylesheets/codemirror/themes/rubyblue.css +2 -2
  38. metadata +6 -39
  39. data/vendor/assets/javascripts/codemirror/keymaps/emacs.js +0 -29
  40. data/vendor/assets/javascripts/codemirror/keymaps/vim.js +0 -766
  41. data/vendor/assets/javascripts/codemirror/modes/ecl.js +0 -203
  42. data/vendor/assets/javascripts/codemirror/modes/erlang.js +0 -251
  43. data/vendor/assets/javascripts/codemirror/modes/pig.js +0 -172
  44. data/vendor/assets/javascripts/codemirror/modes/properties.js +0 -63
  45. data/vendor/assets/javascripts/codemirror/modes/shell.js +0 -103
  46. data/vendor/assets/javascripts/codemirror/modes/smarty.js +0 -148
  47. data/vendor/assets/javascripts/codemirror/modes/tiki.js +0 -316
  48. data/vendor/assets/javascripts/codemirror/modes/vbscript.js +0 -26
  49. data/vendor/assets/javascripts/codemirror/modes/xquery.js +0 -448
  50. data/vendor/assets/javascripts/codemirror/utils/closetag.js +0 -146
  51. data/vendor/assets/javascripts/codemirror/utils/dialog.js +0 -63
  52. data/vendor/assets/javascripts/codemirror/utils/foldcode.js +0 -196
  53. data/vendor/assets/javascripts/codemirror/utils/formatting.js +0 -297
  54. data/vendor/assets/javascripts/codemirror/utils/javascript-hint.js +0 -134
  55. data/vendor/assets/javascripts/codemirror/utils/loadmode.js +0 -51
  56. data/vendor/assets/javascripts/codemirror/utils/match-highlighter.js +0 -44
  57. data/vendor/assets/javascripts/codemirror/utils/multiplex.js +0 -72
  58. data/vendor/assets/javascripts/codemirror/utils/pig-hint.js +0 -123
  59. data/vendor/assets/javascripts/codemirror/utils/runmode.js +0 -49
  60. data/vendor/assets/javascripts/codemirror/utils/search.js +0 -118
  61. data/vendor/assets/javascripts/codemirror/utils/searchcursor.js +0 -117
  62. data/vendor/assets/javascripts/codemirror/utils/simple-hint.js +0 -72
  63. data/vendor/assets/stylesheets/codemirror/modes/tiki.css +0 -26
  64. data/vendor/assets/stylesheets/codemirror/themes/ambiance.css +0 -81
  65. data/vendor/assets/stylesheets/codemirror/themes/blackboard.css +0 -25
  66. data/vendor/assets/stylesheets/codemirror/themes/erlang-dark.css +0 -21
  67. data/vendor/assets/stylesheets/codemirror/themes/lesser-dark.css +0 -44
  68. data/vendor/assets/stylesheets/codemirror/themes/vibrant-ink.css +0 -27
  69. data/vendor/assets/stylesheets/codemirror/themes/xq-dark.css +0 -46
  70. data/vendor/assets/stylesheets/codemirror/utils/dialog.css +0 -23
  71. data/vendor/assets/stylesheets/codemirror/utils/simple-hint.css +0 -16
data/README.md CHANGED
@@ -35,22 +35,6 @@ Additional syntax modes can be added to your application.js:
35
35
  //= require codemirror/modes/ruby
36
36
  ```
37
37
 
38
- ### Adding a util
39
-
40
- Additional reusable util components can be added in your application.js:
41
-
42
- ```js
43
- //= require codemirror/utils/dialog
44
- ```
45
-
46
- ### Adding a keymap
47
-
48
- Additional keymap bindings can be added to your application.js:
49
-
50
- ```js
51
- //= require codemirror/keymaps/vim
52
- ```
53
-
54
38
  ### Adding a theme
55
39
 
56
40
  Additional CSS themes can be added to your application.css
@@ -3,7 +3,7 @@ require File.expand_path('../lib/codemirror/rails/version', __FILE__)
3
3
  Gem::Specification.new do |s|
4
4
  s.name = 'codemirror-rails'
5
5
  s.version = Codemirror::Rails::VERSION
6
- s.authors = ['Nathan Fixler', 'Robin Bühler']
6
+ s.authors = ['Nathan Fixler']
7
7
  s.email = 'nathan@fixler.org'
8
8
  s.summary = 'Use CodeMirror with Rails 3'
9
9
  s.description = 'This gem provides CodeMirror assets for your Rails 3 application.'
@@ -1,6 +1,6 @@
1
1
  module Codemirror
2
2
  module Rails
3
- VERSION = '2.3'
4
- CODEMIRROR_VERSION = '2.3'
3
+ VERSION = '2.21'
4
+ CODEMIRROR_VERSION = '2.21'
5
5
  end
6
6
  end
@@ -1,4 +1,4 @@
1
- // CodeMirror version 2.3
1
+ // CodeMirror version 2.21
2
2
  //
3
3
  // All functions that need access to the editor's state live inside
4
4
  // the CodeMirror function. Below that, at the bottom of the file,
@@ -6,7 +6,7 @@
6
6
 
7
7
  // CodeMirror is the only global var we claim
8
8
  var CodeMirror = (function() {
9
- // This is the function that produces an editor instance. Its
9
+ // This is the function that produces an editor instance. It's
10
10
  // closure is used to store the editor state.
11
11
  function CodeMirror(place, givenOptions) {
12
12
  // Determine effective options based on given values and defaults.
@@ -15,27 +15,23 @@ var CodeMirror = (function() {
15
15
  if (defaults.hasOwnProperty(opt))
16
16
  options[opt] = (givenOptions && givenOptions.hasOwnProperty(opt) ? givenOptions : defaults)[opt];
17
17
 
18
+ var targetDocument = options["document"];
18
19
  // The element in which the editor lives.
19
- var wrapper = document.createElement("div");
20
+ var wrapper = targetDocument.createElement("div");
20
21
  wrapper.className = "CodeMirror" + (options.lineWrapping ? " CodeMirror-wrap" : "");
21
22
  // This mess creates the base DOM structure for the editor.
22
23
  wrapper.innerHTML =
23
24
  '<div style="overflow: hidden; position: relative; width: 3px; height: 0px;">' + // Wraps and hides input textarea
24
25
  '<textarea style="position: absolute; padding: 0; width: 1px; height: 1em" wrap="off" ' +
25
26
  'autocorrect="off" autocapitalize="off"></textarea></div>' +
26
- '<div class="CodeMirror-scrollbar">' + // The vertical scrollbar. Horizontal scrolling is handled by the scroller itself.
27
- '<div class="CodeMirror-scrollbar-inner">' + // The empty scrollbar content, used solely for managing the scrollbar thumb.
28
- '</div></div>' + // This must be before the scroll area because it's float-right.
29
27
  '<div class="CodeMirror-scroll" tabindex="-1">' +
30
28
  '<div style="position: relative">' + // Set to the height of the text, causes scrolling
31
29
  '<div style="position: relative">' + // Moved around its parent to cover visible view
32
30
  '<div class="CodeMirror-gutter"><div class="CodeMirror-gutter-text"></div></div>' +
33
31
  // Provides positioning relative to (visible) text origin
34
32
  '<div class="CodeMirror-lines"><div style="position: relative; z-index: 0">' +
35
- // Used to measure text size
36
- '<div style="position: absolute; width: 100%; height: 0; overflow: hidden; visibility: hidden;"></div>' +
33
+ '<div style="position: absolute; width: 100%; height: 0; overflow: hidden; visibility: hidden; outline: 5px auto none"></div>' +
37
34
  '<pre class="CodeMirror-cursor">&#160;</pre>' + // Absolutely positioned blinky cursor
38
- '<pre class="CodeMirror-cursor" style="visibility: hidden">&#160;</pre>' + // Used to force a width
39
35
  '<div style="position: relative; z-index: -1"></div><div></div>' + // DIVs containing the selection and the actual code
40
36
  '</div></div></div></div></div>';
41
37
  if (place.appendChild) place.appendChild(wrapper); else place(wrapper);
@@ -44,30 +40,15 @@ var CodeMirror = (function() {
44
40
  scroller = wrapper.lastChild, code = scroller.firstChild,
45
41
  mover = code.firstChild, gutter = mover.firstChild, gutterText = gutter.firstChild,
46
42
  lineSpace = gutter.nextSibling.firstChild, measure = lineSpace.firstChild,
47
- cursor = measure.nextSibling, widthForcer = cursor.nextSibling,
48
- selectionDiv = widthForcer.nextSibling, lineDiv = selectionDiv.nextSibling,
49
- scrollbar = inputDiv.nextSibling, scrollbarInner = scrollbar.firstChild;
50
- themeChanged(); keyMapChanged();
43
+ cursor = measure.nextSibling, selectionDiv = cursor.nextSibling,
44
+ lineDiv = selectionDiv.nextSibling;
45
+ themeChanged();
51
46
  // Needed to hide big blue blinking cursor on Mobile Safari
52
47
  if (ios) input.style.width = "0px";
53
- if (!webkit) scroller.draggable = true;
48
+ if (!webkit) lineSpace.draggable = true;
54
49
  lineSpace.style.outline = "none";
55
50
  if (options.tabindex != null) input.tabIndex = options.tabindex;
56
- if (options.autofocus) focusInput();
57
51
  if (!options.gutter && !options.lineNumbers) gutter.style.display = "none";
58
- // Needed to handle Tab key in KHTML
59
- if (khtml) inputDiv.style.height = "1px", inputDiv.style.position = "absolute";
60
-
61
- // Check for OS X >= 10.7. If so, we need to force a width on the scrollbar, and
62
- // make it overlap the content. (But we only do this if the scrollbar doesn't already
63
- // have a natural width. If the mouse is plugged in or the user sets the system pref
64
- // to always show scrollbars, the scrollbar shouldn't overlap.)
65
- if (mac_geLion) {
66
- scrollbar.className += (overlapScrollbars() ? " cm-sb-overlap" : " cm-sb-nonoverlap");
67
- } else if (ie_lt8) {
68
- // Need to set a minimum width to see the scrollbar on IE7 (but must not set it on IE8).
69
- scrollbar.className += " cm-sb-ie7";
70
- }
71
52
 
72
53
  // Check for problem with IE innerHTML not working when we have a
73
54
  // P (or similar) parent node.
@@ -92,7 +73,7 @@ var CodeMirror = (function() {
92
73
  var sel = {from: {line: 0, ch: 0}, to: {line: 0, ch: 0}, inverted: false};
93
74
  // Selection-related flags. shiftSelecting obviously tracks
94
75
  // whether the user is holding shift.
95
- var shiftSelecting, lastClick, lastDoubleClick, lastScrollTop = 0, lastScrollLeft = 0, draggingText,
76
+ var shiftSelecting, lastClick, lastDoubleClick, lastScrollPos = 0, draggingText,
96
77
  overwrite = false, suppressEdits = false;
97
78
  // Variables used by startOperation/endOperation to track what
98
79
  // happened during the operation.
@@ -100,13 +81,12 @@ var CodeMirror = (function() {
100
81
  gutterDirty, callbacks;
101
82
  // Current visible range (may be bigger than the view window).
102
83
  var displayOffset = 0, showingFrom = 0, showingTo = 0, lastSizeC = 0;
103
- // bracketHighlighted is used to remember that a bracket has been
84
+ // bracketHighlighted is used to remember that a backet has been
104
85
  // marked.
105
86
  var bracketHighlighted;
106
87
  // Tracks the maximum line length so that the horizontal scrollbar
107
88
  // can be kept static when scrolling.
108
- var maxLine = "", updateMaxLine = false, maxLineChanged = true;
109
- var tabCache = {};
89
+ var maxLine = "", maxWidth, tabText = computeTabText();
110
90
 
111
91
  // Initialize the content.
112
92
  operation(function(){setValue(options.value || ""); updateInput = false;})();
@@ -115,16 +95,18 @@ var CodeMirror = (function() {
115
95
  // Register our event handlers.
116
96
  connect(scroller, "mousedown", operation(onMouseDown));
117
97
  connect(scroller, "dblclick", operation(onDoubleClick));
98
+ connect(lineSpace, "dragstart", onDragStart);
118
99
  connect(lineSpace, "selectstart", e_preventDefault);
119
100
  // Gecko browsers fire contextmenu *after* opening the menu, at
120
101
  // which point we can't mess with it anymore. Context menu is
121
102
  // handled in onMouseDown for Gecko.
122
103
  if (!gecko) connect(scroller, "contextmenu", onContextMenu);
123
- connect(scroller, "scroll", onScroll);
124
- connect(scrollbar, "scroll", onScroll);
125
- connect(scrollbar, "mousedown", function() {setTimeout(focusInput, 0);});
126
- connect(scroller, "mousewheel", onMouseWheel);
127
- connect(scroller, "DOMMouseScroll", onMouseWheel);
104
+ connect(scroller, "scroll", function() {
105
+ lastScrollPos = scroller.scrollTop;
106
+ updateDisplay([]);
107
+ if (options.fixedGutter) gutter.style.left = scroller.scrollLeft + "px";
108
+ if (options.onScroll) options.onScroll(instance);
109
+ });
128
110
  connect(window, "resize", function() {updateDisplay(true);});
129
111
  connect(input, "keyup", operation(onKeyUp));
130
112
  connect(input, "input", fastPoll);
@@ -133,32 +115,19 @@ var CodeMirror = (function() {
133
115
  connect(input, "focus", onFocus);
134
116
  connect(input, "blur", onBlur);
135
117
 
136
- if (options.dragDrop) {
137
- connect(scroller, "dragstart", onDragStart);
138
- function drag_(e) {
139
- if (options.onDragEvent && options.onDragEvent(instance, addStop(e))) return;
140
- e_stop(e);
141
- }
142
- connect(scroller, "dragenter", drag_);
143
- connect(scroller, "dragover", drag_);
144
- connect(scroller, "drop", operation(onDrop));
145
- }
118
+ connect(scroller, "dragenter", e_stop);
119
+ connect(scroller, "dragover", e_stop);
120
+ connect(scroller, "drop", operation(onDrop));
146
121
  connect(scroller, "paste", function(){focusInput(); fastPoll();});
147
122
  connect(input, "paste", fastPoll);
148
123
  connect(input, "cut", operation(function(){
149
124
  if (!options.readOnly) replaceSelection("");
150
125
  }));
151
126
 
152
- // Needed to handle Tab key in KHTML
153
- if (khtml) connect(code, "mouseup", function() {
154
- if (document.activeElement == input) input.blur();
155
- focusInput();
156
- });
157
-
158
127
  // IE throws unspecified error in certain cases, when
159
128
  // trying to access activeElement before onload
160
- var hasFocus; try { hasFocus = (document.activeElement == input); } catch(e) { }
161
- if (hasFocus || options.autofocus) setTimeout(onFocus, 20);
129
+ var hasFocus; try { hasFocus = (targetDocument.activeElement == input); } catch(e) { }
130
+ if (hasFocus) setTimeout(onFocus, 20);
162
131
  else onBlur();
163
132
 
164
133
  function isLine(l) {return l >= 0 && l < doc.size;}
@@ -172,7 +141,7 @@ var CodeMirror = (function() {
172
141
  setValue: operation(setValue),
173
142
  getSelection: getSelection,
174
143
  replaceSelection: operation(replaceSelection),
175
- focus: function(){window.focus(); focusInput(); onFocus(); fastPoll();},
144
+ focus: function(){focusInput(); onFocus(); fastPoll();},
176
145
  setOption: function(option, value) {
177
146
  var oldVal = options[option];
178
147
  options[option] = value;
@@ -181,12 +150,9 @@ var CodeMirror = (function() {
181
150
  else if (option == "readOnly" && !value) {resetInput(true);}
182
151
  else if (option == "theme") themeChanged();
183
152
  else if (option == "lineWrapping" && oldVal != value) operation(wrappingChanged)();
184
- else if (option == "tabSize") updateDisplay(true);
185
- else if (option == "keyMap") keyMapChanged();
186
- if (option == "lineNumbers" || option == "gutter" || option == "firstLineNumber" || option == "theme") {
187
- gutterChanged();
153
+ else if (option == "tabSize") operation(tabsChanged)();
154
+ if (option == "lineNumbers" || option == "gutter" || option == "firstLineNumber" || option == "theme")
188
155
  updateDisplay(true);
189
- }
190
156
  },
191
157
  getOption: function(option) {return options[option];},
192
158
  undo: operation(undo),
@@ -210,23 +176,17 @@ var CodeMirror = (function() {
210
176
  line = clipLine(line == null ? doc.size - 1: line);
211
177
  return getStateBefore(line + 1);
212
178
  },
213
- cursorCoords: function(start, mode) {
179
+ cursorCoords: function(start){
214
180
  if (start == null) start = sel.inverted;
215
- return this.charCoords(start ? sel.from : sel.to, mode);
216
- },
217
- charCoords: function(pos, mode) {
218
- pos = clipPos(pos);
219
- if (mode == "local") return localCoords(pos, false);
220
- if (mode == "div") return localCoords(pos, true);
221
- return pageCoords(pos);
181
+ return pageCoords(start ? sel.from : sel.to);
222
182
  },
183
+ charCoords: function(pos){return pageCoords(clipPos(pos));},
223
184
  coordsChar: function(coords) {
224
185
  var off = eltOffset(lineSpace);
225
186
  return coordsChar(coords.x - off.left, coords.y - off.top);
226
187
  },
227
188
  markText: operation(markText),
228
189
  setBookmark: setBookmark,
229
- findMarksAt: findMarksAt,
230
190
  setMarker: operation(addGutterMarker),
231
191
  clearMarker: operation(removeGutterMarker),
232
192
  setLineClass: operation(setLineClass),
@@ -294,21 +254,12 @@ var CodeMirror = (function() {
294
254
  replaceRange: operation(replaceRange),
295
255
  getRange: function(from, to) {return getRange(clipPos(from), clipPos(to));},
296
256
 
297
- triggerOnKeyDown: operation(onKeyDown),
298
257
  execCommand: function(cmd) {return commands[cmd](instance);},
299
258
  // Stuff used by commands, probably not much use to outside code.
300
259
  moveH: operation(moveH),
301
260
  deleteH: operation(deleteH),
302
261
  moveV: operation(moveV),
303
- toggleOverwrite: function() {
304
- if(overwrite){
305
- overwrite = false;
306
- cursor.className = cursor.className.replace(" CodeMirror-overwrite", "");
307
- } else {
308
- overwrite = true;
309
- cursor.className += " CodeMirror-overwrite";
310
- }
311
- },
262
+ toggleOverwrite: function() {overwrite = !overwrite;},
312
263
 
313
264
  posFromIndex: function(off) {
314
265
  var lineNo = 0, ch;
@@ -329,21 +280,16 @@ var CodeMirror = (function() {
329
280
  return index;
330
281
  },
331
282
  scrollTo: function(x, y) {
332
- if (x != null) scroller.scrollLeft = x;
333
- if (y != null) scrollbar.scrollTop = y;
283
+ if (x != null) scroller.scrollTop = x;
284
+ if (y != null) scroller.scrollLeft = y;
334
285
  updateDisplay([]);
335
286
  },
336
- getScrollInfo: function() {
337
- return {x: scroller.scrollLeft, y: scrollbar.scrollTop,
338
- height: scrollbar.scrollHeight, width: scroller.scrollWidth};
339
- },
340
287
 
341
288
  operation: function(f){return operation(f)();},
342
- compoundChange: function(f){return compoundChange(f);},
343
289
  refresh: function(){
344
290
  updateDisplay(true);
345
- if (scrollbar.scrollHeight > lastScrollTop)
346
- scrollbar.scrollTop = lastScrollTop;
291
+ if (scroller.scrollHeight > lastScrollPos)
292
+ scroller.scrollTop = lastScrollPos;
347
293
  },
348
294
  getInputField: function(){return input;},
349
295
  getWrapperElement: function(){return wrapper;},
@@ -364,22 +310,12 @@ var CodeMirror = (function() {
364
310
  splitLines(code), top, top);
365
311
  updateInput = true;
366
312
  }
367
- function getValue() {
313
+ function getValue(code) {
368
314
  var text = [];
369
315
  doc.iter(0, doc.size, function(line) { text.push(line.text); });
370
316
  return text.join("\n");
371
317
  }
372
318
 
373
- function onScroll(e) {
374
- if (lastScrollTop != scrollbar.scrollTop || lastScrollLeft != scroller.scrollLeft) {
375
- lastScrollTop = scrollbar.scrollTop;
376
- lastScrollLeft = scroller.scrollLeft;
377
- updateDisplay([]);
378
- if (options.fixedGutter) gutter.style.left = scroller.scrollLeft + "px";
379
- if (options.onScroll) options.onScroll(instance);
380
- }
381
- }
382
-
383
319
  function onMouseDown(e) {
384
320
  setShift(e_prop(e, "shiftKey"));
385
321
  // Check whether this is a click in a widget
@@ -402,8 +338,6 @@ var CodeMirror = (function() {
402
338
  return;
403
339
  case 2:
404
340
  if (start) setCursor(start.line, start.ch, true);
405
- setTimeout(focusInput, 20);
406
- e_preventDefault(e);
407
341
  return;
408
342
  }
409
343
  // For button 1, if it was clicked inside the editor
@@ -425,25 +359,21 @@ var CodeMirror = (function() {
425
359
  } else { lastClick = {time: now, pos: start}; }
426
360
 
427
361
  var last = start, going;
428
- if (options.dragDrop && dragAndDrop && !options.readOnly && !posEq(sel.from, sel.to) &&
362
+ if (dragAndDrop && !options.readOnly && !posEq(sel.from, sel.to) &&
429
363
  !posLess(start, sel.from) && !posLess(sel.to, start)) {
430
364
  // Let the drag handler handle this.
431
- if (webkit) scroller.draggable = true;
432
- function dragEnd(e2) {
433
- if (webkit) scroller.draggable = false;
365
+ if (webkit) lineSpace.draggable = true;
366
+ var up = connect(targetDocument, "mouseup", operation(function(e2) {
367
+ if (webkit) lineSpace.draggable = false;
434
368
  draggingText = false;
435
- up(); drop();
369
+ up();
436
370
  if (Math.abs(e.clientX - e2.clientX) + Math.abs(e.clientY - e2.clientY) < 10) {
437
371
  e_preventDefault(e2);
438
372
  setCursor(start.line, start.ch, true);
439
373
  focusInput();
440
374
  }
441
- }
442
- var up = connect(document, "mouseup", operation(dragEnd), true);
443
- var drop = connect(scroller, "drop", operation(dragEnd), true);
375
+ }), true);
444
376
  draggingText = true;
445
- // IE's approach to draggable
446
- if (scroller.dragDrop) scroller.dragDrop();
447
377
  return;
448
378
  }
449
379
  e_preventDefault(e);
@@ -462,7 +392,12 @@ var CodeMirror = (function() {
462
392
  }
463
393
  }
464
394
 
465
- function done(e) {
395
+ var move = connect(targetDocument, "mousemove", operation(function(e) {
396
+ clearTimeout(going);
397
+ e_preventDefault(e);
398
+ extend(e);
399
+ }), true);
400
+ var up = connect(targetDocument, "mouseup", operation(function(e) {
466
401
  clearTimeout(going);
467
402
  var cur = posFromMouse(e);
468
403
  if (cur) setSelectionUser(start, cur);
@@ -470,14 +405,7 @@ var CodeMirror = (function() {
470
405
  focusInput();
471
406
  updateInput = true;
472
407
  move(); up();
473
- }
474
- var move = connect(document, "mousemove", operation(function(e) {
475
- clearTimeout(going);
476
- e_preventDefault(e);
477
- if (!ie && !e_button(e)) done(e);
478
- else extend(e);
479
408
  }), true);
480
- var up = connect(document, "mouseup", operation(done), true);
481
409
  }
482
410
  function onDoubleClick(e) {
483
411
  for (var n = e_target(e); n != wrapper; n = n.parentNode)
@@ -489,7 +417,6 @@ var CodeMirror = (function() {
489
417
  selectWordAt(start);
490
418
  }
491
419
  function onDrop(e) {
492
- if (options.onDragEvent && options.onDragEvent(instance, addStop(e))) return;
493
420
  e.preventDefault();
494
421
  var pos = posFromMouse(e, true), files = e.dataTransfer.files;
495
422
  if (!pos || options.readOnly) return;
@@ -499,124 +426,86 @@ var CodeMirror = (function() {
499
426
  reader.onload = function() {
500
427
  text[i] = reader.result;
501
428
  if (++read == n) {
502
- pos = clipPos(pos);
503
- operation(function() {
429
+ pos = clipPos(pos);
430
+ operation(function() {
504
431
  var end = replaceRange(text.join(""), pos, pos);
505
432
  setSelectionUser(pos, end);
506
433
  })();
507
- }
434
+ }
508
435
  };
509
436
  reader.readAsText(file);
510
437
  }
511
438
  var n = files.length, text = Array(n), read = 0;
512
439
  for (var i = 0; i < n; ++i) loadFile(files[i], i);
513
- } else {
514
- // Don't do a replace if the drop happened inside of the selected text.
515
- if (draggingText && !(posLess(pos, sel.from) || posLess(sel.to, pos))) return;
440
+ }
441
+ else {
516
442
  try {
517
443
  var text = e.dataTransfer.getData("Text");
518
444
  if (text) {
519
- compoundChange(function() {
520
- var curFrom = sel.from, curTo = sel.to;
521
- setSelectionUser(pos, pos);
522
- if (draggingText) replaceRange("", curFrom, curTo);
523
- replaceSelection(text);
524
- focusInput();
525
- });
526
- }
445
+ var curFrom = sel.from, curTo = sel.to;
446
+ setSelectionUser(pos, pos);
447
+ if (draggingText) replaceRange("", curFrom, curTo);
448
+ replaceSelection(text);
449
+ focusInput();
450
+ }
527
451
  }
528
452
  catch(e){}
529
453
  }
530
454
  }
531
455
  function onDragStart(e) {
532
456
  var txt = getSelection();
457
+ // This will reset escapeElement
458
+ htmlEscape(txt);
459
+ e.dataTransfer.setDragImage(escapeElement, 0, 0);
533
460
  e.dataTransfer.setData("Text", txt);
534
-
535
- // Use dummy image instead of default browsers image.
536
- if (gecko || chrome || opera) {
537
- var img = document.createElement('img');
538
- img.scr = ''; //1x1 image
539
- e.dataTransfer.setDragImage(img, 0, 0);
540
- }
541
461
  }
542
-
543
- function doHandleBinding(bound, dropShift) {
462
+ function handleKeyBinding(e) {
463
+ var name = keyNames[e_prop(e, "keyCode")], next = keyMap[options.keyMap].auto, bound, dropShift;
464
+ function handleNext() {
465
+ return next.call ? next.call(null, instance) : next;
466
+ }
467
+ if (name == null || e.altGraphKey) {
468
+ if (next) options.keyMap = handleNext();
469
+ return null;
470
+ }
471
+ if (e_prop(e, "altKey")) name = "Alt-" + name;
472
+ if (e_prop(e, "ctrlKey")) name = "Ctrl-" + name;
473
+ if (e_prop(e, "metaKey")) name = "Cmd-" + name;
474
+ if (e_prop(e, "shiftKey") &&
475
+ (bound = lookupKey("Shift-" + name, options.extraKeys, options.keyMap))) {
476
+ dropShift = true;
477
+ } else {
478
+ bound = lookupKey(name, options.extraKeys, options.keyMap);
479
+ }
544
480
  if (typeof bound == "string") {
545
- bound = commands[bound];
546
- if (!bound) return false;
481
+ if (commands.propertyIsEnumerable(bound)) bound = commands[bound];
482
+ else bound = null;
547
483
  }
484
+ if (next && (bound || !isModifierKey(e))) options.keyMap = handleNext();
485
+ if (!bound) return false;
548
486
  var prevShift = shiftSelecting;
549
487
  try {
550
488
  if (options.readOnly) suppressEdits = true;
551
489
  if (dropShift) shiftSelecting = null;
552
490
  bound(instance);
553
- } catch(e) {
554
- if (e != Pass) throw e;
555
- return false;
556
491
  } finally {
557
492
  shiftSelecting = prevShift;
558
493
  suppressEdits = false;
559
494
  }
495
+ e_preventDefault(e);
560
496
  return true;
561
497
  }
562
- function handleKeyBinding(e) {
563
- // Handle auto keymap transitions
564
- var startMap = getKeyMap(options.keyMap), next = startMap.auto;
565
- clearTimeout(maybeTransition);
566
- if (next && !isModifierKey(e)) maybeTransition = setTimeout(function() {
567
- if (getKeyMap(options.keyMap) == startMap) {
568
- options.keyMap = (next.call ? next.call(null, instance) : next);
569
- }
570
- }, 50);
571
-
572
- var name = keyNames[e_prop(e, "keyCode")], handled = false;
573
- if (name == null || e.altGraphKey) return false;
574
- if (e_prop(e, "altKey")) name = "Alt-" + name;
575
- if (e_prop(e, "ctrlKey")) name = "Ctrl-" + name;
576
- if (e_prop(e, "metaKey")) name = "Cmd-" + name;
577
-
578
- var stopped = false;
579
- function stop() { stopped = true; }
580
-
581
- if (e_prop(e, "shiftKey")) {
582
- handled = lookupKey("Shift-" + name, options.extraKeys, options.keyMap,
583
- function(b) {return doHandleBinding(b, true);}, stop)
584
- || lookupKey(name, options.extraKeys, options.keyMap, function(b) {
585
- if (typeof b == "string" && /^go[A-Z]/.test(b)) return doHandleBinding(b);
586
- }, stop);
587
- } else {
588
- handled = lookupKey(name, options.extraKeys, options.keyMap, doHandleBinding, stop);
589
- }
590
- if (stopped) handled = false;
591
- if (handled) {
592
- e_preventDefault(e);
593
- restartBlink();
594
- if (ie) { e.oldKeyCode = e.keyCode; e.keyCode = 0; }
595
- }
596
- return handled;
597
- }
598
- function handleCharBinding(e, ch) {
599
- var handled = lookupKey("'" + ch + "'", options.extraKeys,
600
- options.keyMap, function(b) { return doHandleBinding(b, true); });
601
- if (handled) {
602
- e_preventDefault(e);
603
- restartBlink();
604
- }
605
- return handled;
606
- }
607
-
608
- var lastStoppedKey = null, maybeTransition;
498
+ var lastStoppedKey = null;
609
499
  function onKeyDown(e) {
610
500
  if (!focused) onFocus();
611
501
  if (ie && e.keyCode == 27) { e.returnValue = false; }
612
- if (pollingFast) { if (readInput()) pollingFast = false; }
613
502
  if (options.onKeyEvent && options.onKeyEvent(instance, addStop(e))) return;
614
503
  var code = e_prop(e, "keyCode");
615
504
  // IE does strange things with escape.
616
505
  setShift(code == 16 || e_prop(e, "shiftKey"));
617
506
  // First give onKeyEvent option a chance to handle this.
618
507
  var handled = handleKeyBinding(e);
619
- if (opera) {
508
+ if (window.opera) {
620
509
  lastStoppedKey = handled ? code : null;
621
510
  // Opera has no cut event... we try to at least catch the key combo
622
511
  if (!handled && code == 88 && e_prop(e, mac ? "metaKey" : "ctrlKey"))
@@ -624,17 +513,15 @@ var CodeMirror = (function() {
624
513
  }
625
514
  }
626
515
  function onKeyPress(e) {
627
- if (pollingFast) readInput();
628
- if (options.onKeyEvent && options.onKeyEvent(instance, addStop(e))) return;
629
516
  var keyCode = e_prop(e, "keyCode"), charCode = e_prop(e, "charCode");
630
- if (opera && keyCode == lastStoppedKey) {lastStoppedKey = null; e_preventDefault(e); return;}
631
- if (((opera && (!e.which || e.which < 10)) || khtml) && handleKeyBinding(e)) return;
632
- var ch = String.fromCharCode(charCode == null ? keyCode : charCode);
517
+ if (window.opera && keyCode == lastStoppedKey) {lastStoppedKey = null; e_preventDefault(e); return;}
518
+ if (options.onKeyEvent && options.onKeyEvent(instance, addStop(e))) return;
519
+ if (window.opera && !e.which && handleKeyBinding(e)) return;
633
520
  if (options.electricChars && mode.electricChars && options.smartIndent && !options.readOnly) {
521
+ var ch = String.fromCharCode(charCode == null ? keyCode : charCode);
634
522
  if (mode.electricChars.indexOf(ch) > -1)
635
523
  setTimeout(operation(function() {indentLine(sel.to.line, "smart");}), 75);
636
524
  }
637
- if (handleCharBinding(e, ch)) return;
638
525
  fastPoll();
639
526
  }
640
527
  function onKeyUp(e) {
@@ -647,8 +534,8 @@ var CodeMirror = (function() {
647
534
  if (!focused) {
648
535
  if (options.onFocus) options.onFocus(instance);
649
536
  focused = true;
650
- if (scroller.className.search(/\bCodeMirror-focused\b/) == -1)
651
- scroller.className += " CodeMirror-focused";
537
+ if (wrapper.className.search(/\bCodeMirror-focused\b/) == -1)
538
+ wrapper.className += " CodeMirror-focused";
652
539
  if (!leaveInputAlone) resetInput(true);
653
540
  }
654
541
  slowPoll();
@@ -662,48 +549,12 @@ var CodeMirror = (function() {
662
549
  operation(function(){
663
550
  if (bracketHighlighted) { bracketHighlighted(); bracketHighlighted = null; }
664
551
  })();
665
- scroller.className = scroller.className.replace(" CodeMirror-focused", "");
552
+ wrapper.className = wrapper.className.replace(" CodeMirror-focused", "");
666
553
  }
667
554
  clearInterval(blinker);
668
555
  setTimeout(function() {if (!focused) shiftSelecting = null;}, 150);
669
556
  }
670
557
 
671
- function chopDelta(delta) {
672
- // Make sure we always scroll a little bit for any nonzero delta.
673
- if (delta > 0.0 && delta < 1.0) return 1;
674
- else if (delta > -1.0 && delta < 0.0) return -1;
675
- else return Math.round(delta);
676
- }
677
-
678
- function onMouseWheel(e) {
679
- var deltaX = 0, deltaY = 0;
680
- if (e.type == "DOMMouseScroll") { // Firefox
681
- var delta = -e.detail * 8.0;
682
- if (e.axis == e.HORIZONTAL_AXIS) deltaX = delta;
683
- else if (e.axis == e.VERTICAL_AXIS) deltaY = delta;
684
- } else if (e.wheelDeltaX !== undefined && e.wheelDeltaY !== undefined) { // WebKit
685
- deltaX = e.wheelDeltaX / 3.0;
686
- deltaY = e.wheelDeltaY / 3.0;
687
- } else if (e.wheelDelta !== undefined) { // IE or Opera
688
- deltaY = e.wheelDelta / 3.0;
689
- }
690
-
691
- var scrolled = false;
692
- deltaX = chopDelta(deltaX);
693
- deltaY = chopDelta(deltaY);
694
- if ((deltaX > 0 && scroller.scrollLeft > 0) ||
695
- (deltaX < 0 && scroller.scrollLeft + scroller.clientWidth < scroller.scrollWidth)) {
696
- scroller.scrollLeft -= deltaX;
697
- scrolled = true;
698
- }
699
- if ((deltaY > 0 && scrollbar.scrollTop > 0) ||
700
- (deltaY < 0 && scrollbar.scrollTop + scrollbar.clientHeight < scrollbar.scrollHeight)) {
701
- scrollbar.scrollTop -= deltaY;
702
- scrolled = true;
703
- }
704
- if (scrolled) e_stop(e);
705
- }
706
-
707
558
  // Replace the range from from to to by the strings in newText.
708
559
  // Afterwards, set the selection to selFrom, selTo.
709
560
  function updateLines(from, to, newText, selFrom, selTo) {
@@ -716,30 +567,29 @@ var CodeMirror = (function() {
716
567
  }
717
568
  updateLinesNoUndo(from, to, newText, selFrom, selTo);
718
569
  }
719
- function unredoHelper(from, to) {
720
- if (!from.length) return;
721
- var set = from.pop(), out = [];
722
- for (var i = set.length - 1; i >= 0; i -= 1) {
570
+ function unredoHelper(from, to, dir) {
571
+ var set = from.pop(), len = set ? set.length : 0, out = [];
572
+ for (var i = dir > 0 ? 0 : len - 1, e = dir > 0 ? len : -1; i != e; i += dir) {
723
573
  var change = set[i];
724
574
  var replaced = [], end = change.start + change.added;
725
575
  doc.iter(change.start, end, function(line) { replaced.push(line.text); });
726
576
  out.push({start: change.start, added: change.old.length, old: replaced});
727
- var pos = {line: change.start + change.old.length - 1,
728
- ch: editEnd(replaced[replaced.length-1], change.old[change.old.length-1])};
577
+ var pos = clipPos({line: change.start + change.old.length - 1,
578
+ ch: editEnd(replaced[replaced.length-1], change.old[change.old.length-1])});
729
579
  updateLinesNoUndo({line: change.start, ch: 0}, {line: end - 1, ch: getLine(end-1).text.length}, change.old, pos, pos);
730
580
  }
731
581
  updateInput = true;
732
582
  to.push(out);
733
583
  }
734
- function undo() {unredoHelper(history.done, history.undone);}
735
- function redo() {unredoHelper(history.undone, history.done);}
584
+ function undo() {unredoHelper(history.done, history.undone, -1);}
585
+ function redo() {unredoHelper(history.undone, history.done, 1);}
736
586
 
737
587
  function updateLinesNoUndo(from, to, newText, selFrom, selTo) {
738
588
  if (suppressEdits) return;
739
589
  var recomputeMaxLength = false, maxLineLength = maxLine.length;
740
590
  if (!options.lineWrapping)
741
- doc.iter(from.line, to.line + 1, function(line) {
742
- if (!line.hidden && line.text.length == maxLineLength) {recomputeMaxLength = true; return true;}
591
+ doc.iter(from.line, to.line, function(line) {
592
+ if (line.text.length == maxLineLength) {recomputeMaxLength = true; return true;}
743
593
  });
744
594
  if (from.line != to.line || newText.length > 1) gutterDirty = true;
745
595
 
@@ -786,21 +636,29 @@ var CodeMirror = (function() {
786
636
  doc.insert(from.line + 1, added);
787
637
  }
788
638
  if (options.lineWrapping) {
789
- var perLine = Math.max(5, scroller.clientWidth / charWidth() - 3);
639
+ var perLine = scroller.clientWidth / charWidth() - 3;
790
640
  doc.iter(from.line, from.line + newText.length, function(line) {
791
641
  if (line.hidden) return;
792
642
  var guess = Math.ceil(line.text.length / perLine) || 1;
793
643
  if (guess != line.height) updateLineHeight(line, guess);
794
644
  });
795
645
  } else {
796
- doc.iter(from.line, from.line + newText.length, function(line) {
646
+ doc.iter(from.line, i + newText.length, function(line) {
797
647
  var l = line.text;
798
- if (!line.hidden && l.length > maxLineLength) {
799
- maxLine = l; maxLineLength = l.length; maxLineChanged = true;
648
+ if (l.length > maxLineLength) {
649
+ maxLine = l; maxLineLength = l.length; maxWidth = null;
800
650
  recomputeMaxLength = false;
801
651
  }
802
652
  });
803
- if (recomputeMaxLength) updateMaxLine = true;
653
+ if (recomputeMaxLength) {
654
+ maxLineLength = 0; maxLine = ""; maxWidth = null;
655
+ doc.iter(0, doc.size, function(line) {
656
+ var l = line.text;
657
+ if (l.length > maxLineLength) {
658
+ maxLineLength = l.length; maxLine = l;
659
+ }
660
+ });
661
+ }
804
662
  }
805
663
 
806
664
  // Add these lines to the work array, so that they will be
@@ -826,49 +684,11 @@ var CodeMirror = (function() {
826
684
 
827
685
  // Update the selection
828
686
  function updateLine(n) {return n <= Math.min(to.line, to.line + lendiff) ? n : n + lendiff;}
829
- setSelection(clipPos(selFrom), clipPos(selTo),
830
- updateLine(sel.from.line), updateLine(sel.to.line));
831
- }
687
+ setSelection(selFrom, selTo, updateLine(sel.from.line), updateLine(sel.to.line));
832
688
 
833
- function updateVerticalScroll(scrollTop) {
834
- var th = textHeight(), virtualHeight = Math.floor(doc.height * th + 2 * paddingTop()), scrollbarHeight = scroller.clientHeight;
835
- scrollbar.style.height = scrollbarHeight + "px";
689
+ // Make sure the scroll-size div has the correct height.
836
690
  if (scroller.clientHeight)
837
- scrollbarInner.style.height = virtualHeight + "px";
838
- // Position the mover div to align with the current virtual scroll position
839
- if (scrollTop != null) scrollbar.scrollTop = scrollTop;
840
- mover.style.top = (displayOffset * th - scrollbar.scrollTop) + "px";
841
- scrollbar.style.display = (virtualHeight > scrollbarHeight) ? "block" : "none";
842
- }
843
-
844
- // On Mac OS X Lion and up, detect whether the mouse is plugged in by measuring
845
- // the width of a div with a scrollbar in it. If the width is <= 1, then
846
- // the mouse isn't plugged in and scrollbars should overlap the content.
847
- function overlapScrollbars() {
848
- var tmpSb = document.createElement('div'),
849
- tmpSbInner = document.createElement('div');
850
- tmpSb.className = "CodeMirror-scrollbar";
851
- tmpSb.style.cssText = "position: absolute; left: -9999px; height: 100px;";
852
- tmpSbInner.className = "CodeMirror-scrollbar-inner";
853
- tmpSbInner.style.height = "200px";
854
- tmpSb.appendChild(tmpSbInner);
855
-
856
- document.body.appendChild(tmpSb);
857
- var result = (tmpSb.offsetWidth <= 1);
858
- document.body.removeChild(tmpSb);
859
- return result;
860
- }
861
-
862
- function computeMaxLength() {
863
- var maxLineLength = 0;
864
- maxLine = ""; maxLineChanged = true;
865
- doc.iter(0, doc.size, function(line) {
866
- var l = line.text;
867
- if (!line.hidden && l.length > maxLineLength) {
868
- maxLineLength = l.length; maxLine = l;
869
- }
870
- });
871
- updateMaxLine = false;
691
+ code.style.height = (doc.height * textHeight() + 2 * paddingTop()) + "px";
872
692
  }
873
693
 
874
694
  function replaceRange(code, from, to) {
@@ -957,15 +777,14 @@ var CodeMirror = (function() {
957
777
  else if (overwrite && posEq(sel.from, sel.to))
958
778
  sel.to = {line: sel.to.line, ch: Math.min(getLine(sel.to.line).text.length, sel.to.ch + (text.length - same))};
959
779
  replaceSelection(text.slice(same), "end");
960
- if (text.length > 1000) { input.value = prevInput = ""; }
961
- else prevInput = text;
780
+ prevInput = text;
962
781
  return true;
963
782
  }
964
783
  function resetInput(user) {
965
784
  if (!posEq(sel.from, sel.to)) {
966
785
  prevInput = "";
967
786
  input.value = getSelection();
968
- selectInput(input);
787
+ input.select();
969
788
  } else if (user) prevInput = input.value = "";
970
789
  }
971
790
 
@@ -979,67 +798,55 @@ var CodeMirror = (function() {
979
798
  // IE returns bogus coordinates when the instance sits inside of an iframe and the cursor is hidden
980
799
  if (ie && rect.top == rect.bottom) return;
981
800
  var winH = window.innerHeight || Math.max(document.body.offsetHeight, document.documentElement.offsetHeight);
982
- if (rect.top < 0 || rect.bottom > winH) scrollCursorIntoView();
801
+ if (rect.top < 0 || rect.bottom > winH) cursor.scrollIntoView();
983
802
  }
984
803
  function scrollCursorIntoView() {
985
- var coords = calculateCursorCoords();
986
- return scrollIntoView(coords.x, coords.y, coords.x, coords.yBot);
987
- }
988
- function calculateCursorCoords() {
989
804
  var cursor = localCoords(sel.inverted ? sel.from : sel.to);
990
805
  var x = options.lineWrapping ? Math.min(cursor.x, lineSpace.offsetWidth) : cursor.x;
991
- return {x: x, y: cursor.y, yBot: cursor.yBot};
806
+ return scrollIntoView(x, cursor.y, x, cursor.yBot);
992
807
  }
993
808
  function scrollIntoView(x1, y1, x2, y2) {
994
- var scrollPos = calculateScrollPos(x1, y1, x2, y2), scrolled = false;
995
- if (scrollPos.scrollLeft != null) {scroller.scrollLeft = scrollPos.scrollLeft; scrolled = true;}
996
- if (scrollPos.scrollTop != null) {scrollbar.scrollTop = scrollPos.scrollTop; scrolled = true;}
997
- if (scrolled && options.onScroll) options.onScroll(instance);
998
- }
999
- function calculateScrollPos(x1, y1, x2, y2) {
1000
- var pl = paddingLeft(), pt = paddingTop();
809
+ var pl = paddingLeft(), pt = paddingTop(), lh = textHeight();
1001
810
  y1 += pt; y2 += pt; x1 += pl; x2 += pl;
1002
- var screen = scroller.clientHeight, screentop = scrollbar.scrollTop, result = {};
1003
- var atTop = y1 < paddingTop() + 10;
1004
- if (y1 < screentop) result.scrollTop = atTop ? 0 : Math.max(0, y1);
1005
- else if (y2 > screentop + screen) result.scrollTop = y2 - screen;
811
+ var screen = scroller.clientHeight, screentop = scroller.scrollTop, scrolled = false, result = true;
812
+ if (y1 < screentop) {scroller.scrollTop = Math.max(0, y1 - 2*lh); scrolled = true;}
813
+ else if (y2 > screentop + screen) {scroller.scrollTop = y2 + lh - screen; scrolled = true;}
1006
814
 
1007
815
  var screenw = scroller.clientWidth, screenleft = scroller.scrollLeft;
1008
816
  var gutterw = options.fixedGutter ? gutter.clientWidth : 0;
1009
- var atLeft = x1 < gutterw + pl + 10;
1010
- if (x1 < screenleft + gutterw || atLeft) {
1011
- if (atLeft) x1 = 0;
1012
- result.scrollLeft = Math.max(0, x1 - 10 - gutterw);
1013
- } else if (x2 > screenw + screenleft - 3) {
1014
- result.scrollLeft = x2 + 10 - screenw;
817
+ if (x1 < screenleft + gutterw) {
818
+ if (x1 < 50) x1 = 0;
819
+ scroller.scrollLeft = Math.max(0, x1 - 10 - gutterw);
820
+ scrolled = true;
1015
821
  }
822
+ else if (x2 > screenw + screenleft - 3) {
823
+ scroller.scrollLeft = x2 + 10 - screenw;
824
+ scrolled = true;
825
+ if (x2 > code.clientWidth) result = false;
826
+ }
827
+ if (scrolled && options.onScroll) options.onScroll(instance);
1016
828
  return result;
1017
829
  }
1018
830
 
1019
- function visibleLines(scrollTop) {
1020
- var lh = textHeight(), top = (scrollTop != null ? scrollTop : scrollbar.scrollTop) - paddingTop();
1021
- var fromHeight = Math.max(0, Math.floor(top / lh));
1022
- var toHeight = Math.ceil((top + scroller.clientHeight) / lh);
1023
- return {from: lineAtHeight(doc, fromHeight),
1024
- to: lineAtHeight(doc, toHeight)};
831
+ function visibleLines() {
832
+ var lh = textHeight(), top = scroller.scrollTop - paddingTop();
833
+ var from_height = Math.max(0, Math.floor(top / lh));
834
+ var to_height = Math.ceil((top + scroller.clientHeight) / lh);
835
+ return {from: lineAtHeight(doc, from_height),
836
+ to: lineAtHeight(doc, to_height)};
1025
837
  }
1026
838
  // Uses a set of changes plus the current scroll position to
1027
839
  // determine which DOM updates have to be made, and makes the
1028
840
  // updates.
1029
- function updateDisplay(changes, suppressCallback, scrollTop) {
841
+ function updateDisplay(changes, suppressCallback) {
1030
842
  if (!scroller.clientWidth) {
1031
843
  showingFrom = showingTo = displayOffset = 0;
1032
844
  return;
1033
845
  }
1034
846
  // Compute the new visible window
1035
- // If scrollTop is specified, use that to determine which lines
1036
- // to render instead of the current scrollbar position.
1037
- var visible = visibleLines(scrollTop);
847
+ var visible = visibleLines();
1038
848
  // Bail out if the visible area is already rendered and nothing changed.
1039
- if (changes !== true && changes.length == 0 && visible.from > showingFrom && visible.to < showingTo) {
1040
- updateVerticalScroll(scrollTop);
1041
- return;
1042
- }
849
+ if (changes !== true && changes.length == 0 && visible.from > showingFrom && visible.to < showingTo) return;
1043
850
  var from = Math.max(visible.from - 100, 0), to = Math.min(doc.size, visible.to + 100);
1044
851
  if (showingFrom < from && from - showingFrom < 20) from = showingFrom;
1045
852
  if (showingTo > to && showingTo - to < 20) to = Math.min(doc.size, showingTo);
@@ -1057,10 +864,7 @@ var CodeMirror = (function() {
1057
864
  if (range.from >= range.to) intact.splice(i--, 1);
1058
865
  else intactLines += range.to - range.from;
1059
866
  }
1060
- if (intactLines == to - from && from == showingFrom && to == showingTo) {
1061
- updateVerticalScroll(scrollTop);
1062
- return;
1063
- }
867
+ if (intactLines == to - from) return;
1064
868
  intact.sort(function(a, b) {return a.domStart - b.domStart;});
1065
869
 
1066
870
  var th = textHeight(), gutterDisplay = gutter.style.display;
@@ -1068,12 +872,17 @@ var CodeMirror = (function() {
1068
872
  patchDisplay(from, to, intact);
1069
873
  lineDiv.style.display = gutter.style.display = "";
1070
874
 
875
+ // Position the mover div to align with the lines it's supposed
876
+ // to be showing (which will cover the visible display)
1071
877
  var different = from != showingFrom || to != showingTo || lastSizeC != scroller.clientHeight + th;
1072
878
  // This is just a bogus formula that detects when the editor is
1073
879
  // resized or the font size changes.
1074
880
  if (different) lastSizeC = scroller.clientHeight + th;
1075
881
  showingFrom = from; showingTo = to;
1076
882
  displayOffset = heightAtLine(doc, from);
883
+ mover.style.top = (displayOffset * th) + "px";
884
+ if (scroller.clientHeight)
885
+ code.style.height = (doc.height * th + 2 * paddingTop()) + "px";
1077
886
 
1078
887
  // Since this is all rather error prone, it is honoured with the
1079
888
  // only assertion in the whole file.
@@ -1081,7 +890,8 @@ var CodeMirror = (function() {
1081
890
  throw new Error("BAD PATCH! " + JSON.stringify(intact) + " size=" + (showingTo - showingFrom) +
1082
891
  " nodes=" + lineDiv.childNodes.length);
1083
892
 
1084
- function checkHeights() {
893
+ if (options.lineWrapping) {
894
+ maxWidth = scroller.clientWidth;
1085
895
  var curNode = lineDiv.firstChild, heightChanged = false;
1086
896
  doc.iter(showingFrom, showingTo, function(line) {
1087
897
  if (!line.hidden) {
@@ -1093,26 +903,22 @@ var CodeMirror = (function() {
1093
903
  }
1094
904
  curNode = curNode.nextSibling;
1095
905
  });
1096
- return heightChanged;
1097
- }
1098
-
1099
- if (options.lineWrapping) {
1100
- // Guess whether we're going to need the scrollbar, so that we don't end up changing the linewrapping
1101
- // after the scrollbar appears (during updateVerticalScroll()). Only do this if the scrollbar is
1102
- // appearing (if it's disappearing, we don't have to worry about the scroll position, and there are
1103
- // issues on IE7 if we turn it off too early).
1104
- var virtualHeight = Math.floor(doc.height * th + 2 * paddingTop()), scrollbarHeight = scroller.clientHeight;
1105
- if (virtualHeight > scrollbarHeight) scrollbar.style.display = "block";
1106
- checkHeights();
906
+ if (heightChanged)
907
+ code.style.height = (doc.height * th + 2 * paddingTop()) + "px";
908
+ } else {
909
+ if (maxWidth == null) maxWidth = stringWidth(maxLine);
910
+ if (maxWidth > scroller.clientWidth) {
911
+ lineSpace.style.width = maxWidth + "px";
912
+ // Needed to prevent odd wrapping/hiding of widgets placed in here.
913
+ code.style.width = "";
914
+ code.style.width = scroller.scrollWidth + "px";
915
+ } else {
916
+ lineSpace.style.width = code.style.width = "";
917
+ }
1107
918
  }
1108
-
1109
919
  gutter.style.display = gutterDisplay;
1110
- if (different || gutterDirty) {
1111
- // If the gutter grew in size, re-check heights. If those changed, re-draw gutter.
1112
- updateGutter() && options.lineWrapping && checkHeights() && updateGutter();
1113
- }
920
+ if (different || gutterDirty) updateGutter();
1114
921
  updateSelection();
1115
- updateVerticalScroll(scrollTop);
1116
922
  if (!suppressCallback && options.onUpdate) options.onUpdate(instance);
1117
923
  return true;
1118
924
  }
@@ -1159,17 +965,16 @@ var CodeMirror = (function() {
1159
965
  }
1160
966
  // This pass fills in the lines that actually changed.
1161
967
  var nextIntact = intact.shift(), curNode = lineDiv.firstChild, j = from;
1162
- var scratch = document.createElement("div");
968
+ var scratch = targetDocument.createElement("div"), newElt;
1163
969
  doc.iter(from, to, function(line) {
1164
970
  if (nextIntact && nextIntact.to == j) nextIntact = intact.shift();
1165
971
  if (!nextIntact || nextIntact.from > j) {
1166
972
  if (line.hidden) var html = scratch.innerHTML = "<pre></pre>";
1167
973
  else {
1168
- var html = '<pre' + (line.className ? ' class="' + line.className + '"' : '') + '>'
1169
- + line.getHTML(makeTab) + '</pre>';
974
+ var html = '<pre>' + line.getHTML(tabText) + '</pre>';
1170
975
  // Kludge to make sure the styled element lies behind the selection (by z-index)
1171
- if (line.bgClassName)
1172
- html = '<div style="position: relative"><pre class="' + line.bgClassName +
976
+ if (line.className)
977
+ html = '<div style="position: relative"><pre class="' + line.className +
1173
978
  '" style="position: absolute; left: 0; right: 0; top: 0; bottom: 0; z-index: -2">&#160;</pre>' + html + "</div>";
1174
979
  }
1175
980
  scratch.innerHTML = html;
@@ -1185,7 +990,7 @@ var CodeMirror = (function() {
1185
990
  if (!options.gutter && !options.lineNumbers) return;
1186
991
  var hText = mover.offsetHeight, hEditor = scroller.clientHeight;
1187
992
  gutter.style.height = (hText - hEditor < 2 ? hEditor : hText) + "px";
1188
- var html = [], i = showingFrom, normalNode;
993
+ var html = [], i = showingFrom;
1189
994
  doc.iter(showingFrom, Math.max(showingTo, showingFrom + 1), function(line) {
1190
995
  if (line.hidden) {
1191
996
  html.push("<pre></pre>");
@@ -1199,24 +1004,17 @@ var CodeMirror = (function() {
1199
1004
  html.push((marker && marker.style ? '<pre class="' + marker.style + '">' : "<pre>"), text);
1200
1005
  for (var j = 1; j < line.height; ++j) html.push("<br/>&#160;");
1201
1006
  html.push("</pre>");
1202
- if (!marker) normalNode = i;
1203
1007
  }
1204
1008
  ++i;
1205
1009
  });
1206
1010
  gutter.style.display = "none";
1207
1011
  gutterText.innerHTML = html.join("");
1208
- // Make sure scrolling doesn't cause number gutter size to pop
1209
- if (normalNode != null && options.lineNumbers) {
1210
- var node = gutterText.childNodes[normalNode - showingFrom];
1211
- var minwidth = String(doc.size).length, val = eltText(node.firstChild), pad = "";
1212
- while (val.length + pad.length < minwidth) pad += "\u00a0";
1213
- if (pad) node.insertBefore(document.createTextNode(pad), node.firstChild);
1214
- }
1012
+ var minwidth = String(doc.size).length, firstNode = gutterText.firstChild, val = eltText(firstNode), pad = "";
1013
+ while (val.length + pad.length < minwidth) pad += "\u00a0";
1014
+ if (pad) firstNode.insertBefore(targetDocument.createTextNode(pad), firstNode.firstChild);
1215
1015
  gutter.style.display = "";
1216
- var resized = Math.abs((parseInt(lineSpace.style.marginLeft) || 0) - gutter.offsetWidth) > 2;
1217
1016
  lineSpace.style.marginLeft = gutter.offsetWidth + "px";
1218
1017
  gutterDirty = false;
1219
- return resized;
1220
1018
  }
1221
1019
  function updateSelection() {
1222
1020
  var collapsed = posEq(sel.from, sel.to);
@@ -1233,24 +1031,20 @@ var CodeMirror = (function() {
1233
1031
  selectionDiv.style.display = "none";
1234
1032
  } else {
1235
1033
  var sameLine = fromPos.y == toPos.y, html = "";
1236
- var clientWidth = lineSpace.clientWidth || lineSpace.offsetWidth;
1237
- var clientHeight = lineSpace.clientHeight || lineSpace.offsetHeight;
1238
1034
  function add(left, top, right, height) {
1239
- var rstyle = quirksMode ? "width: " + (!right ? clientWidth : clientWidth - right - left) + "px"
1240
- : "right: " + right + "px";
1241
1035
  html += '<div class="CodeMirror-selected" style="position: absolute; left: ' + left +
1242
- 'px; top: ' + top + 'px; ' + rstyle + '; height: ' + height + 'px"></div>';
1036
+ 'px; top: ' + top + 'px; right: ' + right + 'px; height: ' + height + 'px"></div>';
1243
1037
  }
1244
1038
  if (sel.from.ch && fromPos.y >= 0) {
1245
- var right = sameLine ? clientWidth - toPos.x : 0;
1039
+ var right = sameLine ? lineSpace.clientWidth - toPos.x : 0;
1246
1040
  add(fromPos.x, fromPos.y, right, th);
1247
1041
  }
1248
1042
  var middleStart = Math.max(0, fromPos.y + (sel.from.ch ? th : 0));
1249
- var middleHeight = Math.min(toPos.y, clientHeight) - middleStart;
1043
+ var middleHeight = Math.min(toPos.y, lineSpace.clientHeight) - middleStart;
1250
1044
  if (middleHeight > 0.2 * th)
1251
1045
  add(0, middleStart, 0, middleHeight);
1252
- if ((!sameLine || !sel.from.ch) && toPos.y < clientHeight - .5 * th)
1253
- add(0, toPos.y, clientWidth - toPos.x, th);
1046
+ if ((!sameLine || !sel.from.ch) && toPos.y < lineSpace.clientHeight - .5 * th)
1047
+ add(0, toPos.y, lineSpace.clientWidth - toPos.x, th);
1254
1048
  selectionDiv.innerHTML = html;
1255
1049
  cursor.style.display = "none";
1256
1050
  selectionDiv.style.display = "";
@@ -1280,32 +1074,13 @@ var CodeMirror = (function() {
1280
1074
  if (posLess(to, from)) {var tmp = to; to = from; from = tmp;}
1281
1075
 
1282
1076
  // Skip over hidden lines.
1283
- if (from.line != oldFrom) {
1284
- var from1 = skipHidden(from, oldFrom, sel.from.ch);
1285
- // If there is no non-hidden line left, force visibility on current line
1286
- if (!from1) setLineHidden(from.line, false);
1287
- else from = from1;
1288
- }
1077
+ if (from.line != oldFrom) from = skipHidden(from, oldFrom, sel.from.ch);
1289
1078
  if (to.line != oldTo) to = skipHidden(to, oldTo, sel.to.ch);
1290
1079
 
1291
1080
  if (posEq(from, to)) sel.inverted = false;
1292
1081
  else if (posEq(from, sel.to)) sel.inverted = false;
1293
1082
  else if (posEq(to, sel.from)) sel.inverted = true;
1294
1083
 
1295
- if (options.autoClearEmptyLines && posEq(sel.from, sel.to)) {
1296
- var head = sel.inverted ? from : to;
1297
- if (head.line != sel.from.line && sel.from.line < doc.size) {
1298
- var oldLine = getLine(sel.from.line);
1299
- if (/^\s+$/.test(oldLine.text))
1300
- setTimeout(operation(function() {
1301
- if (oldLine.parent && /^\s+$/.test(oldLine.text)) {
1302
- var no = lineNo(oldLine);
1303
- replaceRange("", {line: no, ch: 0}, {line: no, ch: oldLine.text.length});
1304
- }
1305
- }, 10));
1306
- }
1307
- }
1308
-
1309
1084
  sel.from = from; sel.to = to;
1310
1085
  selectionChanged = true;
1311
1086
  }
@@ -1316,14 +1091,13 @@ var CodeMirror = (function() {
1316
1091
  var line = getLine(lNo);
1317
1092
  if (!line.hidden) {
1318
1093
  var ch = pos.ch;
1319
- if (toEnd || ch > oldCh || ch > line.text.length) ch = line.text.length;
1094
+ if (ch > oldCh || ch > line.text.length) ch = line.text.length;
1320
1095
  return {line: lNo, ch: ch};
1321
1096
  }
1322
1097
  lNo += dir;
1323
1098
  }
1324
1099
  }
1325
1100
  var line = getLine(pos.line);
1326
- var toEnd = pos.ch == line.text.length && pos.ch != oldCh;
1327
1101
  if (!line.hidden) return pos;
1328
1102
  if (pos.line >= oldLine) return getNonHidden(1) || getNonHidden(-1);
1329
1103
  else return getNonHidden(-1) || getNonHidden(1);
@@ -1390,7 +1164,6 @@ var CodeMirror = (function() {
1390
1164
  if (unit == "page") dist = Math.min(scroller.clientHeight, window.innerHeight || document.documentElement.clientHeight);
1391
1165
  else if (unit == "line") dist = textHeight();
1392
1166
  var target = coordsChar(pos.x, pos.y + dist * dir + 2);
1393
- if (unit == "page") scrollbar.scrollTop += localCoords(target, true).y - pos.y;
1394
1167
  setCursor(target.line, target.ch, true);
1395
1168
  goalColumn = pos.x;
1396
1169
  }
@@ -1403,7 +1176,7 @@ var CodeMirror = (function() {
1403
1176
  setSelectionUser({line: pos.line, ch: start}, {line: pos.line, ch: end});
1404
1177
  }
1405
1178
  function selectLine(line) {
1406
- setSelectionUser({line: line, ch: 0}, clipPos({line: line + 1, ch: 0}));
1179
+ setSelectionUser({line: line, ch: 0}, {line: line, ch: getLine(line).text.length});
1407
1180
  }
1408
1181
  function indentSelected(mode) {
1409
1182
  if (posEq(sel.from, sel.to)) return indentLine(sel.from.line, mode);
@@ -1420,14 +1193,11 @@ var CodeMirror = (function() {
1420
1193
 
1421
1194
  var line = getLine(n), curSpace = line.indentation(options.tabSize),
1422
1195
  curSpaceString = line.text.match(/^\s*/)[0], indentation;
1423
- if (how == "smart") {
1424
- indentation = mode.indent(state, line.text.slice(curSpaceString.length), line.text);
1425
- if (indentation == Pass) how = "prev";
1426
- }
1427
1196
  if (how == "prev") {
1428
1197
  if (n) indentation = getLine(n-1).indentation(options.tabSize);
1429
1198
  else indentation = 0;
1430
1199
  }
1200
+ else if (how == "smart") indentation = mode.indent(state, line.text.slice(curSpaceString.length), line.text);
1431
1201
  else if (how == "add") indentation = curSpace + options.indentUnit;
1432
1202
  else if (how == "subtract") indentation = curSpace - options.indentUnit;
1433
1203
  indentation = Math.max(0, indentation);
@@ -1436,7 +1206,8 @@ var CodeMirror = (function() {
1436
1206
  if (!diff) {
1437
1207
  if (sel.from.line != n && sel.to.line != n) return;
1438
1208
  var indentString = curSpaceString;
1439
- } else {
1209
+ }
1210
+ else {
1440
1211
  var indentString = "", pos = 0;
1441
1212
  if (options.indentWithTabs)
1442
1213
  for (var i = Math.floor(indentation / options.tabSize); i; --i) {pos += options.tabSize; indentString += "\t";}
@@ -1468,10 +1239,9 @@ var CodeMirror = (function() {
1468
1239
  if (guess != 1) updateLineHeight(line, guess);
1469
1240
  });
1470
1241
  lineSpace.style.width = code.style.width = "";
1471
- widthForcer.style.left = "";
1472
1242
  } else {
1473
1243
  wrapper.className = wrapper.className.replace(" CodeMirror-wrap", "");
1474
- maxLine = ""; maxLineChanged = true;
1244
+ maxWidth = null; maxLine = "";
1475
1245
  doc.iter(0, doc.size, function(line) {
1476
1246
  if (line.height != 1 && !line.hidden) updateLineHeight(line, 1);
1477
1247
  if (line.text.length > maxLine.length) maxLine = line.text;
@@ -1479,21 +1249,18 @@ var CodeMirror = (function() {
1479
1249
  }
1480
1250
  changes.push({from: 0, to: doc.size});
1481
1251
  }
1482
- function makeTab(col) {
1483
- var w = options.tabSize - col % options.tabSize, cached = tabCache[w];
1484
- if (cached) return cached;
1485
- for (var str = '<span class="cm-tab">', i = 0; i < w; ++i) str += " ";
1486
- return (tabCache[w] = {html: str + "</span>", width: w});
1252
+ function computeTabText() {
1253
+ for (var str = '<span class="cm-tab">', i = 0; i < options.tabSize; ++i) str += " ";
1254
+ return str + "</span>";
1255
+ }
1256
+ function tabsChanged() {
1257
+ tabText = computeTabText();
1258
+ updateDisplay(true);
1487
1259
  }
1488
1260
  function themeChanged() {
1489
- scroller.className = scroller.className.replace(/\s*cm-s-\S+/g, "") +
1261
+ scroller.className = scroller.className.replace(/\s*cm-s-\w+/g, "") +
1490
1262
  options.theme.replace(/(^|\s)\s*/g, " cm-s-");
1491
1263
  }
1492
- function keyMapChanged() {
1493
- var style = keyMap[options.keyMap].style;
1494
- wrapper.className = wrapper.className.replace(/\s*cm-keymap-\S+/g, "") +
1495
- (style ? " cm-keymap-" + style : "");
1496
- }
1497
1264
 
1498
1265
  function TextMarker() { this.set = []; }
1499
1266
  TextMarker.prototype.clear = operation(function() {
@@ -1504,7 +1271,7 @@ var CodeMirror = (function() {
1504
1271
  var lineN = lineNo(line);
1505
1272
  min = Math.min(min, lineN); max = Math.max(max, lineN);
1506
1273
  for (var j = 0; j < mk.length; ++j)
1507
- if (mk[j].marker == this) mk.splice(j--, 1);
1274
+ if (mk[j].set == this.set) mk.splice(j--, 1);
1508
1275
  }
1509
1276
  if (min != Infinity)
1510
1277
  changes.push({from: min, to: max + 1});
@@ -1515,7 +1282,7 @@ var CodeMirror = (function() {
1515
1282
  var line = this.set[i], mk = line.marked;
1516
1283
  for (var j = 0; j < mk.length; ++j) {
1517
1284
  var mark = mk[j];
1518
- if (mark.marker == this) {
1285
+ if (mark.set == this.set) {
1519
1286
  if (mark.from != null || mark.to != null) {
1520
1287
  var found = lineNo(line);
1521
1288
  if (found != null) {
@@ -1532,9 +1299,8 @@ var CodeMirror = (function() {
1532
1299
  function markText(from, to, className) {
1533
1300
  from = clipPos(from); to = clipPos(to);
1534
1301
  var tm = new TextMarker();
1535
- if (!posLess(from, to)) return tm;
1536
1302
  function add(line, from, to, className) {
1537
- getLine(line).addMark(new MarkedText(from, to, className, tm));
1303
+ getLine(line).addMark(new MarkedText(from, to, className, tm.set));
1538
1304
  }
1539
1305
  if (from.line == to.line) add(from.line, from.ch, to.ch, className);
1540
1306
  else {
@@ -1554,19 +1320,6 @@ var CodeMirror = (function() {
1554
1320
  return bm;
1555
1321
  }
1556
1322
 
1557
- function findMarksAt(pos) {
1558
- pos = clipPos(pos);
1559
- var markers = [], marked = getLine(pos.line).marked;
1560
- if (!marked) return markers;
1561
- for (var i = 0, e = marked.length; i < e; ++i) {
1562
- var m = marked[i];
1563
- if ((m.from == null || m.from <= pos.ch) &&
1564
- (m.to == null || m.to >= pos.ch))
1565
- markers.push(m.marker || m);
1566
- }
1567
- return markers;
1568
- }
1569
-
1570
1323
  function addGutterMarker(line, text, className) {
1571
1324
  if (typeof line == "number") line = getLine(clipLine(line));
1572
1325
  line.gutterMarker = {text: text, style: className};
@@ -1588,11 +1341,10 @@ var CodeMirror = (function() {
1588
1341
  else return null;
1589
1342
  return line;
1590
1343
  }
1591
- function setLineClass(handle, className, bgClassName) {
1344
+ function setLineClass(handle, className) {
1592
1345
  return changeLine(handle, function(line) {
1593
- if (line.className != className || line.bgClassName != bgClassName) {
1346
+ if (line.className != className) {
1594
1347
  line.className = className;
1595
- line.bgClassName = bgClassName;
1596
1348
  return true;
1597
1349
  }
1598
1350
  });
@@ -1601,21 +1353,11 @@ var CodeMirror = (function() {
1601
1353
  return changeLine(handle, function(line, no) {
1602
1354
  if (line.hidden != hidden) {
1603
1355
  line.hidden = hidden;
1604
- if (!options.lineWrapping) {
1605
- var l = line.text;
1606
- if (hidden && l.length == maxLine.length) {
1607
- updateMaxLine = true;
1608
- } else if (!hidden && l.length > maxLine.length) {
1609
- maxLine = l; maxWidth = null; updateMaxLine = false;
1610
- }
1611
- }
1612
1356
  updateLineHeight(line, hidden ? 0 : 1);
1613
1357
  var fline = sel.from.line, tline = sel.to.line;
1614
1358
  if (hidden && (fline == no || tline == no)) {
1615
1359
  var from = fline == no ? skipHidden({line: fline, ch: 0}, fline, 0) : sel.from;
1616
1360
  var to = tline == no ? skipHidden({line: tline, ch: 0}, tline, 0) : sel.to;
1617
- // Can't hide the last visible line, we'd have no place to put the cursor
1618
- if (!to) return;
1619
1361
  setSelection(from, to);
1620
1362
  }
1621
1363
  return (gutterDirty = true);
@@ -1629,13 +1371,14 @@ var CodeMirror = (function() {
1629
1371
  var n = line;
1630
1372
  line = getLine(line);
1631
1373
  if (!line) return null;
1632
- } else {
1374
+ }
1375
+ else {
1633
1376
  var n = lineNo(line);
1634
1377
  if (n == null) return null;
1635
1378
  }
1636
1379
  var marker = line.gutterMarker;
1637
1380
  return {line: n, handle: line, text: line.text, markerText: marker && marker.text,
1638
- markerClass: marker && marker.style, lineClass: line.className, bgClass: line.bgClassName};
1381
+ markerClass: marker && marker.style, lineClass: line.className};
1639
1382
  }
1640
1383
 
1641
1384
  function stringWidth(str) {
@@ -1649,7 +1392,8 @@ var CodeMirror = (function() {
1649
1392
  if (x <= 0) return 0;
1650
1393
  var lineObj = getLine(line), text = lineObj.text;
1651
1394
  function getX(len) {
1652
- return measureLine(lineObj, len).left;
1395
+ measure.innerHTML = "<pre><span>" + lineObj.getHTML(tabText, len) + "</span></pre>";
1396
+ return measure.firstChild.firstChild.offsetWidth;
1653
1397
  }
1654
1398
  var from = 0, fromX = 0, to = text.length, toX;
1655
1399
  // Guess a suitable upper bound for our search.
@@ -1672,13 +1416,19 @@ var CodeMirror = (function() {
1672
1416
  }
1673
1417
  }
1674
1418
 
1675
- var tempId = "CodeMirror-temp-" + Math.floor(Math.random() * 0xffffff).toString(16);
1419
+ var tempId = Math.floor(Math.random() * 0xffffff).toString(16);
1676
1420
  function measureLine(line, ch) {
1677
1421
  if (ch == 0) return {top: 0, left: 0};
1678
- var wbr = options.lineWrapping && ch < line.text.length &&
1679
- spanAffectsWrapping.test(line.text.slice(ch - 1, ch + 1));
1680
- measure.innerHTML = "<pre>" + line.getHTML(makeTab, ch, tempId, wbr) + "</pre>";
1681
- var elt = document.getElementById(tempId);
1422
+ var extra = "";
1423
+ // Include extra text at the end to make sure the measured line is wrapped in the right way.
1424
+ if (options.lineWrapping) {
1425
+ var end = line.text.indexOf(" ", ch + 2);
1426
+ extra = htmlEscape(line.text.slice(ch + 1, end < 0 ? line.text.length : end + (ie ? 5 : 0)));
1427
+ }
1428
+ measure.innerHTML = "<pre>" + line.getHTML(tabText, ch) +
1429
+ '<span id="CodeMirror-temp-' + tempId + '">' + htmlEscape(line.text.charAt(ch) || " ") + "</span>" +
1430
+ extra + "</pre>";
1431
+ var elt = document.getElementById("CodeMirror-temp-" + tempId);
1682
1432
  var top = elt.offsetTop, left = elt.offsetLeft;
1683
1433
  // Older IEs report zero offsets for spans directly after a wrap
1684
1434
  if (ie && top == 0 && left == 0) {
@@ -1778,8 +1528,8 @@ var CodeMirror = (function() {
1778
1528
  return coordsChar(x - offL.left, y - offL.top);
1779
1529
  }
1780
1530
  function onContextMenu(e) {
1781
- var pos = posFromMouse(e), scrollPos = scrollbar.scrollTop;
1782
- if (!pos || opera) return; // Opera is difficult.
1531
+ var pos = posFromMouse(e);
1532
+ if (!pos || window.opera) return; // Opera is difficult.
1783
1533
  if (posEq(sel.from, sel.to) || posLess(pos, sel.from) || !posLess(pos, sel.to))
1784
1534
  operation(setCursor)(pos.line, pos.ch);
1785
1535
 
@@ -1791,13 +1541,12 @@ var CodeMirror = (function() {
1791
1541
  leaveInputAlone = true;
1792
1542
  var val = input.value = getSelection();
1793
1543
  focusInput();
1794
- selectInput(input);
1544
+ input.select();
1795
1545
  function rehide() {
1796
1546
  var newVal = splitLines(input.value).join("\n");
1797
- if (newVal != val && !options.readOnly) operation(replaceSelection)(newVal, "end");
1547
+ if (newVal != val) operation(replaceSelection)(newVal, "end");
1798
1548
  inputDiv.style.position = "relative";
1799
1549
  input.style.cssText = oldCSS;
1800
- if (ie_lt9) scrollbar.scrollTop = scrollPos;
1801
1550
  leaveInputAlone = false;
1802
1551
  resetInput(true);
1803
1552
  slowPoll();
@@ -1809,7 +1558,8 @@ var CodeMirror = (function() {
1809
1558
  mouseup();
1810
1559
  setTimeout(rehide, 20);
1811
1560
  }, true);
1812
- } else {
1561
+ }
1562
+ else {
1813
1563
  setTimeout(rehide, 50);
1814
1564
  }
1815
1565
  }
@@ -1839,7 +1589,7 @@ var CodeMirror = (function() {
1839
1589
  var st = line.styles, pos = forward ? 0 : line.text.length - 1, cur;
1840
1590
  for (var i = forward ? 0 : st.length - 2, e = forward ? st.length : -2; i != e; i += 2*d) {
1841
1591
  var text = st[i];
1842
- if (st[i+1] != style) {pos += d * text.length; continue;}
1592
+ if (st[i+1] != null && st[i+1] != style) {pos += d * text.length; continue;}
1843
1593
  for (var j = forward ? 0 : text.length - 1, te = forward ? text.length : -1; j != te; j += d, pos+=d) {
1844
1594
  if (pos >= from && pos < to && re.test(cur = text.charAt(j))) {
1845
1595
  var match = matching[cur];
@@ -1926,17 +1676,13 @@ var CodeMirror = (function() {
1926
1676
  var changed = line.highlight(mode, state, options.tabSize);
1927
1677
  if (changed) realChange = true;
1928
1678
  line.stateAfter = copyState(mode, state);
1929
- var done = null;
1930
1679
  if (compare) {
1931
- var same = hadState && compare(hadState, state);
1932
- if (same != Pass) done = !!same;
1933
- }
1934
- if (done == null) {
1680
+ if (hadState && compare(hadState, state)) return true;
1681
+ } else {
1935
1682
  if (changed !== false || !hadState) unchanged = 0;
1936
1683
  else if (++unchanged > 3 && (!mode.indent || mode.indent(hadState, "") == mode.indent(state, "")))
1937
- done = true;
1684
+ return true;
1938
1685
  }
1939
- if (done) return true;
1940
1686
  ++i;
1941
1687
  });
1942
1688
  if (bail) return;
@@ -1959,22 +1705,14 @@ var CodeMirror = (function() {
1959
1705
  changes = []; selectionChanged = false; callbacks = [];
1960
1706
  }
1961
1707
  function endOperation() {
1962
- if (updateMaxLine) computeMaxLength();
1963
- if (maxLineChanged && !options.lineWrapping) {
1964
- widthForcer.style.left = stringWidth(maxLine) + "px";
1965
- maxLineChanged = false;
1966
- }
1967
- var newScrollPos, updated;
1968
- if (selectionChanged) {
1969
- var coords = calculateCursorCoords();
1970
- newScrollPos = calculateScrollPos(coords.x, coords.y, coords.x, coords.yBot);
1971
- }
1972
- if (changes.length) updated = updateDisplay(changes, true, (newScrollPos ? newScrollPos.scrollTop : null));
1708
+ var reScroll = false, updated;
1709
+ if (selectionChanged) reScroll = !scrollCursorIntoView();
1710
+ if (changes.length) updated = updateDisplay(changes, true);
1973
1711
  else {
1974
1712
  if (selectionChanged) updateSelection();
1975
1713
  if (gutterDirty) updateGutter();
1976
1714
  }
1977
- if (newScrollPos) scrollCursorIntoView();
1715
+ if (reScroll) scrollCursorIntoView();
1978
1716
  if (selectionChanged) {scrollEditorIntoView(); restartBlink();}
1979
1717
 
1980
1718
  if (focused && !leaveInputAlone &&
@@ -1986,11 +1724,11 @@ var CodeMirror = (function() {
1986
1724
  if (bracketHighlighted) {bracketHighlighted(); bracketHighlighted = null;}
1987
1725
  if (posEq(sel.from, sel.to)) matchBrackets(false);
1988
1726
  }), 20);
1989
- var sc = selectionChanged, cbs = callbacks; // these can be reset by callbacks
1990
- if (textChanged && options.onChange && instance)
1991
- options.onChange(instance, textChanged);
1992
- if (sc && options.onCursorActivity)
1727
+ var tc = textChanged, cbs = callbacks; // these can be reset by callbacks
1728
+ if (selectionChanged && options.onCursorActivity)
1993
1729
  options.onCursorActivity(instance);
1730
+ if (tc && options.onChange && instance)
1731
+ options.onChange(instance, tc);
1994
1732
  for (var i = 0; i < cbs.length; ++i) cbs[i](instance);
1995
1733
  if (updated && options.onUpdate) options.onUpdate(instance);
1996
1734
  }
@@ -2004,11 +1742,6 @@ var CodeMirror = (function() {
2004
1742
  };
2005
1743
  }
2006
1744
 
2007
- function compoundChange(f) {
2008
- history.startCompound();
2009
- try { return f(); } finally { history.endCompound(); }
2010
- }
2011
-
2012
1745
  for (var ext in extensions)
2013
1746
  if (extensions.propertyIsEnumerable(ext) &&
2014
1747
  !instance.propertyIsEnumerable(ext))
@@ -2028,16 +1761,13 @@ var CodeMirror = (function() {
2028
1761
  keyMap: "default",
2029
1762
  extraKeys: null,
2030
1763
  electricChars: true,
2031
- autoClearEmptyLines: false,
2032
1764
  onKeyEvent: null,
2033
- onDragEvent: null,
2034
1765
  lineWrapping: false,
2035
1766
  lineNumbers: false,
2036
1767
  gutter: false,
2037
1768
  fixedGutter: false,
2038
1769
  firstLineNumber: 1,
2039
1770
  readOnly: false,
2040
- dragDrop: true,
2041
1771
  onChange: null,
2042
1772
  onCursorActivity: null,
2043
1773
  onGutterClick: null,
@@ -2050,7 +1780,7 @@ var CodeMirror = (function() {
2050
1780
  pollInterval: 100,
2051
1781
  undoDepth: 40,
2052
1782
  tabindex: null,
2053
- autofocus: null
1783
+ document: window.document
2054
1784
  };
2055
1785
 
2056
1786
  var ios = /AppleWebKit/.test(navigator.userAgent) && /Mobile\/\w+/.test(navigator.userAgent);
@@ -2058,31 +1788,27 @@ var CodeMirror = (function() {
2058
1788
  var win = /Win/.test(navigator.platform);
2059
1789
 
2060
1790
  // Known modes, by name and by MIME
2061
- var modes = CodeMirror.modes = {}, mimeModes = CodeMirror.mimeModes = {};
1791
+ var modes = {}, mimeModes = {};
2062
1792
  CodeMirror.defineMode = function(name, mode) {
2063
1793
  if (!CodeMirror.defaults.mode && name != "null") CodeMirror.defaults.mode = name;
2064
- if (arguments.length > 2) {
2065
- mode.dependencies = [];
2066
- for (var i = 2; i < arguments.length; ++i) mode.dependencies.push(arguments[i]);
2067
- }
2068
1794
  modes[name] = mode;
2069
1795
  };
2070
1796
  CodeMirror.defineMIME = function(mime, spec) {
2071
1797
  mimeModes[mime] = spec;
2072
1798
  };
2073
- CodeMirror.resolveMode = function(spec) {
1799
+ CodeMirror.getMode = function(options, spec) {
2074
1800
  if (typeof spec == "string" && mimeModes.hasOwnProperty(spec))
2075
1801
  spec = mimeModes[spec];
2076
- else if (typeof spec == "string" && /^[\w\-]+\/[\w\-]+\+xml$/.test(spec))
2077
- return CodeMirror.resolveMode("application/xml");
2078
- if (typeof spec == "string") return {name: spec};
2079
- else return spec || {name: "null"};
2080
- };
2081
- CodeMirror.getMode = function(options, spec) {
2082
- var spec = CodeMirror.resolveMode(spec);
2083
- var mfactory = modes[spec.name];
2084
- if (!mfactory) return CodeMirror.getMode(options, "text/plain");
2085
- return mfactory(options, spec);
1802
+ if (typeof spec == "string")
1803
+ var mname = spec, config = {};
1804
+ else if (spec != null)
1805
+ var mname = spec.name, config = spec;
1806
+ var mfactory = modes[mname];
1807
+ if (!mfactory) {
1808
+ if (window.console) console.warn("No mode " + mname + " found, falling back to plain text.");
1809
+ return CodeMirror.getMode(options, "text/plain");
1810
+ }
1811
+ return mfactory(options, config || {});
2086
1812
  };
2087
1813
  CodeMirror.listModes = function() {
2088
1814
  var list = [];
@@ -2139,10 +1865,6 @@ var CodeMirror = (function() {
2139
1865
  indentMore: function(cm) {cm.indentSelection("add");},
2140
1866
  indentLess: function(cm) {cm.indentSelection("subtract");},
2141
1867
  insertTab: function(cm) {cm.replaceSelection("\t", "end");},
2142
- defaultTab: function(cm) {
2143
- if (cm.somethingSelected()) cm.indentSelection("add");
2144
- else cm.replaceSelection("\t", "end");
2145
- },
2146
1868
  transposeChars: function(cm) {
2147
1869
  var cur = cm.getCursor(), line = cm.getLine(cur.line);
2148
1870
  if (cur.ch > 0 && cur.ch < line.length - 1)
@@ -2160,7 +1882,7 @@ var CodeMirror = (function() {
2160
1882
  keyMap.basic = {
2161
1883
  "Left": "goCharLeft", "Right": "goCharRight", "Up": "goLineUp", "Down": "goLineDown",
2162
1884
  "End": "goLineEnd", "Home": "goLineStartSmart", "PageUp": "goPageUp", "PageDown": "goPageDown",
2163
- "Delete": "delCharRight", "Backspace": "delCharLeft", "Tab": "defaultTab", "Shift-Tab": "indentAuto",
1885
+ "Delete": "delCharRight", "Backspace": "delCharLeft", "Tab": "indentMore", "Shift-Tab": "indentLess",
2164
1886
  "Enter": "newlineAndIndent", "Insert": "toggleOverwrite"
2165
1887
  };
2166
1888
  // Note that the save and find-related commands aren't defined by
@@ -2171,7 +1893,6 @@ var CodeMirror = (function() {
2171
1893
  "Ctrl-Left": "goWordLeft", "Ctrl-Right": "goWordRight", "Alt-Left": "goLineStart", "Alt-Right": "goLineEnd",
2172
1894
  "Ctrl-Backspace": "delWordLeft", "Ctrl-Delete": "delWordRight", "Ctrl-S": "save", "Ctrl-F": "find",
2173
1895
  "Ctrl-G": "findNext", "Shift-Ctrl-G": "findPrev", "Shift-Ctrl-F": "replace", "Shift-Ctrl-R": "replaceAll",
2174
- "Ctrl-[": "indentLess", "Ctrl-]": "indentMore",
2175
1896
  fallthrough: "basic"
2176
1897
  };
2177
1898
  keyMap.macDefault = {
@@ -2180,7 +1901,6 @@ var CodeMirror = (function() {
2180
1901
  "Alt-Right": "goWordRight", "Cmd-Left": "goLineStart", "Cmd-Right": "goLineEnd", "Alt-Backspace": "delWordLeft",
2181
1902
  "Ctrl-Alt-Backspace": "delWordRight", "Alt-Delete": "delWordRight", "Cmd-S": "save", "Cmd-F": "find",
2182
1903
  "Cmd-G": "findNext", "Shift-Cmd-G": "findPrev", "Cmd-Alt-F": "replace", "Shift-Cmd-Alt-F": "replaceAll",
2183
- "Cmd-[": "indentLess", "Cmd-]": "indentMore",
2184
1904
  fallthrough: ["basic", "emacsy"]
2185
1905
  };
2186
1906
  keyMap["default"] = mac ? keyMap.macDefault : keyMap.pcDefault;
@@ -2191,30 +1911,20 @@ var CodeMirror = (function() {
2191
1911
  "Alt-D": "delWordRight", "Alt-Backspace": "delWordLeft", "Ctrl-K": "killLine", "Ctrl-T": "transposeChars"
2192
1912
  };
2193
1913
 
2194
- function getKeyMap(val) {
2195
- if (typeof val == "string") return keyMap[val];
2196
- else return val;
2197
- }
2198
- function lookupKey(name, extraMap, map, handle, stop) {
2199
- function lookup(map) {
2200
- map = getKeyMap(map);
1914
+ function lookupKey(name, extraMap, map) {
1915
+ function lookup(name, map, ft) {
2201
1916
  var found = map[name];
2202
- if (found != null && handle(found)) return true;
2203
- if (map.nofallthrough) {
2204
- if (stop) stop();
2205
- return true;
1917
+ if (found != null) return found;
1918
+ if (ft == null) ft = map.fallthrough;
1919
+ if (ft == null) return map.catchall;
1920
+ if (typeof ft == "string") return lookup(name, keyMap[ft]);
1921
+ for (var i = 0, e = ft.length; i < e; ++i) {
1922
+ found = lookup(name, keyMap[ft[i]]);
1923
+ if (found != null) return found;
2206
1924
  }
2207
- var fallthrough = map.fallthrough;
2208
- if (fallthrough == null) return false;
2209
- if (Object.prototype.toString.call(fallthrough) != "[object Array]")
2210
- return lookup(fallthrough);
2211
- for (var i = 0, e = fallthrough.length; i < e; ++i) {
2212
- if (lookup(fallthrough[i])) return true;
2213
- }
2214
- return false;
1925
+ return null;
2215
1926
  }
2216
- if (extraMap && lookup(extraMap)) return true;
2217
- return lookup(map);
1927
+ return extraMap ? lookup(name, extraMap, map) : lookup(name, keyMap[map]);
2218
1928
  }
2219
1929
  function isModifierKey(event) {
2220
1930
  var name = keyNames[e_prop(event, "keyCode")];
@@ -2226,8 +1936,6 @@ var CodeMirror = (function() {
2226
1936
  options.value = textarea.value;
2227
1937
  if (!options.tabindex && textarea.tabindex)
2228
1938
  options.tabindex = textarea.tabindex;
2229
- if (options.autofocus == null && textarea.getAttribute("autofocus") != null)
2230
- options.autofocus = true;
2231
1939
 
2232
1940
  function save() {textarea.value = instance.getValue();}
2233
1941
  if (textarea.form) {
@@ -2328,7 +2036,8 @@ var CodeMirror = (function() {
2328
2036
  if (consume !== false) this.pos += pattern.length;
2329
2037
  return true;
2330
2038
  }
2331
- } else {
2039
+ }
2040
+ else {
2332
2041
  var match = this.string.slice(this.pos).match(pattern);
2333
2042
  if (match && consume !== false) this.pos += match[0].length;
2334
2043
  return match;
@@ -2338,34 +2047,34 @@ var CodeMirror = (function() {
2338
2047
  };
2339
2048
  CodeMirror.StringStream = StringStream;
2340
2049
 
2341
- function MarkedText(from, to, className, marker) {
2342
- this.from = from; this.to = to; this.style = className; this.marker = marker;
2050
+ function MarkedText(from, to, className, set) {
2051
+ this.from = from; this.to = to; this.style = className; this.set = set;
2343
2052
  }
2344
2053
  MarkedText.prototype = {
2345
- attach: function(line) { this.marker.set.push(line); },
2054
+ attach: function(line) { this.set.push(line); },
2346
2055
  detach: function(line) {
2347
- var ix = indexOf(this.marker.set, line);
2348
- if (ix > -1) this.marker.set.splice(ix, 1);
2056
+ var ix = indexOf(this.set, line);
2057
+ if (ix > -1) this.set.splice(ix, 1);
2349
2058
  },
2350
2059
  split: function(pos, lenBefore) {
2351
2060
  if (this.to <= pos && this.to != null) return null;
2352
2061
  var from = this.from < pos || this.from == null ? null : this.from - pos + lenBefore;
2353
2062
  var to = this.to == null ? null : this.to - pos + lenBefore;
2354
- return new MarkedText(from, to, this.style, this.marker);
2063
+ return new MarkedText(from, to, this.style, this.set);
2355
2064
  },
2356
- dup: function() { return new MarkedText(null, null, this.style, this.marker); },
2065
+ dup: function() { return new MarkedText(null, null, this.style, this.set); },
2357
2066
  clipTo: function(fromOpen, from, toOpen, to, diff) {
2067
+ if (this.from != null && this.from >= from)
2068
+ this.from = Math.max(to, this.from) + diff;
2069
+ if (this.to != null && this.to > from)
2070
+ this.to = to < this.to ? this.to + diff : from;
2358
2071
  if (fromOpen && to > this.from && (to < this.to || this.to == null))
2359
2072
  this.from = null;
2360
- else if (this.from != null && this.from >= from)
2361
- this.from = Math.max(to, this.from) + diff;
2362
2073
  if (toOpen && (from < this.to || this.to == null) && (from > this.from || this.from == null))
2363
2074
  this.to = null;
2364
- else if (this.to != null && this.to > from)
2365
- this.to = to < this.to ? this.to + diff : from;
2366
2075
  },
2367
2076
  isDead: function() { return this.from != null && this.to != null && this.from >= this.to; },
2368
- sameSet: function(x) { return this.marker == x.marker; }
2077
+ sameSet: function(x) { return this.set == x.set; }
2369
2078
  };
2370
2079
 
2371
2080
  function Bookmark(pos) {
@@ -2408,7 +2117,7 @@ var CodeMirror = (function() {
2408
2117
  this.styles = styles || [text, null];
2409
2118
  this.text = text;
2410
2119
  this.height = 1;
2411
- this.marked = this.gutterMarker = this.className = this.bgClassName = this.handlers = null;
2120
+ this.marked = this.gutterMarker = this.className = this.handlers = null;
2412
2121
  this.stateAfter = this.parent = this.hidden = null;
2413
2122
  }
2414
2123
  Line.inheritMarks = function(text, orig) {
@@ -2454,7 +2163,6 @@ var CodeMirror = (function() {
2454
2163
  if (newmark) {
2455
2164
  if (!taken.marked) taken.marked = [];
2456
2165
  taken.marked.push(newmark); newmark.attach(taken);
2457
- if (newmark == mark) mk.splice(i--, 1);
2458
2166
  }
2459
2167
  }
2460
2168
  }
@@ -2564,82 +2272,34 @@ var CodeMirror = (function() {
2564
2272
  indentation: function(tabSize) {return countColumn(this.text, null, tabSize);},
2565
2273
  // Produces an HTML fragment for the line, taking selection,
2566
2274
  // marking, and highlighting into account.
2567
- getHTML: function(makeTab, wrapAt, wrapId, wrapWBR) {
2568
- var html = [], first = true, col = 0;
2569
- function span_(text, style) {
2275
+ getHTML: function(tabText, endAt) {
2276
+ var html = [], first = true;
2277
+ function span(text, style) {
2570
2278
  if (!text) return;
2571
2279
  // Work around a bug where, in some compat modes, IE ignores leading spaces
2572
2280
  if (first && ie && text.charAt(0) == " ") text = "\u00a0" + text.slice(1);
2573
2281
  first = false;
2574
- if (text.indexOf("\t") == -1) {
2575
- col += text.length;
2576
- var escaped = htmlEscape(text);
2577
- } else {
2578
- var escaped = "";
2579
- for (var pos = 0;;) {
2580
- var idx = text.indexOf("\t", pos);
2581
- if (idx == -1) {
2582
- escaped += htmlEscape(text.slice(pos));
2583
- col += text.length - pos;
2584
- break;
2585
- } else {
2586
- col += idx - pos;
2587
- var tab = makeTab(col);
2588
- escaped += htmlEscape(text.slice(pos, idx)) + tab.html;
2589
- col += tab.width;
2590
- pos = idx + 1;
2591
- }
2592
- }
2593
- }
2594
- if (style) html.push('<span class="', style, '">', escaped, "</span>");
2595
- else html.push(escaped);
2596
- }
2597
- var span = span_;
2598
- if (wrapAt != null) {
2599
- var outPos = 0, open = "<span id=\"" + wrapId + "\">";
2600
- span = function(text, style) {
2601
- var l = text.length;
2602
- if (wrapAt >= outPos && wrapAt < outPos + l) {
2603
- if (wrapAt > outPos) {
2604
- span_(text.slice(0, wrapAt - outPos), style);
2605
- // See comment at the definition of spanAffectsWrapping
2606
- if (wrapWBR) html.push("<wbr>");
2607
- }
2608
- html.push(open);
2609
- var cut = wrapAt - outPos;
2610
- span_(opera ? text.slice(cut, cut + 1) : text.slice(cut), style);
2611
- html.push("</span>");
2612
- if (opera) span_(text.slice(cut + 1), style);
2613
- wrapAt--;
2614
- outPos += l;
2615
- } else {
2616
- outPos += l;
2617
- span_(text, style);
2618
- // Output empty wrapper when at end of line
2619
- if (outPos == wrapAt && outPos == len) html.push(open + " </span>");
2620
- // Stop outputting HTML when gone sufficiently far beyond measure
2621
- else if (outPos > wrapAt + 10 && /\s/.test(text)) span = function(){};
2622
- }
2623
- }
2282
+ if (style) html.push('<span class="', style, '">', htmlEscape(text).replace(/\t/g, tabText), "</span>");
2283
+ else html.push(htmlEscape(text).replace(/\t/g, tabText));
2624
2284
  }
2625
-
2626
2285
  var st = this.styles, allText = this.text, marked = this.marked;
2627
2286
  var len = allText.length;
2287
+ if (endAt != null) len = Math.min(endAt, len);
2628
2288
  function styleToClass(style) {
2629
2289
  if (!style) return null;
2630
2290
  return "cm-" + style.replace(/ +/g, " cm-");
2631
2291
  }
2632
2292
 
2633
- if (!allText && wrapAt == null) {
2293
+ if (!allText && endAt == null)
2634
2294
  span(" ");
2635
- } else if (!marked || !marked.length) {
2295
+ else if (!marked || !marked.length)
2636
2296
  for (var i = 0, ch = 0; ch < len; i+=2) {
2637
2297
  var str = st[i], style = st[i+1], l = str.length;
2638
2298
  if (ch + l > len) str = str.slice(0, len - ch);
2639
2299
  ch += l;
2640
2300
  span(str, styleToClass(style));
2641
2301
  }
2642
- } else {
2302
+ else {
2643
2303
  var pos = 0, i = 0, text = "", style, sg = 0;
2644
2304
  var nextChange = marked[0].from || 0, marks = [], markpos = 0;
2645
2305
  function advanceMarks() {
@@ -2689,7 +2349,8 @@ var CodeMirror = (function() {
2689
2349
  if (state == 0) {
2690
2350
  if (end > from) dest.push(part.slice(from - pos, Math.min(part.length, to - pos)), source[i+1]);
2691
2351
  if (end >= from) state = 1;
2692
- } else if (state == 1) {
2352
+ }
2353
+ else if (state == 1) {
2693
2354
  if (end > to) dest.push(part.slice(0, to - pos), source[i+1]);
2694
2355
  else dest.push(part, source[i+1]);
2695
2356
  }
@@ -2724,7 +2385,7 @@ var CodeMirror = (function() {
2724
2385
  },
2725
2386
  insertHeight: function(at, lines, height) {
2726
2387
  this.height += height;
2727
- this.lines = this.lines.slice(0, at).concat(lines).concat(this.lines.slice(at));
2388
+ this.lines.splice.apply(this.lines, [at, 0].concat(lines));
2728
2389
  for (var i = 0, e = lines.length; i < e; ++i) lines[i].parent = this;
2729
2390
  },
2730
2391
  iterN: function(at, n, op) {
@@ -2890,36 +2551,33 @@ var CodeMirror = (function() {
2890
2551
  function History() {
2891
2552
  this.time = 0;
2892
2553
  this.done = []; this.undone = [];
2893
- this.compound = 0;
2894
- this.closed = false;
2895
2554
  }
2896
2555
  History.prototype = {
2897
2556
  addChange: function(start, added, old) {
2898
2557
  this.undone.length = 0;
2899
2558
  var time = +new Date, cur = this.done[this.done.length - 1], last = cur && cur[cur.length - 1];
2900
2559
  var dtime = time - this.time;
2901
-
2902
- if (this.compound && cur && !this.closed) {
2903
- cur.push({start: start, added: added, old: old});
2904
- } else if (dtime > 400 || !last || this.closed ||
2905
- last.start > start + old.length || last.start + last.added < start) {
2560
+ if (dtime > 400 || !last) {
2906
2561
  this.done.push([{start: start, added: added, old: old}]);
2907
- this.closed = false;
2562
+ } else if (last.start > start + added || last.start + last.added < start - last.added + last.old.length) {
2563
+ cur.push({start: start, added: added, old: old});
2908
2564
  } else {
2909
- var startBefore = Math.max(0, last.start - start),
2910
- endAfter = Math.max(0, (start + old.length) - (last.start + last.added));
2911
- for (var i = startBefore; i > 0; --i) last.old.unshift(old[i - 1]);
2912
- for (var i = endAfter; i > 0; --i) last.old.push(old[old.length - i]);
2913
- if (startBefore) last.start = start;
2914
- last.added += added - (old.length - startBefore - endAfter);
2565
+ var oldoff = 0;
2566
+ if (start < last.start) {
2567
+ for (var i = last.start - start - 1; i >= 0; --i)
2568
+ last.old.unshift(old[i]);
2569
+ last.added += last.start - start;
2570
+ last.start = start;
2571
+ }
2572
+ else if (last.start < start) {
2573
+ oldoff = start - last.start;
2574
+ added += oldoff;
2575
+ }
2576
+ for (var i = last.added - oldoff, e = old.length; i < e; ++i)
2577
+ last.old.push(old[i]);
2578
+ if (last.added < added) last.added = added;
2915
2579
  }
2916
2580
  this.time = time;
2917
- },
2918
- startCompound: function() {
2919
- if (!this.compound++) this.closed = true;
2920
- },
2921
- endCompound: function() {
2922
- if (!--this.compound) this.closed = true;
2923
2581
  }
2924
2582
  };
2925
2583
 
@@ -2964,7 +2622,8 @@ var CodeMirror = (function() {
2964
2622
  if (typeof node.addEventListener == "function") {
2965
2623
  node.addEventListener(type, handler, false);
2966
2624
  if (disconnect) return function() {node.removeEventListener(type, handler, false);};
2967
- } else {
2625
+ }
2626
+ else {
2968
2627
  var wrapHandler = function(event) {handler(event || window.event);};
2969
2628
  node.attachEvent("on" + type, wrapHandler);
2970
2629
  if (disconnect) return function() {node.detachEvent("on" + type, wrapHandler);};
@@ -2975,48 +2634,26 @@ var CodeMirror = (function() {
2975
2634
  function Delayed() {this.id = null;}
2976
2635
  Delayed.prototype = {set: function(ms, f) {clearTimeout(this.id); this.id = setTimeout(f, ms);}};
2977
2636
 
2978
- var Pass = CodeMirror.Pass = {toString: function(){return "CodeMirror.Pass";}};
2979
-
2980
- var gecko = /gecko\/\d{7}/i.test(navigator.userAgent);
2981
- var ie = /MSIE \d/.test(navigator.userAgent);
2982
- var ie_lt8 = /MSIE [1-7]\b/.test(navigator.userAgent);
2983
- var ie_lt9 = /MSIE [1-8]\b/.test(navigator.userAgent);
2984
- var quirksMode = ie && document.documentMode == 5;
2985
- var webkit = /WebKit\//.test(navigator.userAgent);
2986
- var chrome = /Chrome\//.test(navigator.userAgent);
2987
- var opera = /Opera\//.test(navigator.userAgent);
2988
- var safari = /Apple Computer/.test(navigator.vendor);
2989
- var khtml = /KHTML\//.test(navigator.userAgent);
2990
- var mac_geLion = /Mac OS X 10\D([7-9]|\d\d)\D/.test(navigator.userAgent);
2991
-
2992
2637
  // Detect drag-and-drop
2993
2638
  var dragAndDrop = function() {
2994
- // There is *some* kind of drag-and-drop support in IE6-8, but I
2995
- // couldn't get it to work yet.
2996
- if (ie_lt9) return false;
2639
+ // IE8 has ondragstart and ondrop properties, but doesn't seem to
2640
+ // actually support ondragstart the way it's supposed to work.
2641
+ if (/MSIE [1-8]\b/.test(navigator.userAgent)) return false;
2997
2642
  var div = document.createElement('div');
2998
- return "draggable" in div || "dragDrop" in div;
2643
+ return "draggable" in div;
2999
2644
  }();
3000
2645
 
2646
+ var gecko = /gecko\/\d{7}/i.test(navigator.userAgent);
2647
+ var ie = /MSIE \d/.test(navigator.userAgent);
2648
+ var webkit = /WebKit\//.test(navigator.userAgent);
2649
+
2650
+ var lineSep = "\n";
3001
2651
  // Feature-detect whether newlines in textareas are converted to \r\n
3002
- var lineSep = function () {
2652
+ (function () {
3003
2653
  var te = document.createElement("textarea");
3004
2654
  te.value = "foo\nbar";
3005
- if (te.value.indexOf("\r") > -1) return "\r\n";
3006
- return "\n";
3007
- }();
3008
-
3009
- // For a reason I have yet to figure out, some browsers disallow
3010
- // word wrapping between certain characters *only* if a new inline
3011
- // element is started between them. This makes it hard to reliably
3012
- // measure the position of things, since that requires inserting an
3013
- // extra span. This terribly fragile set of regexps matches the
3014
- // character combinations that suffer from this phenomenon on the
3015
- // various browsers.
3016
- var spanAffectsWrapping = /^$/; // Won't match any two-character string
3017
- if (gecko) spanAffectsWrapping = /$'/;
3018
- else if (safari) spanAffectsWrapping = /\-[^ \-?]|\?[^ !'\"\),.\-\/:;\?\]\}]/;
3019
- else if (chrome) spanAffectsWrapping = /\-[^ \-\.?]|\?[^ \-\.?\]\}:;!'\"\),\/]|[\.!\"#&%\)*+,:;=>\]|\}~][\(\{\[<]|\$'/;
2655
+ if (te.value.indexOf("\r") > -1) lineSep = "\r\n";
2656
+ }());
3020
2657
 
3021
2658
  // Counts the column offset in a string, taking tabs into account.
3022
2659
  // Used mostly to find indentation.
@@ -3097,19 +2734,18 @@ var CodeMirror = (function() {
3097
2734
  }
3098
2735
  // Recent (late 2011) Opera betas insert bogus newlines at the start
3099
2736
  // of the textContent, so we strip those.
3100
- if (htmlEscape("a") == "\na") {
2737
+ if (htmlEscape("a") == "\na")
3101
2738
  htmlEscape = function(str) {
3102
2739
  escapeElement.textContent = str;
3103
2740
  return escapeElement.innerHTML.slice(1);
3104
2741
  };
3105
2742
  // Some IEs don't preserve tabs through innerHTML
3106
- } else if (htmlEscape("\t") != "\t") {
2743
+ else if (htmlEscape("\t") != "\t")
3107
2744
  htmlEscape = function(str) {
3108
2745
  escapeElement.innerHTML = "";
3109
2746
  escapeElement.appendChild(document.createTextNode(str));
3110
2747
  return escapeElement.innerHTML;
3111
2748
  };
3112
- }
3113
2749
  CodeMirror.htmlEscape = htmlEscape;
3114
2750
 
3115
2751
  // Used to position the cursor after an undo/redo by finding the
@@ -3163,10 +2799,10 @@ var CodeMirror = (function() {
3163
2799
  var keyNames = {3: "Enter", 8: "Backspace", 9: "Tab", 13: "Enter", 16: "Shift", 17: "Ctrl", 18: "Alt",
3164
2800
  19: "Pause", 20: "CapsLock", 27: "Esc", 32: "Space", 33: "PageUp", 34: "PageDown", 35: "End",
3165
2801
  36: "Home", 37: "Left", 38: "Up", 39: "Right", 40: "Down", 44: "PrintScrn", 45: "Insert",
3166
- 46: "Delete", 59: ";", 91: "Mod", 92: "Mod", 93: "Mod", 109: "-", 107: "=", 127: "Delete",
3167
- 186: ";", 187: "=", 188: ",", 189: "-", 190: ".", 191: "/", 192: "`", 219: "[", 220: "\\",
3168
- 221: "]", 222: "'", 63276: "PageUp", 63277: "PageDown", 63275: "End", 63273: "Home",
3169
- 63234: "Left", 63232: "Up", 63235: "Right", 63233: "Down", 63302: "Insert", 63272: "Delete"};
2802
+ 46: "Delete", 59: ";", 91: "Mod", 92: "Mod", 93: "Mod", 186: ";", 187: "=", 188: ",",
2803
+ 189: "-", 190: ".", 191: "/", 192: "`", 219: "[", 220: "\\", 221: "]", 222: "'", 63276: "PageUp",
2804
+ 63277: "PageDown", 63275: "End", 63273: "Home", 63234: "Left", 63232: "Up", 63235: "Right",
2805
+ 63233: "Down", 63302: "Insert", 63272: "Delete"};
3170
2806
  CodeMirror.keyNames = keyNames;
3171
2807
  (function() {
3172
2808
  // Number keys