annotate .cms/lib/codemirror/keymap/vim.js @ 0:78edf6b517a0 draft

24.10
author Coffee CMS <info@coffee-cms.ru>
date Fri, 11 Oct 2024 22:40:23 +0000
parents
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1 (function(mod) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2 if (typeof exports == "object" && typeof module == "object") // CommonJS
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3 mod(require("../lib/codemirror"), require("../addon/search/searchcursor"), require("../addon/dialog/dialog"), require("../addon/edit/matchbrackets.js"));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4 else if (typeof define == "function" && define.amd) // AMD
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5 define(["../lib/codemirror", "../addon/search/searchcursor", "../addon/dialog/dialog", "../addon/edit/matchbrackets"], mod);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
6 else // Plain browser env
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
7 mod(CodeMirror);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
8 })(function(CodeMirror) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
9 'use strict';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
10 // CodeMirror, copyright (c) by Marijn Haverbeke and others
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
11 // Distributed under an MIT license: https://codemirror.net/5/LICENSE
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
12
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
13 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
14 * Supported keybindings:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
15 * Too many to list. Refer to defaultKeymap below.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
16 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
17 * Supported Ex commands:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
18 * Refer to defaultExCommandMap below.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
19 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
20 * Registers: unnamed, -, ., :, /, _, a-z, A-Z, 0-9
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
21 * (Does not respect the special case for number registers when delete
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
22 * operator is made with these commands: %, (, ), , /, ?, n, N, {, } )
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
23 * TODO: Implement the remaining registers.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
24 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
25 * Marks: a-z, A-Z, and 0-9
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
26 * TODO: Implement the remaining special marks. They have more complex
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
27 * behavior.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
28 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
29 * Events:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
30 * 'vim-mode-change' - raised on the editor anytime the current mode changes,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
31 * Event object: {mode: "visual", subMode: "linewise"}
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
32 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
33 * Code structure:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
34 * 1. Default keymap
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
35 * 2. Variable declarations and short basic helpers
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
36 * 3. Instance (External API) implementation
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
37 * 4. Internal state tracking objects (input state, counter) implementation
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
38 * and instantiation
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
39 * 5. Key handler (the main command dispatcher) implementation
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
40 * 6. Motion, operator, and action implementations
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
41 * 7. Helper functions for the key handler, motions, operators, and actions
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
42 * 8. Set up Vim to work as a keymap for CodeMirror.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
43 * 9. Ex command implementations.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
44 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
45
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
46 function initVim$1(CodeMirror) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
47
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
48 var Pos = CodeMirror.Pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
49
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
50 function transformCursor(cm, range) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
51 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
52 if (!vim || vim.insertMode) return range.head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
53 var head = vim.sel.head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
54 if (!head) return range.head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
55
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
56 if (vim.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
57 if (range.head.line != head.line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
58 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
59 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
60 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
61 if (range.from() == range.anchor && !range.empty()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
62 if (range.head.line == head.line && range.head.ch != head.ch)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
63 return new Pos(range.head.line, range.head.ch - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
64 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
65
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
66 return range.head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
67 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
68
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
69 var defaultKeymap = [
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
70 // Key to key mapping. This goes first to make it possible to override
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
71 // existing mappings.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
72 { keys: '<Left>', type: 'keyToKey', toKeys: 'h' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
73 { keys: '<Right>', type: 'keyToKey', toKeys: 'l' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
74 { keys: '<Up>', type: 'keyToKey', toKeys: 'k' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
75 { keys: '<Down>', type: 'keyToKey', toKeys: 'j' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
76 { keys: 'g<Up>', type: 'keyToKey', toKeys: 'gk' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
77 { keys: 'g<Down>', type: 'keyToKey', toKeys: 'gj' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
78 { keys: '<Space>', type: 'keyToKey', toKeys: 'l' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
79 { keys: '<BS>', type: 'keyToKey', toKeys: 'h', context: 'normal'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
80 { keys: '<Del>', type: 'keyToKey', toKeys: 'x', context: 'normal'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
81 { keys: '<C-Space>', type: 'keyToKey', toKeys: 'W' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
82 { keys: '<C-BS>', type: 'keyToKey', toKeys: 'B', context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
83 { keys: '<S-Space>', type: 'keyToKey', toKeys: 'w' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
84 { keys: '<S-BS>', type: 'keyToKey', toKeys: 'b', context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
85 { keys: '<C-n>', type: 'keyToKey', toKeys: 'j' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
86 { keys: '<C-p>', type: 'keyToKey', toKeys: 'k' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
87 { keys: '<C-[>', type: 'keyToKey', toKeys: '<Esc>' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
88 { keys: '<C-c>', type: 'keyToKey', toKeys: '<Esc>' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
89 { keys: '<C-[>', type: 'keyToKey', toKeys: '<Esc>', context: 'insert' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
90 { keys: '<C-c>', type: 'keyToKey', toKeys: '<Esc>', context: 'insert' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
91 { keys: '<C-Esc>', type: 'keyToKey', toKeys: '<Esc>' }, // ipad keyboard sends C-Esc instead of C-[
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
92 { keys: '<C-Esc>', type: 'keyToKey', toKeys: '<Esc>', context: 'insert' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
93 { keys: 's', type: 'keyToKey', toKeys: 'cl', context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
94 { keys: 's', type: 'keyToKey', toKeys: 'c', context: 'visual'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
95 { keys: 'S', type: 'keyToKey', toKeys: 'cc', context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
96 { keys: 'S', type: 'keyToKey', toKeys: 'VdO', context: 'visual' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
97 { keys: '<Home>', type: 'keyToKey', toKeys: '0' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
98 { keys: '<End>', type: 'keyToKey', toKeys: '$' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
99 { keys: '<PageUp>', type: 'keyToKey', toKeys: '<C-b>' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
100 { keys: '<PageDown>', type: 'keyToKey', toKeys: '<C-f>' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
101 { keys: '<CR>', type: 'keyToKey', toKeys: 'j^', context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
102 { keys: '<Ins>', type: 'keyToKey', toKeys: 'i', context: 'normal'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
103 { keys: '<Ins>', type: 'action', action: 'toggleOverwrite', context: 'insert' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
104 // Motions
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
105 { keys: 'H', type: 'motion', motion: 'moveToTopLine', motionArgs: { linewise: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
106 { keys: 'M', type: 'motion', motion: 'moveToMiddleLine', motionArgs: { linewise: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
107 { keys: 'L', type: 'motion', motion: 'moveToBottomLine', motionArgs: { linewise: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
108 { keys: 'h', type: 'motion', motion: 'moveByCharacters', motionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
109 { keys: 'l', type: 'motion', motion: 'moveByCharacters', motionArgs: { forward: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
110 { keys: 'j', type: 'motion', motion: 'moveByLines', motionArgs: { forward: true, linewise: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
111 { keys: 'k', type: 'motion', motion: 'moveByLines', motionArgs: { forward: false, linewise: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
112 { keys: 'gj', type: 'motion', motion: 'moveByDisplayLines', motionArgs: { forward: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
113 { keys: 'gk', type: 'motion', motion: 'moveByDisplayLines', motionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
114 { keys: 'w', type: 'motion', motion: 'moveByWords', motionArgs: { forward: true, wordEnd: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
115 { keys: 'W', type: 'motion', motion: 'moveByWords', motionArgs: { forward: true, wordEnd: false, bigWord: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
116 { keys: 'e', type: 'motion', motion: 'moveByWords', motionArgs: { forward: true, wordEnd: true, inclusive: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
117 { keys: 'E', type: 'motion', motion: 'moveByWords', motionArgs: { forward: true, wordEnd: true, bigWord: true, inclusive: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
118 { keys: 'b', type: 'motion', motion: 'moveByWords', motionArgs: { forward: false, wordEnd: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
119 { keys: 'B', type: 'motion', motion: 'moveByWords', motionArgs: { forward: false, wordEnd: false, bigWord: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
120 { keys: 'ge', type: 'motion', motion: 'moveByWords', motionArgs: { forward: false, wordEnd: true, inclusive: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
121 { keys: 'gE', type: 'motion', motion: 'moveByWords', motionArgs: { forward: false, wordEnd: true, bigWord: true, inclusive: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
122 { keys: '{', type: 'motion', motion: 'moveByParagraph', motionArgs: { forward: false, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
123 { keys: '}', type: 'motion', motion: 'moveByParagraph', motionArgs: { forward: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
124 { keys: '(', type: 'motion', motion: 'moveBySentence', motionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
125 { keys: ')', type: 'motion', motion: 'moveBySentence', motionArgs: { forward: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
126 { keys: '<C-f>', type: 'motion', motion: 'moveByPage', motionArgs: { forward: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
127 { keys: '<C-b>', type: 'motion', motion: 'moveByPage', motionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
128 { keys: '<C-d>', type: 'motion', motion: 'moveByScroll', motionArgs: { forward: true, explicitRepeat: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
129 { keys: '<C-u>', type: 'motion', motion: 'moveByScroll', motionArgs: { forward: false, explicitRepeat: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
130 { keys: 'gg', type: 'motion', motion: 'moveToLineOrEdgeOfDocument', motionArgs: { forward: false, explicitRepeat: true, linewise: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
131 { keys: 'G', type: 'motion', motion: 'moveToLineOrEdgeOfDocument', motionArgs: { forward: true, explicitRepeat: true, linewise: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
132 {keys: "g$", type: "motion", motion: "moveToEndOfDisplayLine"},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
133 {keys: "g^", type: "motion", motion: "moveToStartOfDisplayLine"},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
134 {keys: "g0", type: "motion", motion: "moveToStartOfDisplayLine"},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
135 { keys: '0', type: 'motion', motion: 'moveToStartOfLine' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
136 { keys: '^', type: 'motion', motion: 'moveToFirstNonWhiteSpaceCharacter' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
137 { keys: '+', type: 'motion', motion: 'moveByLines', motionArgs: { forward: true, toFirstChar:true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
138 { keys: '-', type: 'motion', motion: 'moveByLines', motionArgs: { forward: false, toFirstChar:true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
139 { keys: '_', type: 'motion', motion: 'moveByLines', motionArgs: { forward: true, toFirstChar:true, repeatOffset:-1 }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
140 { keys: '$', type: 'motion', motion: 'moveToEol', motionArgs: { inclusive: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
141 { keys: '%', type: 'motion', motion: 'moveToMatchedSymbol', motionArgs: { inclusive: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
142 { keys: 'f<character>', type: 'motion', motion: 'moveToCharacter', motionArgs: { forward: true , inclusive: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
143 { keys: 'F<character>', type: 'motion', motion: 'moveToCharacter', motionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
144 { keys: 't<character>', type: 'motion', motion: 'moveTillCharacter', motionArgs: { forward: true, inclusive: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
145 { keys: 'T<character>', type: 'motion', motion: 'moveTillCharacter', motionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
146 { keys: ';', type: 'motion', motion: 'repeatLastCharacterSearch', motionArgs: { forward: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
147 { keys: ',', type: 'motion', motion: 'repeatLastCharacterSearch', motionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
148 { keys: '\'<character>', type: 'motion', motion: 'goToMark', motionArgs: {toJumplist: true, linewise: true}},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
149 { keys: '`<character>', type: 'motion', motion: 'goToMark', motionArgs: {toJumplist: true}},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
150 { keys: ']`', type: 'motion', motion: 'jumpToMark', motionArgs: { forward: true } },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
151 { keys: '[`', type: 'motion', motion: 'jumpToMark', motionArgs: { forward: false } },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
152 { keys: ']\'', type: 'motion', motion: 'jumpToMark', motionArgs: { forward: true, linewise: true } },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
153 { keys: '[\'', type: 'motion', motion: 'jumpToMark', motionArgs: { forward: false, linewise: true } },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
154 // the next two aren't motions but must come before more general motion declarations
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
155 { keys: ']p', type: 'action', action: 'paste', isEdit: true, actionArgs: { after: true, isEdit: true, matchIndent: true}},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
156 { keys: '[p', type: 'action', action: 'paste', isEdit: true, actionArgs: { after: false, isEdit: true, matchIndent: true}},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
157 { keys: ']<character>', type: 'motion', motion: 'moveToSymbol', motionArgs: { forward: true, toJumplist: true}},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
158 { keys: '[<character>', type: 'motion', motion: 'moveToSymbol', motionArgs: { forward: false, toJumplist: true}},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
159 { keys: '|', type: 'motion', motion: 'moveToColumn'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
160 { keys: 'o', type: 'motion', motion: 'moveToOtherHighlightedEnd', context:'visual'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
161 { keys: 'O', type: 'motion', motion: 'moveToOtherHighlightedEnd', motionArgs: {sameLine: true}, context:'visual'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
162 // Operators
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
163 { keys: 'd', type: 'operator', operator: 'delete' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
164 { keys: 'y', type: 'operator', operator: 'yank' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
165 { keys: 'c', type: 'operator', operator: 'change' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
166 { keys: '=', type: 'operator', operator: 'indentAuto' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
167 { keys: '>', type: 'operator', operator: 'indent', operatorArgs: { indentRight: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
168 { keys: '<', type: 'operator', operator: 'indent', operatorArgs: { indentRight: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
169 { keys: 'g~', type: 'operator', operator: 'changeCase' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
170 { keys: 'gu', type: 'operator', operator: 'changeCase', operatorArgs: {toLower: true}, isEdit: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
171 { keys: 'gU', type: 'operator', operator: 'changeCase', operatorArgs: {toLower: false}, isEdit: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
172 { keys: 'n', type: 'motion', motion: 'findNext', motionArgs: { forward: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
173 { keys: 'N', type: 'motion', motion: 'findNext', motionArgs: { forward: false, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
174 { keys: 'gn', type: 'motion', motion: 'findAndSelectNextInclusive', motionArgs: { forward: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
175 { keys: 'gN', type: 'motion', motion: 'findAndSelectNextInclusive', motionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
176 // Operator-Motion dual commands
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
177 { keys: 'x', type: 'operatorMotion', operator: 'delete', motion: 'moveByCharacters', motionArgs: { forward: true }, operatorMotionArgs: { visualLine: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
178 { keys: 'X', type: 'operatorMotion', operator: 'delete', motion: 'moveByCharacters', motionArgs: { forward: false }, operatorMotionArgs: { visualLine: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
179 { keys: 'D', type: 'operatorMotion', operator: 'delete', motion: 'moveToEol', motionArgs: { inclusive: true }, context: 'normal'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
180 { keys: 'D', type: 'operator', operator: 'delete', operatorArgs: { linewise: true }, context: 'visual'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
181 { keys: 'Y', type: 'operatorMotion', operator: 'yank', motion: 'expandToLine', motionArgs: { linewise: true }, context: 'normal'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
182 { keys: 'Y', type: 'operator', operator: 'yank', operatorArgs: { linewise: true }, context: 'visual'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
183 { keys: 'C', type: 'operatorMotion', operator: 'change', motion: 'moveToEol', motionArgs: { inclusive: true }, context: 'normal'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
184 { keys: 'C', type: 'operator', operator: 'change', operatorArgs: { linewise: true }, context: 'visual'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
185 { keys: '~', type: 'operatorMotion', operator: 'changeCase', motion: 'moveByCharacters', motionArgs: { forward: true }, operatorArgs: { shouldMoveCursor: true }, context: 'normal'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
186 { keys: '~', type: 'operator', operator: 'changeCase', context: 'visual'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
187 { keys: '<C-u>', type: 'operatorMotion', operator: 'delete', motion: 'moveToStartOfLine', context: 'insert' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
188 { keys: '<C-w>', type: 'operatorMotion', operator: 'delete', motion: 'moveByWords', motionArgs: { forward: false, wordEnd: false }, context: 'insert' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
189 //ignore C-w in normal mode
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
190 { keys: '<C-w>', type: 'idle', context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
191 // Actions
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
192 { keys: '<C-i>', type: 'action', action: 'jumpListWalk', actionArgs: { forward: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
193 { keys: '<C-o>', type: 'action', action: 'jumpListWalk', actionArgs: { forward: false }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
194 { keys: '<C-e>', type: 'action', action: 'scroll', actionArgs: { forward: true, linewise: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
195 { keys: '<C-y>', type: 'action', action: 'scroll', actionArgs: { forward: false, linewise: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
196 { keys: 'a', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { insertAt: 'charAfter' }, context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
197 { keys: 'A', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { insertAt: 'eol' }, context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
198 { keys: 'A', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { insertAt: 'endOfSelectedArea' }, context: 'visual' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
199 { keys: 'i', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { insertAt: 'inplace' }, context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
200 { keys: 'gi', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { insertAt: 'lastEdit' }, context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
201 { keys: 'I', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { insertAt: 'firstNonBlank'}, context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
202 { keys: 'gI', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { insertAt: 'bol'}, context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
203 { keys: 'I', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { insertAt: 'startOfSelectedArea' }, context: 'visual' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
204 { keys: 'o', type: 'action', action: 'newLineAndEnterInsertMode', isEdit: true, interlaceInsertRepeat: true, actionArgs: { after: true }, context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
205 { keys: 'O', type: 'action', action: 'newLineAndEnterInsertMode', isEdit: true, interlaceInsertRepeat: true, actionArgs: { after: false }, context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
206 { keys: 'v', type: 'action', action: 'toggleVisualMode' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
207 { keys: 'V', type: 'action', action: 'toggleVisualMode', actionArgs: { linewise: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
208 { keys: '<C-v>', type: 'action', action: 'toggleVisualMode', actionArgs: { blockwise: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
209 { keys: '<C-q>', type: 'action', action: 'toggleVisualMode', actionArgs: { blockwise: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
210 { keys: 'gv', type: 'action', action: 'reselectLastSelection' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
211 { keys: 'J', type: 'action', action: 'joinLines', isEdit: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
212 { keys: 'gJ', type: 'action', action: 'joinLines', actionArgs: { keepSpaces: true }, isEdit: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
213 { keys: 'p', type: 'action', action: 'paste', isEdit: true, actionArgs: { after: true, isEdit: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
214 { keys: 'P', type: 'action', action: 'paste', isEdit: true, actionArgs: { after: false, isEdit: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
215 { keys: 'r<character>', type: 'action', action: 'replace', isEdit: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
216 { keys: '@<character>', type: 'action', action: 'replayMacro' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
217 { keys: 'q<character>', type: 'action', action: 'enterMacroRecordMode' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
218 // Handle Replace-mode as a special case of insert mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
219 { keys: 'R', type: 'action', action: 'enterInsertMode', isEdit: true, actionArgs: { replace: true }, context: 'normal'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
220 { keys: 'R', type: 'operator', operator: 'change', operatorArgs: { linewise: true, fullLine: true }, context: 'visual', exitVisualBlock: true},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
221 { keys: 'u', type: 'action', action: 'undo', context: 'normal' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
222 { keys: 'u', type: 'operator', operator: 'changeCase', operatorArgs: {toLower: true}, context: 'visual', isEdit: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
223 { keys: 'U', type: 'operator', operator: 'changeCase', operatorArgs: {toLower: false}, context: 'visual', isEdit: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
224 { keys: '<C-r>', type: 'action', action: 'redo' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
225 { keys: 'm<character>', type: 'action', action: 'setMark' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
226 { keys: '"<character>', type: 'action', action: 'setRegister' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
227 { keys: 'zz', type: 'action', action: 'scrollToCursor', actionArgs: { position: 'center' }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
228 { keys: 'z.', type: 'action', action: 'scrollToCursor', actionArgs: { position: 'center' }, motion: 'moveToFirstNonWhiteSpaceCharacter' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
229 { keys: 'zt', type: 'action', action: 'scrollToCursor', actionArgs: { position: 'top' }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
230 { keys: 'z<CR>', type: 'action', action: 'scrollToCursor', actionArgs: { position: 'top' }, motion: 'moveToFirstNonWhiteSpaceCharacter' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
231 { keys: 'zb', type: 'action', action: 'scrollToCursor', actionArgs: { position: 'bottom' }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
232 { keys: 'z-', type: 'action', action: 'scrollToCursor', actionArgs: { position: 'bottom' }, motion: 'moveToFirstNonWhiteSpaceCharacter' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
233 { keys: '.', type: 'action', action: 'repeatLastEdit' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
234 { keys: '<C-a>', type: 'action', action: 'incrementNumberToken', isEdit: true, actionArgs: {increase: true, backtrack: false}},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
235 { keys: '<C-x>', type: 'action', action: 'incrementNumberToken', isEdit: true, actionArgs: {increase: false, backtrack: false}},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
236 { keys: '<C-t>', type: 'action', action: 'indent', actionArgs: { indentRight: true }, context: 'insert' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
237 { keys: '<C-d>', type: 'action', action: 'indent', actionArgs: { indentRight: false }, context: 'insert' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
238 // Text object motions
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
239 { keys: 'a<character>', type: 'motion', motion: 'textObjectManipulation' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
240 { keys: 'i<character>', type: 'motion', motion: 'textObjectManipulation', motionArgs: { textObjectInner: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
241 // Search
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
242 { keys: '/', type: 'search', searchArgs: { forward: true, querySrc: 'prompt', toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
243 { keys: '?', type: 'search', searchArgs: { forward: false, querySrc: 'prompt', toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
244 { keys: '*', type: 'search', searchArgs: { forward: true, querySrc: 'wordUnderCursor', wholeWordOnly: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
245 { keys: '#', type: 'search', searchArgs: { forward: false, querySrc: 'wordUnderCursor', wholeWordOnly: true, toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
246 { keys: 'g*', type: 'search', searchArgs: { forward: true, querySrc: 'wordUnderCursor', toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
247 { keys: 'g#', type: 'search', searchArgs: { forward: false, querySrc: 'wordUnderCursor', toJumplist: true }},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
248 // Ex command
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
249 { keys: ':', type: 'ex' }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
250 ];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
251 var defaultKeymapLength = defaultKeymap.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
252
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
253 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
254 * Ex commands
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
255 * Care must be taken when adding to the default Ex command map. For any
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
256 * pair of commands that have a shared prefix, at least one of their
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
257 * shortNames must not match the prefix of the other command.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
258 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
259 var defaultExCommandMap = [
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
260 { name: 'colorscheme', shortName: 'colo' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
261 { name: 'map' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
262 { name: 'imap', shortName: 'im' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
263 { name: 'nmap', shortName: 'nm' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
264 { name: 'vmap', shortName: 'vm' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
265 { name: 'unmap' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
266 { name: 'write', shortName: 'w' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
267 { name: 'undo', shortName: 'u' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
268 { name: 'redo', shortName: 'red' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
269 { name: 'set', shortName: 'se' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
270 { name: 'setlocal', shortName: 'setl' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
271 { name: 'setglobal', shortName: 'setg' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
272 { name: 'sort', shortName: 'sor' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
273 { name: 'substitute', shortName: 's', possiblyAsync: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
274 { name: 'nohlsearch', shortName: 'noh' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
275 { name: 'yank', shortName: 'y' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
276 { name: 'delmarks', shortName: 'delm' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
277 { name: 'registers', shortName: 'reg', excludeFromCommandHistory: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
278 { name: 'vglobal', shortName: 'v' },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
279 { name: 'global', shortName: 'g' }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
280 ];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
281
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
282 function enterVimMode(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
283 cm.setOption('disableInput', true);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
284 cm.setOption('showCursorWhenSelecting', false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
285 CodeMirror.signal(cm, "vim-mode-change", {mode: "normal"});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
286 cm.on('cursorActivity', onCursorActivity);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
287 maybeInitVimState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
288 CodeMirror.on(cm.getInputField(), 'paste', getOnPasteFn(cm));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
289 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
290
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
291 function leaveVimMode(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
292 cm.setOption('disableInput', false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
293 cm.off('cursorActivity', onCursorActivity);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
294 CodeMirror.off(cm.getInputField(), 'paste', getOnPasteFn(cm));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
295 cm.state.vim = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
296 if (highlightTimeout) clearTimeout(highlightTimeout);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
297 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
298
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
299 function detachVimMap(cm, next) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
300 if (this == CodeMirror.keyMap.vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
301 cm.options.$customCursor = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
302 CodeMirror.rmClass(cm.getWrapperElement(), "cm-fat-cursor");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
303 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
304
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
305 if (!next || next.attach != attachVimMap)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
306 leaveVimMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
307 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
308 function attachVimMap(cm, prev) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
309 if (this == CodeMirror.keyMap.vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
310 if (cm.curOp) cm.curOp.selectionChanged = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
311 cm.options.$customCursor = transformCursor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
312 CodeMirror.addClass(cm.getWrapperElement(), "cm-fat-cursor");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
313 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
314
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
315 if (!prev || prev.attach != attachVimMap)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
316 enterVimMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
317 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
318
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
319 // Deprecated, simply setting the keymap works again.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
320 CodeMirror.defineOption('vimMode', false, function(cm, val, prev) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
321 if (val && cm.getOption("keyMap") != "vim")
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
322 cm.setOption("keyMap", "vim");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
323 else if (!val && prev != CodeMirror.Init && /^vim/.test(cm.getOption("keyMap")))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
324 cm.setOption("keyMap", "default");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
325 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
326
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
327 function cmKey(key, cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
328 if (!cm) { return undefined; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
329 if (this[key]) { return this[key]; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
330 var vimKey = cmKeyToVimKey(key);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
331 if (!vimKey) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
332 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
333 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
334 var cmd = vimApi.findKey(cm, vimKey);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
335 if (typeof cmd == 'function') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
336 CodeMirror.signal(cm, 'vim-keypress', vimKey);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
337 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
338 return cmd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
339 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
340
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
341 var modifiers = {Shift:'S',Ctrl:'C',Alt:'A',Cmd:'D',Mod:'A',CapsLock:''};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
342 var specialKeys = {Enter:'CR',Backspace:'BS',Delete:'Del',Insert:'Ins'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
343 function cmKeyToVimKey(key) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
344 if (key.charAt(0) == '\'') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
345 // Keypress character binding of format "'a'"
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
346 return key.charAt(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
347 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
348 var pieces = key.split(/-(?!$)/);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
349 var lastPiece = pieces[pieces.length - 1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
350 if (pieces.length == 1 && pieces[0].length == 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
351 // No-modifier bindings use literal character bindings above. Skip.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
352 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
353 } else if (pieces.length == 2 && pieces[0] == 'Shift' && lastPiece.length == 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
354 // Ignore Shift+char bindings as they should be handled by literal character.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
355 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
356 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
357 var hasCharacter = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
358 for (var i = 0; i < pieces.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
359 var piece = pieces[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
360 if (piece in modifiers) { pieces[i] = modifiers[piece]; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
361 else { hasCharacter = true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
362 if (piece in specialKeys) { pieces[i] = specialKeys[piece]; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
363 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
364 if (!hasCharacter) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
365 // Vim does not support modifier only keys.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
366 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
367 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
368 // TODO: Current bindings expect the character to be lower case, but
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
369 // it looks like vim key notation uses upper case.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
370 if (isUpperCase(lastPiece)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
371 pieces[pieces.length - 1] = lastPiece.toLowerCase();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
372 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
373 return '<' + pieces.join('-') + '>';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
374 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
375
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
376 function getOnPasteFn(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
377 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
378 if (!vim.onPasteFn) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
379 vim.onPasteFn = function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
380 if (!vim.insertMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
381 cm.setCursor(offsetCursor(cm.getCursor(), 0, 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
382 actions.enterInsertMode(cm, {}, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
383 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
384 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
385 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
386 return vim.onPasteFn;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
387 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
388
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
389 var numberRegex = /[\d]/;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
390 var wordCharTest = [CodeMirror.isWordChar, function(ch) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
391 return ch && !CodeMirror.isWordChar(ch) && !/\s/.test(ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
392 }], bigWordCharTest = [function(ch) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
393 return /\S/.test(ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
394 }];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
395 function makeKeyRange(start, size) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
396 var keys = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
397 for (var i = start; i < start + size; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
398 keys.push(String.fromCharCode(i));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
399 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
400 return keys;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
401 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
402 var upperCaseAlphabet = makeKeyRange(65, 26);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
403 var lowerCaseAlphabet = makeKeyRange(97, 26);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
404 var numbers = makeKeyRange(48, 10);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
405 var validMarks = [].concat(upperCaseAlphabet, lowerCaseAlphabet, numbers, ['<', '>']);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
406 var validRegisters = [].concat(upperCaseAlphabet, lowerCaseAlphabet, numbers, ['-', '"', '.', ':', '_', '/']);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
407 var upperCaseChars;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
408 try { upperCaseChars = new RegExp("^[\\p{Lu}]$", "u"); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
409 catch (_) { upperCaseChars = /^[A-Z]$/; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
410
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
411 function isLine(cm, line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
412 return line >= cm.firstLine() && line <= cm.lastLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
413 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
414 function isLowerCase(k) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
415 return (/^[a-z]$/).test(k);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
416 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
417 function isMatchableSymbol(k) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
418 return '()[]{}'.indexOf(k) != -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
419 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
420 function isNumber(k) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
421 return numberRegex.test(k);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
422 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
423 function isUpperCase(k) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
424 return upperCaseChars.test(k);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
425 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
426 function isWhiteSpaceString(k) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
427 return (/^\s*$/).test(k);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
428 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
429 function isEndOfSentenceSymbol(k) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
430 return '.?!'.indexOf(k) != -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
431 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
432 function inArray(val, arr) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
433 for (var i = 0; i < arr.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
434 if (arr[i] == val) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
435 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
436 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
437 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
438 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
439 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
440
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
441 var options = {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
442 function defineOption(name, defaultValue, type, aliases, callback) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
443 if (defaultValue === undefined && !callback) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
444 throw Error('defaultValue is required unless callback is provided');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
445 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
446 if (!type) { type = 'string'; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
447 options[name] = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
448 type: type,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
449 defaultValue: defaultValue,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
450 callback: callback
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
451 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
452 if (aliases) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
453 for (var i = 0; i < aliases.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
454 options[aliases[i]] = options[name];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
455 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
456 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
457 if (defaultValue) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
458 setOption(name, defaultValue);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
459 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
460 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
461
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
462 function setOption(name, value, cm, cfg) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
463 var option = options[name];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
464 cfg = cfg || {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
465 var scope = cfg.scope;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
466 if (!option) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
467 return new Error('Unknown option: ' + name);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
468 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
469 if (option.type == 'boolean') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
470 if (value && value !== true) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
471 return new Error('Invalid argument: ' + name + '=' + value);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
472 } else if (value !== false) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
473 // Boolean options are set to true if value is not defined.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
474 value = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
475 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
476 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
477 if (option.callback) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
478 if (scope !== 'local') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
479 option.callback(value, undefined);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
480 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
481 if (scope !== 'global' && cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
482 option.callback(value, cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
483 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
484 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
485 if (scope !== 'local') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
486 option.value = option.type == 'boolean' ? !!value : value;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
487 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
488 if (scope !== 'global' && cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
489 cm.state.vim.options[name] = {value: value};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
490 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
491 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
492 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
493
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
494 function getOption(name, cm, cfg) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
495 var option = options[name];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
496 cfg = cfg || {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
497 var scope = cfg.scope;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
498 if (!option) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
499 return new Error('Unknown option: ' + name);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
500 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
501 if (option.callback) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
502 var local = cm && option.callback(undefined, cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
503 if (scope !== 'global' && local !== undefined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
504 return local;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
505 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
506 if (scope !== 'local') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
507 return option.callback();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
508 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
509 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
510 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
511 var local = (scope !== 'global') && (cm && cm.state.vim.options[name]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
512 return (local || (scope !== 'local') && option || {}).value;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
513 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
514 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
515
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
516 defineOption('filetype', undefined, 'string', ['ft'], function(name, cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
517 // Option is local. Do nothing for global.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
518 if (cm === undefined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
519 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
520 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
521 // The 'filetype' option proxies to the CodeMirror 'mode' option.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
522 if (name === undefined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
523 var mode = cm.getOption('mode');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
524 return mode == 'null' ? '' : mode;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
525 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
526 var mode = name == '' ? 'null' : name;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
527 cm.setOption('mode', mode);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
528 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
529 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
530
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
531 var createCircularJumpList = function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
532 var size = 100;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
533 var pointer = -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
534 var head = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
535 var tail = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
536 var buffer = new Array(size);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
537 function add(cm, oldCur, newCur) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
538 var current = pointer % size;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
539 var curMark = buffer[current];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
540 function useNextSlot(cursor) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
541 var next = ++pointer % size;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
542 var trashMark = buffer[next];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
543 if (trashMark) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
544 trashMark.clear();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
545 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
546 buffer[next] = cm.setBookmark(cursor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
547 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
548 if (curMark) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
549 var markPos = curMark.find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
550 // avoid recording redundant cursor position
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
551 if (markPos && !cursorEqual(markPos, oldCur)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
552 useNextSlot(oldCur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
553 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
554 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
555 useNextSlot(oldCur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
556 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
557 useNextSlot(newCur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
558 head = pointer;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
559 tail = pointer - size + 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
560 if (tail < 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
561 tail = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
562 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
563 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
564 function move(cm, offset) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
565 pointer += offset;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
566 if (pointer > head) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
567 pointer = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
568 } else if (pointer < tail) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
569 pointer = tail;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
570 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
571 var mark = buffer[(size + pointer) % size];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
572 // skip marks that are temporarily removed from text buffer
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
573 if (mark && !mark.find()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
574 var inc = offset > 0 ? 1 : -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
575 var newCur;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
576 var oldCur = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
577 do {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
578 pointer += inc;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
579 mark = buffer[(size + pointer) % size];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
580 // skip marks that are the same as current position
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
581 if (mark &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
582 (newCur = mark.find()) &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
583 !cursorEqual(oldCur, newCur)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
584 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
585 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
586 } while (pointer < head && pointer > tail);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
587 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
588 return mark;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
589 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
590 function find(cm, offset) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
591 var oldPointer = pointer;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
592 var mark = move(cm, offset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
593 pointer = oldPointer;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
594 return mark && mark.find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
595 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
596 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
597 cachedCursor: undefined, //used for # and * jumps
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
598 add: add,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
599 find: find,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
600 move: move
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
601 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
602 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
603
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
604 // Returns an object to track the changes associated insert mode. It
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
605 // clones the object that is passed in, or creates an empty object one if
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
606 // none is provided.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
607 var createInsertModeChanges = function(c) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
608 if (c) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
609 // Copy construction
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
610 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
611 changes: c.changes,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
612 expectCursorActivityForChange: c.expectCursorActivityForChange
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
613 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
614 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
615 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
616 // Change list
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
617 changes: [],
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
618 // Set to true on change, false on cursorActivity.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
619 expectCursorActivityForChange: false
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
620 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
621 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
622
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
623 function MacroModeState() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
624 this.latestRegister = undefined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
625 this.isPlaying = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
626 this.isRecording = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
627 this.replaySearchQueries = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
628 this.onRecordingDone = undefined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
629 this.lastInsertModeChanges = createInsertModeChanges();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
630 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
631 MacroModeState.prototype = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
632 exitMacroRecordMode: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
633 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
634 if (macroModeState.onRecordingDone) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
635 macroModeState.onRecordingDone(); // close dialog
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
636 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
637 macroModeState.onRecordingDone = undefined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
638 macroModeState.isRecording = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
639 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
640 enterMacroRecordMode: function(cm, registerName) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
641 var register =
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
642 vimGlobalState.registerController.getRegister(registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
643 if (register) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
644 register.clear();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
645 this.latestRegister = registerName;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
646 if (cm.openDialog) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
647 var template = dom('span', {class: 'cm-vim-message'}, 'recording @' + registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
648 this.onRecordingDone = cm.openDialog(template, null, {bottom:true});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
649 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
650 this.isRecording = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
651 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
652 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
653 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
654
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
655 function maybeInitVimState(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
656 if (!cm.state.vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
657 // Store instance state in the CodeMirror object.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
658 cm.state.vim = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
659 inputState: new InputState(),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
660 // Vim's input state that triggered the last edit, used to repeat
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
661 // motions and operators with '.'.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
662 lastEditInputState: undefined,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
663 // Vim's action command before the last edit, used to repeat actions
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
664 // with '.' and insert mode repeat.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
665 lastEditActionCommand: undefined,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
666 // When using jk for navigation, if you move from a longer line to a
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
667 // shorter line, the cursor may clip to the end of the shorter line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
668 // If j is pressed again and cursor goes to the next line, the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
669 // cursor should go back to its horizontal position on the longer
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
670 // line if it can. This is to keep track of the horizontal position.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
671 lastHPos: -1,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
672 // Doing the same with screen-position for gj/gk
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
673 lastHSPos: -1,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
674 // The last motion command run. Cleared if a non-motion command gets
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
675 // executed in between.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
676 lastMotion: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
677 marks: {},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
678 insertMode: false,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
679 // Repeat count for changes made in insert mode, triggered by key
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
680 // sequences like 3,i. Only exists when insertMode is true.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
681 insertModeRepeat: undefined,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
682 visualMode: false,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
683 // If we are in visual line mode. No effect if visualMode is false.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
684 visualLine: false,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
685 visualBlock: false,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
686 lastSelection: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
687 lastPastedText: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
688 sel: {},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
689 // Buffer-local/window-local values of vim options.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
690 options: {}
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
691 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
692 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
693 return cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
694 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
695 var vimGlobalState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
696 function resetVimGlobalState() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
697 vimGlobalState = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
698 // The current search query.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
699 searchQuery: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
700 // Whether we are searching backwards.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
701 searchIsReversed: false,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
702 // Replace part of the last substituted pattern
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
703 lastSubstituteReplacePart: undefined,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
704 jumpList: createCircularJumpList(),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
705 macroModeState: new MacroModeState,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
706 // Recording latest f, t, F or T motion command.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
707 lastCharacterSearch: {increment:0, forward:true, selectedCharacter:''},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
708 registerController: new RegisterController({}),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
709 // search history buffer
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
710 searchHistoryController: new HistoryController(),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
711 // ex Command history buffer
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
712 exCommandHistoryController : new HistoryController()
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
713 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
714 for (var optionName in options) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
715 var option = options[optionName];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
716 option.value = option.defaultValue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
717 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
718 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
719
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
720 var lastInsertModeKeyTimer;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
721 var vimApi = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
722 enterVimMode: enterVimMode,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
723 buildKeyMap: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
724 // TODO: Convert keymap into dictionary format for fast lookup.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
725 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
726 // Testing hook, though it might be useful to expose the register
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
727 // controller anyway.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
728 getRegisterController: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
729 return vimGlobalState.registerController;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
730 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
731 // Testing hook.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
732 resetVimGlobalState_: resetVimGlobalState,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
733
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
734 // Testing hook.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
735 getVimGlobalState_: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
736 return vimGlobalState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
737 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
738
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
739 // Testing hook.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
740 maybeInitVimState_: maybeInitVimState,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
741
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
742 suppressErrorLogging: false,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
743
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
744 InsertModeKey: InsertModeKey,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
745 map: function(lhs, rhs, ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
746 // Add user defined key bindings.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
747 exCommandDispatcher.map(lhs, rhs, ctx);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
748 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
749 unmap: function(lhs, ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
750 return exCommandDispatcher.unmap(lhs, ctx);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
751 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
752 // Non-recursive map function.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
753 // NOTE: This will not create mappings to key maps that aren't present
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
754 // in the default key map. See TODO at bottom of function.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
755 noremap: function(lhs, rhs, ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
756 function toCtxArray(ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
757 return ctx ? [ctx] : ['normal', 'insert', 'visual'];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
758 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
759 var ctxsToMap = toCtxArray(ctx);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
760 // Look through all actual defaults to find a map candidate.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
761 var actualLength = defaultKeymap.length, origLength = defaultKeymapLength;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
762 for (var i = actualLength - origLength;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
763 i < actualLength && ctxsToMap.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
764 i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
765 var mapping = defaultKeymap[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
766 // Omit mappings that operate in the wrong context(s) and those of invalid type.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
767 if (mapping.keys == rhs &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
768 (!ctx || !mapping.context || mapping.context === ctx) &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
769 mapping.type.substr(0, 2) !== 'ex' &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
770 mapping.type.substr(0, 3) !== 'key') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
771 // Make a shallow copy of the original keymap entry.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
772 var newMapping = {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
773 for (var key in mapping) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
774 newMapping[key] = mapping[key];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
775 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
776 // Modify it point to the new mapping with the proper context.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
777 newMapping.keys = lhs;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
778 if (ctx && !newMapping.context) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
779 newMapping.context = ctx;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
780 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
781 // Add it to the keymap with a higher priority than the original.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
782 this._mapCommand(newMapping);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
783 // Record the mapped contexts as complete.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
784 var mappedCtxs = toCtxArray(mapping.context);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
785 ctxsToMap = ctxsToMap.filter(function(el) { return mappedCtxs.indexOf(el) === -1; });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
786 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
787 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
788 // TODO: Create non-recursive keyToKey mappings for the unmapped contexts once those exist.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
789 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
790 // Remove all user-defined mappings for the provided context.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
791 mapclear: function(ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
792 // Partition the existing keymap into user-defined and true defaults.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
793 var actualLength = defaultKeymap.length,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
794 origLength = defaultKeymapLength;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
795 var userKeymap = defaultKeymap.slice(0, actualLength - origLength);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
796 defaultKeymap = defaultKeymap.slice(actualLength - origLength);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
797 if (ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
798 // If a specific context is being cleared, we need to keep mappings
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
799 // from all other contexts.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
800 for (var i = userKeymap.length - 1; i >= 0; i--) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
801 var mapping = userKeymap[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
802 if (ctx !== mapping.context) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
803 if (mapping.context) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
804 this._mapCommand(mapping);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
805 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
806 // `mapping` applies to all contexts so create keymap copies
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
807 // for each context except the one being cleared.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
808 var contexts = ['normal', 'insert', 'visual'];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
809 for (var j in contexts) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
810 if (contexts[j] !== ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
811 var newMapping = {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
812 for (var key in mapping) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
813 newMapping[key] = mapping[key];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
814 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
815 newMapping.context = contexts[j];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
816 this._mapCommand(newMapping);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
817 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
818 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
819 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
820 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
821 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
822 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
823 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
824 // TODO: Expose setOption and getOption as instance methods. Need to decide how to namespace
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
825 // them, or somehow make them work with the existing CodeMirror setOption/getOption API.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
826 setOption: setOption,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
827 getOption: getOption,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
828 defineOption: defineOption,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
829 defineEx: function(name, prefix, func){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
830 if (!prefix) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
831 prefix = name;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
832 } else if (name.indexOf(prefix) !== 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
833 throw new Error('(Vim.defineEx) "'+prefix+'" is not a prefix of "'+name+'", command not registered');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
834 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
835 exCommands[name]=func;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
836 exCommandDispatcher.commandMap_[prefix]={name:name, shortName:prefix, type:'api'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
837 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
838 handleKey: function (cm, key, origin) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
839 var command = this.findKey(cm, key, origin);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
840 if (typeof command === 'function') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
841 return command();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
842 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
843 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
844 multiSelectHandleKey: multiSelectHandleKey,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
845
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
846 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
847 * This is the outermost function called by CodeMirror, after keys have
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
848 * been mapped to their Vim equivalents.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
849 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
850 * Finds a command based on the key (and cached keys if there is a
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
851 * multi-key sequence). Returns `undefined` if no key is matched, a noop
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
852 * function if a partial match is found (multi-key), and a function to
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
853 * execute the bound command if a a key is matched. The function always
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
854 * returns true.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
855 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
856 findKey: function(cm, key, origin) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
857 var vim = maybeInitVimState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
858 function handleMacroRecording() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
859 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
860 if (macroModeState.isRecording) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
861 if (key == 'q') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
862 macroModeState.exitMacroRecordMode();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
863 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
864 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
865 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
866 if (origin != 'mapping') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
867 logKey(macroModeState, key);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
868 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
869 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
870 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
871 function handleEsc() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
872 if (key == '<Esc>') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
873 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
874 // Get back to normal mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
875 exitVisualMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
876 } else if (vim.insertMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
877 // Get back to normal mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
878 exitInsertMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
879 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
880 // We're already in normal mode. Let '<Esc>' be handled normally.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
881 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
882 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
883 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
884 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
885 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
886 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
887 function doKeyToKey(keys) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
888 // TODO: prevent infinite recursion.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
889 var match;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
890 while (keys) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
891 // Pull off one command key, which is either a single character
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
892 // or a special sequence wrapped in '<' and '>', e.g. '<Space>'.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
893 match = (/<\w+-.+?>|<\w+>|./).exec(keys);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
894 key = match[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
895 keys = keys.substring(match.index + key.length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
896 vimApi.handleKey(cm, key, 'mapping');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
897 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
898 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
899
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
900 function handleKeyInsertMode() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
901 if (handleEsc()) { return true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
902 var keys = vim.inputState.keyBuffer = vim.inputState.keyBuffer + key;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
903 var keysAreChars = key.length == 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
904 var match = commandDispatcher.matchCommand(keys, defaultKeymap, vim.inputState, 'insert');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
905 // Need to check all key substrings in insert mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
906 while (keys.length > 1 && match.type != 'full') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
907 var keys = vim.inputState.keyBuffer = keys.slice(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
908 var thisMatch = commandDispatcher.matchCommand(keys, defaultKeymap, vim.inputState, 'insert');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
909 if (thisMatch.type != 'none') { match = thisMatch; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
910 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
911 if (match.type == 'none') { clearInputState(cm); return false; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
912 else if (match.type == 'partial') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
913 if (lastInsertModeKeyTimer) { window.clearTimeout(lastInsertModeKeyTimer); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
914 lastInsertModeKeyTimer = window.setTimeout(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
915 function() { if (vim.insertMode && vim.inputState.keyBuffer) { clearInputState(cm); } },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
916 getOption('insertModeEscKeysTimeout'));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
917 return !keysAreChars;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
918 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
919
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
920 if (lastInsertModeKeyTimer) { window.clearTimeout(lastInsertModeKeyTimer); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
921 if (keysAreChars) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
922 var selections = cm.listSelections();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
923 for (var i = 0; i < selections.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
924 var here = selections[i].head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
925 cm.replaceRange('', offsetCursor(here, 0, -(keys.length - 1)), here, '+input');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
926 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
927 vimGlobalState.macroModeState.lastInsertModeChanges.changes.pop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
928 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
929 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
930 return match.command;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
931 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
932
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
933 function handleKeyNonInsertMode() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
934 if (handleMacroRecording() || handleEsc()) { return true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
935
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
936 var keys = vim.inputState.keyBuffer = vim.inputState.keyBuffer + key;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
937 if (/^[1-9]\d*$/.test(keys)) { return true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
938
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
939 var keysMatcher = /^(\d*)(.*)$/.exec(keys);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
940 if (!keysMatcher) { clearInputState(cm); return false; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
941 var context = vim.visualMode ? 'visual' :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
942 'normal';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
943 var mainKey = keysMatcher[2] || keysMatcher[1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
944 if (vim.inputState.operatorShortcut && vim.inputState.operatorShortcut.slice(-1) == mainKey) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
945 // multikey operators act linewise by repeating only the last character
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
946 mainKey = vim.inputState.operatorShortcut;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
947 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
948 var match = commandDispatcher.matchCommand(mainKey, defaultKeymap, vim.inputState, context);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
949 if (match.type == 'none') { clearInputState(cm); return false; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
950 else if (match.type == 'partial') { return true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
951 else if (match.type == 'clear') { clearInputState(cm); return true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
952
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
953 vim.inputState.keyBuffer = '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
954 keysMatcher = /^(\d*)(.*)$/.exec(keys);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
955 if (keysMatcher[1] && keysMatcher[1] != '0') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
956 vim.inputState.pushRepeatDigit(keysMatcher[1]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
957 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
958 return match.command;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
959 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
960
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
961 var command;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
962 if (vim.insertMode) { command = handleKeyInsertMode(); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
963 else { command = handleKeyNonInsertMode(); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
964 if (command === false) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
965 return !vim.insertMode && key.length === 1 ? function() { return true; } : undefined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
966 } else if (command === true) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
967 // TODO: Look into using CodeMirror's multi-key handling.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
968 // Return no-op since we are caching the key. Counts as handled, but
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
969 // don't want act on it just yet.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
970 return function() { return true; };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
971 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
972 return function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
973 return cm.operation(function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
974 cm.curOp.isVimOp = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
975 try {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
976 if (command.type == 'keyToKey') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
977 doKeyToKey(command.toKeys);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
978 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
979 commandDispatcher.processCommand(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
980 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
981 } catch (e) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
982 // clear VIM state in case it's in a bad state.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
983 cm.state.vim = undefined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
984 maybeInitVimState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
985 if (!vimApi.suppressErrorLogging) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
986 console['log'](e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
987 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
988 throw e;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
989 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
990 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
991 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
992 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
993 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
994 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
995 handleEx: function(cm, input) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
996 exCommandDispatcher.processCommand(cm, input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
997 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
998
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
999 defineMotion: defineMotion,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1000 defineAction: defineAction,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1001 defineOperator: defineOperator,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1002 mapCommand: mapCommand,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1003 _mapCommand: _mapCommand,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1004
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1005 defineRegister: defineRegister,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1006
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1007 exitVisualMode: exitVisualMode,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1008 exitInsertMode: exitInsertMode
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1009 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1010
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1011 // Represents the current input state.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1012 function InputState() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1013 this.prefixRepeat = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1014 this.motionRepeat = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1015
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1016 this.operator = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1017 this.operatorArgs = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1018 this.motion = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1019 this.motionArgs = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1020 this.keyBuffer = []; // For matching multi-key commands.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1021 this.registerName = null; // Defaults to the unnamed register.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1022 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1023 InputState.prototype.pushRepeatDigit = function(n) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1024 if (!this.operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1025 this.prefixRepeat = this.prefixRepeat.concat(n);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1026 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1027 this.motionRepeat = this.motionRepeat.concat(n);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1028 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1029 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1030 InputState.prototype.getRepeat = function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1031 var repeat = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1032 if (this.prefixRepeat.length > 0 || this.motionRepeat.length > 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1033 repeat = 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1034 if (this.prefixRepeat.length > 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1035 repeat *= parseInt(this.prefixRepeat.join(''), 10);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1036 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1037 if (this.motionRepeat.length > 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1038 repeat *= parseInt(this.motionRepeat.join(''), 10);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1039 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1040 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1041 return repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1042 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1043
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1044 function clearInputState(cm, reason) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1045 cm.state.vim.inputState = new InputState();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1046 CodeMirror.signal(cm, 'vim-command-done', reason);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1047 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1048
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1049 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1050 * Register stores information about copy and paste registers. Besides
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1051 * text, a register must store whether it is linewise (i.e., when it is
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1052 * pasted, should it insert itself into a new line, or should the text be
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1053 * inserted at the cursor position.)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1054 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1055 function Register(text, linewise, blockwise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1056 this.clear();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1057 this.keyBuffer = [text || ''];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1058 this.insertModeChanges = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1059 this.searchQueries = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1060 this.linewise = !!linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1061 this.blockwise = !!blockwise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1062 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1063 Register.prototype = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1064 setText: function(text, linewise, blockwise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1065 this.keyBuffer = [text || ''];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1066 this.linewise = !!linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1067 this.blockwise = !!blockwise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1068 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1069 pushText: function(text, linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1070 // if this register has ever been set to linewise, use linewise.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1071 if (linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1072 if (!this.linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1073 this.keyBuffer.push('\n');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1074 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1075 this.linewise = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1076 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1077 this.keyBuffer.push(text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1078 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1079 pushInsertModeChanges: function(changes) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1080 this.insertModeChanges.push(createInsertModeChanges(changes));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1081 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1082 pushSearchQuery: function(query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1083 this.searchQueries.push(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1084 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1085 clear: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1086 this.keyBuffer = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1087 this.insertModeChanges = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1088 this.searchQueries = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1089 this.linewise = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1090 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1091 toString: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1092 return this.keyBuffer.join('');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1093 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1094 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1095
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1096 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1097 * Defines an external register.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1098 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1099 * The name should be a single character that will be used to reference the register.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1100 * The register should support setText, pushText, clear, and toString(). See Register
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1101 * for a reference implementation.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1102 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1103 function defineRegister(name, register) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1104 var registers = vimGlobalState.registerController.registers;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1105 if (!name || name.length != 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1106 throw Error('Register name must be 1 character');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1107 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1108 if (registers[name]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1109 throw Error('Register already defined ' + name);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1110 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1111 registers[name] = register;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1112 validRegisters.push(name);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1113 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1114
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1115 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1116 * vim registers allow you to keep many independent copy and paste buffers.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1117 * See http://usevim.com/2012/04/13/registers/ for an introduction.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1118 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1119 * RegisterController keeps the state of all the registers. An initial
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1120 * state may be passed in. The unnamed register '"' will always be
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1121 * overridden.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1122 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1123 function RegisterController(registers) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1124 this.registers = registers;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1125 this.unnamedRegister = registers['"'] = new Register();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1126 registers['.'] = new Register();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1127 registers[':'] = new Register();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1128 registers['/'] = new Register();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1129 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1130 RegisterController.prototype = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1131 pushText: function(registerName, operator, text, linewise, blockwise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1132 // The black hole register, "_, means delete/yank to nowhere.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1133 if (registerName === '_') return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1134 if (linewise && text.charAt(text.length - 1) !== '\n'){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1135 text += '\n';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1136 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1137 // Lowercase and uppercase registers refer to the same register.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1138 // Uppercase just means append.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1139 var register = this.isValidRegister(registerName) ?
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1140 this.getRegister(registerName) : null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1141 // if no register/an invalid register was specified, things go to the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1142 // default registers
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1143 if (!register) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1144 switch (operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1145 case 'yank':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1146 // The 0 register contains the text from the most recent yank.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1147 this.registers['0'] = new Register(text, linewise, blockwise);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1148 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1149 case 'delete':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1150 case 'change':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1151 if (text.indexOf('\n') == -1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1152 // Delete less than 1 line. Update the small delete register.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1153 this.registers['-'] = new Register(text, linewise);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1154 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1155 // Shift down the contents of the numbered registers and put the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1156 // deleted text into register 1.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1157 this.shiftNumericRegisters_();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1158 this.registers['1'] = new Register(text, linewise);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1159 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1160 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1161 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1162 // Make sure the unnamed register is set to what just happened
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1163 this.unnamedRegister.setText(text, linewise, blockwise);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1164 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1165 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1166
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1167 // If we've gotten to this point, we've actually specified a register
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1168 var append = isUpperCase(registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1169 if (append) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1170 register.pushText(text, linewise);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1171 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1172 register.setText(text, linewise, blockwise);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1173 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1174 // The unnamed register always has the same value as the last used
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1175 // register.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1176 this.unnamedRegister.setText(register.toString(), linewise);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1177 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1178 // Gets the register named @name. If one of @name doesn't already exist,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1179 // create it. If @name is invalid, return the unnamedRegister.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1180 getRegister: function(name) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1181 if (!this.isValidRegister(name)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1182 return this.unnamedRegister;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1183 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1184 name = name.toLowerCase();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1185 if (!this.registers[name]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1186 this.registers[name] = new Register();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1187 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1188 return this.registers[name];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1189 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1190 isValidRegister: function(name) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1191 return name && inArray(name, validRegisters);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1192 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1193 shiftNumericRegisters_: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1194 for (var i = 9; i >= 2; i--) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1195 this.registers[i] = this.getRegister('' + (i - 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1196 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1197 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1198 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1199 function HistoryController() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1200 this.historyBuffer = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1201 this.iterator = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1202 this.initialPrefix = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1203 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1204 HistoryController.prototype = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1205 // the input argument here acts a user entered prefix for a small time
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1206 // until we start autocompletion in which case it is the autocompleted.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1207 nextMatch: function (input, up) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1208 var historyBuffer = this.historyBuffer;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1209 var dir = up ? -1 : 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1210 if (this.initialPrefix === null) this.initialPrefix = input;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1211 for (var i = this.iterator + dir; up ? i >= 0 : i < historyBuffer.length; i+= dir) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1212 var element = historyBuffer[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1213 for (var j = 0; j <= element.length; j++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1214 if (this.initialPrefix == element.substring(0, j)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1215 this.iterator = i;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1216 return element;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1217 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1218 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1219 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1220 // should return the user input in case we reach the end of buffer.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1221 if (i >= historyBuffer.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1222 this.iterator = historyBuffer.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1223 return this.initialPrefix;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1224 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1225 // return the last autocompleted query or exCommand as it is.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1226 if (i < 0 ) return input;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1227 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1228 pushInput: function(input) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1229 var index = this.historyBuffer.indexOf(input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1230 if (index > -1) this.historyBuffer.splice(index, 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1231 if (input.length) this.historyBuffer.push(input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1232 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1233 reset: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1234 this.initialPrefix = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1235 this.iterator = this.historyBuffer.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1236 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1237 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1238 var commandDispatcher = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1239 matchCommand: function(keys, keyMap, inputState, context) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1240 var matches = commandMatches(keys, keyMap, context, inputState);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1241 if (!matches.full && !matches.partial) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1242 return {type: 'none'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1243 } else if (!matches.full && matches.partial) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1244 return {type: 'partial'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1245 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1246
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1247 var bestMatch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1248 for (var i = 0; i < matches.full.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1249 var match = matches.full[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1250 if (!bestMatch) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1251 bestMatch = match;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1252 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1253 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1254 if (bestMatch.keys.slice(-11) == '<character>') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1255 var character = lastChar(keys);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1256 if (!character || character.length > 1) return {type: 'clear'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1257 inputState.selectedCharacter = character;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1258 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1259 return {type: 'full', command: bestMatch};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1260 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1261 processCommand: function(cm, vim, command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1262 vim.inputState.repeatOverride = command.repeatOverride;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1263 switch (command.type) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1264 case 'motion':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1265 this.processMotion(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1266 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1267 case 'operator':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1268 this.processOperator(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1269 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1270 case 'operatorMotion':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1271 this.processOperatorMotion(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1272 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1273 case 'action':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1274 this.processAction(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1275 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1276 case 'search':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1277 this.processSearch(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1278 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1279 case 'ex':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1280 case 'keyToEx':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1281 this.processEx(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1282 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1283 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1284 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1285 processMotion: function(cm, vim, command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1286 vim.inputState.motion = command.motion;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1287 vim.inputState.motionArgs = copyArgs(command.motionArgs);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1288 this.evalInput(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1289 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1290 processOperator: function(cm, vim, command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1291 var inputState = vim.inputState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1292 if (inputState.operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1293 if (inputState.operator == command.operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1294 // Typing an operator twice like 'dd' makes the operator operate
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1295 // linewise
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1296 inputState.motion = 'expandToLine';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1297 inputState.motionArgs = { linewise: true };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1298 this.evalInput(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1299 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1300 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1301 // 2 different operators in a row doesn't make sense.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1302 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1303 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1304 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1305 inputState.operator = command.operator;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1306 inputState.operatorArgs = copyArgs(command.operatorArgs);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1307 if (command.keys.length > 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1308 inputState.operatorShortcut = command.keys;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1309 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1310 if (command.exitVisualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1311 vim.visualBlock = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1312 updateCmSelection(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1313 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1314 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1315 // Operating on a selection in visual mode. We don't need a motion.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1316 this.evalInput(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1317 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1318 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1319 processOperatorMotion: function(cm, vim, command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1320 var visualMode = vim.visualMode;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1321 var operatorMotionArgs = copyArgs(command.operatorMotionArgs);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1322 if (operatorMotionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1323 // Operator motions may have special behavior in visual mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1324 if (visualMode && operatorMotionArgs.visualLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1325 vim.visualLine = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1326 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1327 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1328 this.processOperator(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1329 if (!visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1330 this.processMotion(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1331 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1332 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1333 processAction: function(cm, vim, command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1334 var inputState = vim.inputState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1335 var repeat = inputState.getRepeat();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1336 var repeatIsExplicit = !!repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1337 var actionArgs = copyArgs(command.actionArgs) || {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1338 if (inputState.selectedCharacter) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1339 actionArgs.selectedCharacter = inputState.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1340 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1341 // Actions may or may not have motions and operators. Do these first.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1342 if (command.operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1343 this.processOperator(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1344 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1345 if (command.motion) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1346 this.processMotion(cm, vim, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1347 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1348 if (command.motion || command.operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1349 this.evalInput(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1350 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1351 actionArgs.repeat = repeat || 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1352 actionArgs.repeatIsExplicit = repeatIsExplicit;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1353 actionArgs.registerName = inputState.registerName;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1354 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1355 vim.lastMotion = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1356 if (command.isEdit) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1357 this.recordLastEdit(vim, inputState, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1358 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1359 actions[command.action](cm, actionArgs, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1360 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1361 processSearch: function(cm, vim, command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1362 if (!cm.getSearchCursor) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1363 // Search depends on SearchCursor.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1364 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1365 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1366 var forward = command.searchArgs.forward;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1367 var wholeWordOnly = command.searchArgs.wholeWordOnly;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1368 getSearchState(cm).setReversed(!forward);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1369 var promptPrefix = (forward) ? '/' : '?';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1370 var originalQuery = getSearchState(cm).getQuery();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1371 var originalScrollPos = cm.getScrollInfo();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1372 function handleQuery(query, ignoreCase, smartCase) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1373 vimGlobalState.searchHistoryController.pushInput(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1374 vimGlobalState.searchHistoryController.reset();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1375 try {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1376 updateSearchQuery(cm, query, ignoreCase, smartCase);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1377 } catch (e) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1378 showConfirm(cm, 'Invalid regex: ' + query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1379 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1380 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1381 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1382 commandDispatcher.processMotion(cm, vim, {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1383 type: 'motion',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1384 motion: 'findNext',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1385 motionArgs: { forward: true, toJumplist: command.searchArgs.toJumplist }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1386 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1387 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1388 function onPromptClose(query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1389 cm.scrollTo(originalScrollPos.left, originalScrollPos.top);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1390 handleQuery(query, true /** ignoreCase */, true /** smartCase */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1391 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1392 if (macroModeState.isRecording) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1393 logSearchQuery(macroModeState, query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1394 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1395 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1396 function onPromptKeyUp(e, query, close) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1397 var keyName = CodeMirror.keyName(e), up, offset;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1398 if (keyName == 'Up' || keyName == 'Down') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1399 up = keyName == 'Up' ? true : false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1400 offset = e.target ? e.target.selectionEnd : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1401 query = vimGlobalState.searchHistoryController.nextMatch(query, up) || '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1402 close(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1403 if (offset && e.target) e.target.selectionEnd = e.target.selectionStart = Math.min(offset, e.target.value.length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1404 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1405 if ( keyName != 'Left' && keyName != 'Right' && keyName != 'Ctrl' && keyName != 'Alt' && keyName != 'Shift')
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1406 vimGlobalState.searchHistoryController.reset();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1407 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1408 var parsedQuery;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1409 try {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1410 parsedQuery = updateSearchQuery(cm, query,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1411 true /** ignoreCase */, true /** smartCase */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1412 } catch (e) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1413 // Swallow bad regexes for incremental search.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1414 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1415 if (parsedQuery) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1416 cm.scrollIntoView(findNext(cm, !forward, parsedQuery), 30);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1417 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1418 clearSearchHighlight(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1419 cm.scrollTo(originalScrollPos.left, originalScrollPos.top);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1420 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1421 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1422 function onPromptKeyDown(e, query, close) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1423 var keyName = CodeMirror.keyName(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1424 if (keyName == 'Esc' || keyName == 'Ctrl-C' || keyName == 'Ctrl-[' ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1425 (keyName == 'Backspace' && query == '')) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1426 vimGlobalState.searchHistoryController.pushInput(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1427 vimGlobalState.searchHistoryController.reset();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1428 updateSearchQuery(cm, originalQuery);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1429 clearSearchHighlight(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1430 cm.scrollTo(originalScrollPos.left, originalScrollPos.top);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1431 CodeMirror.e_stop(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1432 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1433 close();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1434 cm.focus();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1435 } else if (keyName == 'Up' || keyName == 'Down') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1436 CodeMirror.e_stop(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1437 } else if (keyName == 'Ctrl-U') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1438 // Ctrl-U clears input.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1439 CodeMirror.e_stop(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1440 close('');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1441 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1442 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1443 switch (command.searchArgs.querySrc) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1444 case 'prompt':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1445 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1446 if (macroModeState.isPlaying) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1447 var query = macroModeState.replaySearchQueries.shift();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1448 handleQuery(query, true /** ignoreCase */, false /** smartCase */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1449 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1450 showPrompt(cm, {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1451 onClose: onPromptClose,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1452 prefix: promptPrefix,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1453 desc: '(JavaScript regexp)',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1454 onKeyUp: onPromptKeyUp,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1455 onKeyDown: onPromptKeyDown
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1456 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1457 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1458 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1459 case 'wordUnderCursor':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1460 var word = expandWordUnderCursor(cm, false /** inclusive */,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1461 true /** forward */, false /** bigWord */,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1462 true /** noSymbol */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1463 var isKeyword = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1464 if (!word) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1465 word = expandWordUnderCursor(cm, false /** inclusive */,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1466 true /** forward */, false /** bigWord */,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1467 false /** noSymbol */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1468 isKeyword = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1469 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1470 if (!word) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1471 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1472 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1473 var query = cm.getLine(word.start.line).substring(word.start.ch,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1474 word.end.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1475 if (isKeyword && wholeWordOnly) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1476 query = '\\b' + query + '\\b';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1477 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1478 query = escapeRegex(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1479 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1480
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1481 // cachedCursor is used to save the old position of the cursor
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1482 // when * or # causes vim to seek for the nearest word and shift
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1483 // the cursor before entering the motion.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1484 vimGlobalState.jumpList.cachedCursor = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1485 cm.setCursor(word.start);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1486
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1487 handleQuery(query, true /** ignoreCase */, false /** smartCase */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1488 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1489 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1490 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1491 processEx: function(cm, vim, command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1492 function onPromptClose(input) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1493 // Give the prompt some time to close so that if processCommand shows
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1494 // an error, the elements don't overlap.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1495 vimGlobalState.exCommandHistoryController.pushInput(input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1496 vimGlobalState.exCommandHistoryController.reset();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1497 exCommandDispatcher.processCommand(cm, input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1498 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1499 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1500 function onPromptKeyDown(e, input, close) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1501 var keyName = CodeMirror.keyName(e), up, offset;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1502 if (keyName == 'Esc' || keyName == 'Ctrl-C' || keyName == 'Ctrl-[' ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1503 (keyName == 'Backspace' && input == '')) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1504 vimGlobalState.exCommandHistoryController.pushInput(input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1505 vimGlobalState.exCommandHistoryController.reset();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1506 CodeMirror.e_stop(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1507 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1508 close();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1509 cm.focus();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1510 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1511 if (keyName == 'Up' || keyName == 'Down') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1512 CodeMirror.e_stop(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1513 up = keyName == 'Up' ? true : false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1514 offset = e.target ? e.target.selectionEnd : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1515 input = vimGlobalState.exCommandHistoryController.nextMatch(input, up) || '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1516 close(input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1517 if (offset && e.target) e.target.selectionEnd = e.target.selectionStart = Math.min(offset, e.target.value.length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1518 } else if (keyName == 'Ctrl-U') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1519 // Ctrl-U clears input.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1520 CodeMirror.e_stop(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1521 close('');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1522 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1523 if ( keyName != 'Left' && keyName != 'Right' && keyName != 'Ctrl' && keyName != 'Alt' && keyName != 'Shift')
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1524 vimGlobalState.exCommandHistoryController.reset();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1525 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1526 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1527 if (command.type == 'keyToEx') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1528 // Handle user defined Ex to Ex mappings
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1529 exCommandDispatcher.processCommand(cm, command.exArgs.input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1530 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1531 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1532 showPrompt(cm, { onClose: onPromptClose, prefix: ':', value: '\'<,\'>',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1533 onKeyDown: onPromptKeyDown, selectValueOnOpen: false});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1534 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1535 showPrompt(cm, { onClose: onPromptClose, prefix: ':',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1536 onKeyDown: onPromptKeyDown});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1537 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1538 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1539 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1540 evalInput: function(cm, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1541 // If the motion command is set, execute both the operator and motion.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1542 // Otherwise return.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1543 var inputState = vim.inputState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1544 var motion = inputState.motion;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1545 var motionArgs = inputState.motionArgs || {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1546 var operator = inputState.operator;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1547 var operatorArgs = inputState.operatorArgs || {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1548 var registerName = inputState.registerName;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1549 var sel = vim.sel;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1550 // TODO: Make sure cm and vim selections are identical outside visual mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1551 var origHead = copyCursor(vim.visualMode ? clipCursorToContent(cm, sel.head): cm.getCursor('head'));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1552 var origAnchor = copyCursor(vim.visualMode ? clipCursorToContent(cm, sel.anchor) : cm.getCursor('anchor'));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1553 var oldHead = copyCursor(origHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1554 var oldAnchor = copyCursor(origAnchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1555 var newHead, newAnchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1556 var repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1557 if (operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1558 this.recordLastEdit(vim, inputState);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1559 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1560 if (inputState.repeatOverride !== undefined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1561 // If repeatOverride is specified, that takes precedence over the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1562 // input state's repeat. Used by Ex mode and can be user defined.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1563 repeat = inputState.repeatOverride;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1564 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1565 repeat = inputState.getRepeat();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1566 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1567 if (repeat > 0 && motionArgs.explicitRepeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1568 motionArgs.repeatIsExplicit = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1569 } else if (motionArgs.noRepeat ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1570 (!motionArgs.explicitRepeat && repeat === 0)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1571 repeat = 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1572 motionArgs.repeatIsExplicit = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1573 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1574 if (inputState.selectedCharacter) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1575 // If there is a character input, stick it in all of the arg arrays.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1576 motionArgs.selectedCharacter = operatorArgs.selectedCharacter =
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1577 inputState.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1578 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1579 motionArgs.repeat = repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1580 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1581 if (motion) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1582 var motionResult = motions[motion](cm, origHead, motionArgs, vim, inputState);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1583 vim.lastMotion = motions[motion];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1584 if (!motionResult) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1585 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1586 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1587 if (motionArgs.toJumplist) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1588 var jumpList = vimGlobalState.jumpList;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1589 // if the current motion is # or *, use cachedCursor
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1590 var cachedCursor = jumpList.cachedCursor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1591 if (cachedCursor) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1592 recordJumpPosition(cm, cachedCursor, motionResult);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1593 delete jumpList.cachedCursor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1594 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1595 recordJumpPosition(cm, origHead, motionResult);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1596 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1597 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1598 if (motionResult instanceof Array) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1599 newAnchor = motionResult[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1600 newHead = motionResult[1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1601 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1602 newHead = motionResult;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1603 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1604 // TODO: Handle null returns from motion commands better.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1605 if (!newHead) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1606 newHead = copyCursor(origHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1607 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1608 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1609 if (!(vim.visualBlock && newHead.ch === Infinity)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1610 newHead = clipCursorToContent(cm, newHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1611 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1612 if (newAnchor) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1613 newAnchor = clipCursorToContent(cm, newAnchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1614 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1615 newAnchor = newAnchor || oldAnchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1616 sel.anchor = newAnchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1617 sel.head = newHead;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1618 updateCmSelection(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1619 updateMark(cm, vim, '<',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1620 cursorIsBefore(newAnchor, newHead) ? newAnchor
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1621 : newHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1622 updateMark(cm, vim, '>',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1623 cursorIsBefore(newAnchor, newHead) ? newHead
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1624 : newAnchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1625 } else if (!operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1626 newHead = clipCursorToContent(cm, newHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1627 cm.setCursor(newHead.line, newHead.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1628 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1629 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1630 if (operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1631 if (operatorArgs.lastSel) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1632 // Replaying a visual mode operation
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1633 newAnchor = oldAnchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1634 var lastSel = operatorArgs.lastSel;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1635 var lineOffset = Math.abs(lastSel.head.line - lastSel.anchor.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1636 var chOffset = Math.abs(lastSel.head.ch - lastSel.anchor.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1637 if (lastSel.visualLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1638 // Linewise Visual mode: The same number of lines.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1639 newHead = new Pos(oldAnchor.line + lineOffset, oldAnchor.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1640 } else if (lastSel.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1641 // Blockwise Visual mode: The same number of lines and columns.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1642 newHead = new Pos(oldAnchor.line + lineOffset, oldAnchor.ch + chOffset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1643 } else if (lastSel.head.line == lastSel.anchor.line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1644 // Normal Visual mode within one line: The same number of characters.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1645 newHead = new Pos(oldAnchor.line, oldAnchor.ch + chOffset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1646 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1647 // Normal Visual mode with several lines: The same number of lines, in the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1648 // last line the same number of characters as in the last line the last time.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1649 newHead = new Pos(oldAnchor.line + lineOffset, oldAnchor.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1650 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1651 vim.visualMode = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1652 vim.visualLine = lastSel.visualLine;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1653 vim.visualBlock = lastSel.visualBlock;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1654 sel = vim.sel = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1655 anchor: newAnchor,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1656 head: newHead
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1657 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1658 updateCmSelection(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1659 } else if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1660 operatorArgs.lastSel = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1661 anchor: copyCursor(sel.anchor),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1662 head: copyCursor(sel.head),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1663 visualBlock: vim.visualBlock,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1664 visualLine: vim.visualLine
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1665 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1666 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1667 var curStart, curEnd, linewise, mode;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1668 var cmSel;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1669 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1670 // Init visual op
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1671 curStart = cursorMin(sel.head, sel.anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1672 curEnd = cursorMax(sel.head, sel.anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1673 linewise = vim.visualLine || operatorArgs.linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1674 mode = vim.visualBlock ? 'block' :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1675 linewise ? 'line' :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1676 'char';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1677 cmSel = makeCmSelection(cm, {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1678 anchor: curStart,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1679 head: curEnd
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1680 }, mode);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1681 if (linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1682 var ranges = cmSel.ranges;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1683 if (mode == 'block') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1684 // Linewise operators in visual block mode extend to end of line
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1685 for (var i = 0; i < ranges.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1686 ranges[i].head.ch = lineLength(cm, ranges[i].head.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1687 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1688 } else if (mode == 'line') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1689 ranges[0].head = new Pos(ranges[0].head.line + 1, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1690 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1691 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1692 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1693 // Init motion op
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1694 curStart = copyCursor(newAnchor || oldAnchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1695 curEnd = copyCursor(newHead || oldHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1696 if (cursorIsBefore(curEnd, curStart)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1697 var tmp = curStart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1698 curStart = curEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1699 curEnd = tmp;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1700 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1701 linewise = motionArgs.linewise || operatorArgs.linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1702 if (linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1703 // Expand selection to entire line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1704 expandSelectionToLine(cm, curStart, curEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1705 } else if (motionArgs.forward) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1706 // Clip to trailing newlines only if the motion goes forward.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1707 clipToLine(cm, curStart, curEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1708 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1709 mode = 'char';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1710 var exclusive = !motionArgs.inclusive || linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1711 cmSel = makeCmSelection(cm, {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1712 anchor: curStart,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1713 head: curEnd
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1714 }, mode, exclusive);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1715 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1716 cm.setSelections(cmSel.ranges, cmSel.primary);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1717 vim.lastMotion = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1718 operatorArgs.repeat = repeat; // For indent in visual mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1719 operatorArgs.registerName = registerName;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1720 // Keep track of linewise as it affects how paste and change behave.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1721 operatorArgs.linewise = linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1722 var operatorMoveTo = operators[operator](
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1723 cm, operatorArgs, cmSel.ranges, oldAnchor, newHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1724 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1725 exitVisualMode(cm, operatorMoveTo != null);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1726 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1727 if (operatorMoveTo) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1728 cm.setCursor(operatorMoveTo);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1729 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1730 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1731 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1732 recordLastEdit: function(vim, inputState, actionCommand) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1733 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1734 if (macroModeState.isPlaying) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1735 vim.lastEditInputState = inputState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1736 vim.lastEditActionCommand = actionCommand;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1737 macroModeState.lastInsertModeChanges.changes = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1738 macroModeState.lastInsertModeChanges.expectCursorActivityForChange = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1739 macroModeState.lastInsertModeChanges.visualBlock = vim.visualBlock ? vim.sel.head.line - vim.sel.anchor.line : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1740 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1741 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1742
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1743 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1744 * typedef {Object{line:number,ch:number}} Cursor An object containing the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1745 * position of the cursor.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1746 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1747 // All of the functions below return Cursor objects.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1748 var motions = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1749 moveToTopLine: function(cm, _head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1750 var line = getUserVisibleLines(cm).top + motionArgs.repeat -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1751 return new Pos(line, findFirstNonWhiteSpaceCharacter(cm.getLine(line)));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1752 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1753 moveToMiddleLine: function(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1754 var range = getUserVisibleLines(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1755 var line = Math.floor((range.top + range.bottom) * 0.5);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1756 return new Pos(line, findFirstNonWhiteSpaceCharacter(cm.getLine(line)));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1757 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1758 moveToBottomLine: function(cm, _head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1759 var line = getUserVisibleLines(cm).bottom - motionArgs.repeat +1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1760 return new Pos(line, findFirstNonWhiteSpaceCharacter(cm.getLine(line)));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1761 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1762 expandToLine: function(_cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1763 // Expands forward to end of line, and then to next line if repeat is
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1764 // >1. Does not handle backward motion!
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1765 var cur = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1766 return new Pos(cur.line + motionArgs.repeat - 1, Infinity);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1767 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1768 findNext: function(cm, _head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1769 var state = getSearchState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1770 var query = state.getQuery();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1771 if (!query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1772 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1773 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1774 var prev = !motionArgs.forward;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1775 // If search is initiated with ? instead of /, negate direction.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1776 prev = (state.isReversed()) ? !prev : prev;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1777 highlightSearchMatches(cm, query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1778 return findNext(cm, prev/** prev */, query, motionArgs.repeat);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1779 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1780 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1781 * Find and select the next occurrence of the search query. If the cursor is currently
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1782 * within a match, then find and select the current match. Otherwise, find the next occurrence in the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1783 * appropriate direction.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1784 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1785 * This differs from `findNext` in the following ways:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1786 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1787 * 1. Instead of only returning the "from", this returns a "from", "to" range.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1788 * 2. If the cursor is currently inside a search match, this selects the current match
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1789 * instead of the next match.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1790 * 3. If there is no associated operator, this will turn on visual mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1791 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1792 findAndSelectNextInclusive: function(cm, _head, motionArgs, vim, prevInputState) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1793 var state = getSearchState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1794 var query = state.getQuery();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1795
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1796 if (!query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1797 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1798 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1799
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1800 var prev = !motionArgs.forward;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1801 prev = (state.isReversed()) ? !prev : prev;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1802
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1803 // next: [from, to] | null
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1804 var next = findNextFromAndToInclusive(cm, prev, query, motionArgs.repeat, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1805
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1806 // No matches.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1807 if (!next) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1808 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1809 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1810
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1811 // If there's an operator that will be executed, return the selection.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1812 if (prevInputState.operator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1813 return next;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1814 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1815
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1816 // At this point, we know that there is no accompanying operator -- let's
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1817 // deal with visual mode in order to select an appropriate match.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1818
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1819 var from = next[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1820 // For whatever reason, when we use the "to" as returned by searchcursor.js directly,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1821 // the resulting selection is extended by 1 char. Let's shrink it so that only the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1822 // match is selected.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1823 var to = new Pos(next[1].line, next[1].ch - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1824
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1825 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1826 // If we were in visualLine or visualBlock mode, get out of it.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1827 if (vim.visualLine || vim.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1828 vim.visualLine = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1829 vim.visualBlock = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1830 CodeMirror.signal(cm, "vim-mode-change", {mode: "visual", subMode: ""});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1831 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1832
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1833 // If we're currently in visual mode, we should extend the selection to include
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1834 // the search result.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1835 var anchor = vim.sel.anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1836 if (anchor) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1837 if (state.isReversed()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1838 if (motionArgs.forward) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1839 return [anchor, from];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1840 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1841
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1842 return [anchor, to];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1843 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1844 if (motionArgs.forward) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1845 return [anchor, to];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1846 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1847
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1848 return [anchor, from];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1849 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1850 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1851 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1852 // Let's turn visual mode on.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1853 vim.visualMode = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1854 vim.visualLine = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1855 vim.visualBlock = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1856 CodeMirror.signal(cm, "vim-mode-change", {mode: "visual", subMode: ""});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1857 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1858
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1859 return prev ? [to, from] : [from, to];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1860 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1861 goToMark: function(cm, _head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1862 var pos = getMarkPos(cm, vim, motionArgs.selectedCharacter);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1863 if (pos) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1864 return motionArgs.linewise ? { line: pos.line, ch: findFirstNonWhiteSpaceCharacter(cm.getLine(pos.line)) } : pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1865 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1866 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1867 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1868 moveToOtherHighlightedEnd: function(cm, _head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1869 if (vim.visualBlock && motionArgs.sameLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1870 var sel = vim.sel;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1871 return [
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1872 clipCursorToContent(cm, new Pos(sel.anchor.line, sel.head.ch)),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1873 clipCursorToContent(cm, new Pos(sel.head.line, sel.anchor.ch))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1874 ];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1875 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1876 return ([vim.sel.head, vim.sel.anchor]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1877 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1878 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1879 jumpToMark: function(cm, head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1880 var best = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1881 for (var i = 0; i < motionArgs.repeat; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1882 var cursor = best;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1883 for (var key in vim.marks) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1884 if (!isLowerCase(key)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1885 continue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1886 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1887 var mark = vim.marks[key].find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1888 var isWrongDirection = (motionArgs.forward) ?
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1889 cursorIsBefore(mark, cursor) : cursorIsBefore(cursor, mark);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1890
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1891 if (isWrongDirection) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1892 continue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1893 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1894 if (motionArgs.linewise && (mark.line == cursor.line)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1895 continue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1896 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1897
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1898 var equal = cursorEqual(cursor, best);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1899 var between = (motionArgs.forward) ?
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1900 cursorIsBetween(cursor, mark, best) :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1901 cursorIsBetween(best, mark, cursor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1902
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1903 if (equal || between) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1904 best = mark;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1905 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1906 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1907 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1908
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1909 if (motionArgs.linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1910 // Vim places the cursor on the first non-whitespace character of
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1911 // the line if there is one, else it places the cursor at the end
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1912 // of the line, regardless of whether a mark was found.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1913 best = new Pos(best.line, findFirstNonWhiteSpaceCharacter(cm.getLine(best.line)));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1914 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1915 return best;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1916 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1917 moveByCharacters: function(_cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1918 var cur = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1919 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1920 var ch = motionArgs.forward ? cur.ch + repeat : cur.ch - repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1921 return new Pos(cur.line, ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1922 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1923 moveByLines: function(cm, head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1924 var cur = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1925 var endCh = cur.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1926 // Depending what our last motion was, we may want to do different
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1927 // things. If our last motion was moving vertically, we want to
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1928 // preserve the HPos from our last horizontal move. If our last motion
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1929 // was going to the end of a line, moving vertically we should go to
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1930 // the end of the line, etc.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1931 switch (vim.lastMotion) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1932 case this.moveByLines:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1933 case this.moveByDisplayLines:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1934 case this.moveByScroll:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1935 case this.moveToColumn:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1936 case this.moveToEol:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1937 endCh = vim.lastHPos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1938 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1939 default:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1940 vim.lastHPos = endCh;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1941 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1942 var repeat = motionArgs.repeat+(motionArgs.repeatOffset||0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1943 var line = motionArgs.forward ? cur.line + repeat : cur.line - repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1944 var first = cm.firstLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1945 var last = cm.lastLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1946 var posV = cm.findPosV(cur, (motionArgs.forward ? repeat : -repeat), 'line', vim.lastHSPos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1947 var hasMarkedText = motionArgs.forward ? posV.line > line : posV.line < line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1948 if (hasMarkedText) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1949 line = posV.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1950 endCh = posV.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1951 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1952 // Vim go to line begin or line end when cursor at first/last line and
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1953 // move to previous/next line is triggered.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1954 if (line < first && cur.line == first){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1955 return this.moveToStartOfLine(cm, head, motionArgs, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1956 } else if (line > last && cur.line == last){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1957 return moveToEol(cm, head, motionArgs, vim, true);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1958 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1959 if (motionArgs.toFirstChar){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1960 endCh=findFirstNonWhiteSpaceCharacter(cm.getLine(line));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1961 vim.lastHPos = endCh;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1962 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1963 vim.lastHSPos = cm.charCoords(new Pos(line, endCh),'div').left;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1964 return new Pos(line, endCh);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1965 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1966 moveByDisplayLines: function(cm, head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1967 var cur = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1968 switch (vim.lastMotion) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1969 case this.moveByDisplayLines:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1970 case this.moveByScroll:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1971 case this.moveByLines:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1972 case this.moveToColumn:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1973 case this.moveToEol:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1974 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1975 default:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1976 vim.lastHSPos = cm.charCoords(cur,'div').left;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1977 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1978 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1979 var res=cm.findPosV(cur,(motionArgs.forward ? repeat : -repeat),'line',vim.lastHSPos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1980 if (res.hitSide) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1981 if (motionArgs.forward) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1982 var lastCharCoords = cm.charCoords(res, 'div');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1983 var goalCoords = { top: lastCharCoords.top + 8, left: vim.lastHSPos };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1984 var res = cm.coordsChar(goalCoords, 'div');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1985 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1986 var resCoords = cm.charCoords(new Pos(cm.firstLine(), 0), 'div');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1987 resCoords.left = vim.lastHSPos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1988 res = cm.coordsChar(resCoords, 'div');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1989 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1990 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1991 vim.lastHPos = res.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1992 return res;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1993 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1994 moveByPage: function(cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1995 // CodeMirror only exposes functions that move the cursor page down, so
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1996 // doing this bad hack to move the cursor and move it back. evalInput
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1997 // will move the cursor to where it should be in the end.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1998 var curStart = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1999 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2000 return cm.findPosV(curStart, (motionArgs.forward ? repeat : -repeat), 'page');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2001 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2002 moveByParagraph: function(cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2003 var dir = motionArgs.forward ? 1 : -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2004 return findParagraph(cm, head, motionArgs.repeat, dir);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2005 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2006 moveBySentence: function(cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2007 var dir = motionArgs.forward ? 1 : -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2008 return findSentence(cm, head, motionArgs.repeat, dir);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2009 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2010 moveByScroll: function(cm, head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2011 var scrollbox = cm.getScrollInfo();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2012 var curEnd = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2013 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2014 if (!repeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2015 repeat = scrollbox.clientHeight / (2 * cm.defaultTextHeight());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2016 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2017 var orig = cm.charCoords(head, 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2018 motionArgs.repeat = repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2019 curEnd = motions.moveByDisplayLines(cm, head, motionArgs, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2020 if (!curEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2021 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2022 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2023 var dest = cm.charCoords(curEnd, 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2024 cm.scrollTo(null, scrollbox.top + dest.top - orig.top);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2025 return curEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2026 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2027 moveByWords: function(cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2028 return moveToWord(cm, head, motionArgs.repeat, !!motionArgs.forward,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2029 !!motionArgs.wordEnd, !!motionArgs.bigWord);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2030 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2031 moveTillCharacter: function(cm, _head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2032 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2033 var curEnd = moveToCharacter(cm, repeat, motionArgs.forward,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2034 motionArgs.selectedCharacter);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2035 var increment = motionArgs.forward ? -1 : 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2036 recordLastCharacterSearch(increment, motionArgs);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2037 if (!curEnd) return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2038 curEnd.ch += increment;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2039 return curEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2040 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2041 moveToCharacter: function(cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2042 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2043 recordLastCharacterSearch(0, motionArgs);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2044 return moveToCharacter(cm, repeat, motionArgs.forward,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2045 motionArgs.selectedCharacter) || head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2046 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2047 moveToSymbol: function(cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2048 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2049 return findSymbol(cm, repeat, motionArgs.forward,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2050 motionArgs.selectedCharacter) || head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2051 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2052 moveToColumn: function(cm, head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2053 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2054 // repeat is equivalent to which column we want to move to!
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2055 vim.lastHPos = repeat - 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2056 vim.lastHSPos = cm.charCoords(head,'div').left;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2057 return moveToColumn(cm, repeat);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2058 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2059 moveToEol: function(cm, head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2060 return moveToEol(cm, head, motionArgs, vim, false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2061 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2062 moveToFirstNonWhiteSpaceCharacter: function(cm, head) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2063 // Go to the start of the line where the text begins, or the end for
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2064 // whitespace-only lines
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2065 var cursor = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2066 return new Pos(cursor.line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2067 findFirstNonWhiteSpaceCharacter(cm.getLine(cursor.line)));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2068 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2069 moveToMatchedSymbol: function(cm, head) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2070 var cursor = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2071 var line = cursor.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2072 var ch = cursor.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2073 var lineText = cm.getLine(line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2074 var symbol;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2075 for (; ch < lineText.length; ch++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2076 symbol = lineText.charAt(ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2077 if (symbol && isMatchableSymbol(symbol)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2078 var style = cm.getTokenTypeAt(new Pos(line, ch + 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2079 if (style !== "string" && style !== "comment") {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2080 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2081 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2082 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2083 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2084 if (ch < lineText.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2085 // Only include angle brackets in analysis if they are being matched.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2086 var re = (ch === '<' || ch === '>') ? /[(){}[\]<>]/ : /[(){}[\]]/;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2087 var matched = cm.findMatchingBracket(new Pos(line, ch), {bracketRegex: re});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2088 return matched.to;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2089 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2090 return cursor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2091 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2092 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2093 moveToStartOfLine: function(_cm, head) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2094 return new Pos(head.line, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2095 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2096 moveToLineOrEdgeOfDocument: function(cm, _head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2097 var lineNum = motionArgs.forward ? cm.lastLine() : cm.firstLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2098 if (motionArgs.repeatIsExplicit) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2099 lineNum = motionArgs.repeat - cm.getOption('firstLineNumber');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2100 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2101 return new Pos(lineNum,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2102 findFirstNonWhiteSpaceCharacter(cm.getLine(lineNum)));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2103 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2104 moveToStartOfDisplayLine: function(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2105 cm.execCommand("goLineLeft");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2106 return cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2107 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2108 moveToEndOfDisplayLine: function(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2109 cm.execCommand("goLineRight");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2110 var head = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2111 if (head.sticky == "before") head.ch--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2112 return head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2113 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2114 textObjectManipulation: function(cm, head, motionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2115 // TODO: lots of possible exceptions that can be thrown here. Try da(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2116 // outside of a () block.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2117 var mirroredPairs = {'(': ')', ')': '(',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2118 '{': '}', '}': '{',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2119 '[': ']', ']': '[',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2120 '<': '>', '>': '<'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2121 var selfPaired = {'\'': true, '"': true, '`': true};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2122
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2123 var character = motionArgs.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2124 // 'b' refers to '()' block.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2125 // 'B' refers to '{}' block.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2126 if (character == 'b') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2127 character = '(';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2128 } else if (character == 'B') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2129 character = '{';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2130 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2131
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2132 // Inclusive is the difference between a and i
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2133 // TODO: Instead of using the additional text object map to perform text
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2134 // object operations, merge the map into the defaultKeyMap and use
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2135 // motionArgs to define behavior. Define separate entries for 'aw',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2136 // 'iw', 'a[', 'i[', etc.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2137 var inclusive = !motionArgs.textObjectInner;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2138
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2139 var tmp;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2140 if (mirroredPairs[character]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2141 tmp = selectCompanionObject(cm, head, character, inclusive);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2142 } else if (selfPaired[character]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2143 tmp = findBeginningAndEnd(cm, head, character, inclusive);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2144 } else if (character === 'W') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2145 tmp = expandWordUnderCursor(cm, inclusive, true /** forward */,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2146 true /** bigWord */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2147 } else if (character === 'w') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2148 tmp = expandWordUnderCursor(cm, inclusive, true /** forward */,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2149 false /** bigWord */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2150 } else if (character === 'p') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2151 tmp = findParagraph(cm, head, motionArgs.repeat, 0, inclusive);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2152 motionArgs.linewise = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2153 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2154 if (!vim.visualLine) { vim.visualLine = true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2155 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2156 var operatorArgs = vim.inputState.operatorArgs;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2157 if (operatorArgs) { operatorArgs.linewise = true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2158 tmp.end.line--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2159 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2160 } else if (character === 't') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2161 tmp = expandTagUnderCursor(cm, head, inclusive);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2162 } else if (character === 's') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2163 // account for cursor on end of sentence symbol
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2164 var content = cm.getLine(head.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2165 if (head.ch > 0 && isEndOfSentenceSymbol(content[head.ch])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2166 head.ch -= 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2167 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2168 var end = getSentence(cm, head, motionArgs.repeat, 1, inclusive);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2169 var start = getSentence(cm, head, motionArgs.repeat, -1, inclusive);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2170 // closer vim behaviour, 'a' only takes the space after the sentence if there is one before and after
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2171 if (isWhiteSpaceString(cm.getLine(start.line)[start.ch])
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2172 && isWhiteSpaceString(cm.getLine(end.line)[end.ch -1])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2173 start = {line: start.line, ch: start.ch + 1};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2174 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2175 tmp = {start: start, end: end};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2176 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2177 // No text object defined for this, don't move.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2178 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2179 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2180
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2181 if (!cm.state.vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2182 return [tmp.start, tmp.end];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2183 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2184 return expandSelection(cm, tmp.start, tmp.end);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2185 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2186 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2187
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2188 repeatLastCharacterSearch: function(cm, head, motionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2189 var lastSearch = vimGlobalState.lastCharacterSearch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2190 var repeat = motionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2191 var forward = motionArgs.forward === lastSearch.forward;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2192 var increment = (lastSearch.increment ? 1 : 0) * (forward ? -1 : 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2193 cm.moveH(-increment, 'char');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2194 motionArgs.inclusive = forward ? true : false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2195 var curEnd = moveToCharacter(cm, repeat, forward, lastSearch.selectedCharacter);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2196 if (!curEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2197 cm.moveH(increment, 'char');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2198 return head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2199 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2200 curEnd.ch += increment;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2201 return curEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2202 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2203 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2204
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2205 function defineMotion(name, fn) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2206 motions[name] = fn;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2207 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2208
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2209 function fillArray(val, times) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2210 var arr = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2211 for (var i = 0; i < times; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2212 arr.push(val);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2213 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2214 return arr;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2215 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2216 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2217 * An operator acts on a text selection. It receives the list of selections
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2218 * as input. The corresponding CodeMirror selection is guaranteed to
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2219 * match the input selection.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2220 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2221 var operators = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2222 change: function(cm, args, ranges) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2223 var finalHead, text;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2224 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2225 var anchor = ranges[0].anchor,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2226 head = ranges[0].head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2227 if (!vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2228 text = cm.getRange(anchor, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2229 var lastState = vim.lastEditInputState || {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2230 if (lastState.motion == "moveByWords" && !isWhiteSpaceString(text)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2231 // Exclude trailing whitespace if the range is not all whitespace.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2232 var match = (/\s+$/).exec(text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2233 if (match && lastState.motionArgs && lastState.motionArgs.forward) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2234 head = offsetCursor(head, 0, - match[0].length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2235 text = text.slice(0, - match[0].length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2236 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2237 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2238 var prevLineEnd = new Pos(anchor.line - 1, Number.MAX_VALUE);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2239 var wasLastLine = cm.firstLine() == cm.lastLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2240 if (head.line > cm.lastLine() && args.linewise && !wasLastLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2241 cm.replaceRange('', prevLineEnd, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2242 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2243 cm.replaceRange('', anchor, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2244 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2245 if (args.linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2246 // Push the next line back down, if there is a next line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2247 if (!wasLastLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2248 cm.setCursor(prevLineEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2249 CodeMirror.commands.newlineAndIndent(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2250 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2251 // make sure cursor ends up at the end of the line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2252 anchor.ch = Number.MAX_VALUE;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2253 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2254 finalHead = anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2255 } else if (args.fullLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2256 head.ch = Number.MAX_VALUE;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2257 head.line--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2258 cm.setSelection(anchor, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2259 text = cm.getSelection();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2260 cm.replaceSelection("");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2261 finalHead = anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2262 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2263 text = cm.getSelection();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2264 var replacement = fillArray('', ranges.length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2265 cm.replaceSelections(replacement);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2266 finalHead = cursorMin(ranges[0].head, ranges[0].anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2267 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2268 vimGlobalState.registerController.pushText(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2269 args.registerName, 'change', text,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2270 args.linewise, ranges.length > 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2271 actions.enterInsertMode(cm, {head: finalHead}, cm.state.vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2272 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2273 // delete is a javascript keyword.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2274 'delete': function(cm, args, ranges) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2275 var finalHead, text;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2276 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2277 if (!vim.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2278 var anchor = ranges[0].anchor,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2279 head = ranges[0].head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2280 if (args.linewise &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2281 head.line != cm.firstLine() &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2282 anchor.line == cm.lastLine() &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2283 anchor.line == head.line - 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2284 // Special case for dd on last line (and first line).
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2285 if (anchor.line == cm.firstLine()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2286 anchor.ch = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2287 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2288 anchor = new Pos(anchor.line - 1, lineLength(cm, anchor.line - 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2289 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2290 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2291 text = cm.getRange(anchor, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2292 cm.replaceRange('', anchor, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2293 finalHead = anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2294 if (args.linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2295 finalHead = motions.moveToFirstNonWhiteSpaceCharacter(cm, anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2296 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2297 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2298 text = cm.getSelection();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2299 var replacement = fillArray('', ranges.length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2300 cm.replaceSelections(replacement);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2301 finalHead = cursorMin(ranges[0].head, ranges[0].anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2302 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2303 vimGlobalState.registerController.pushText(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2304 args.registerName, 'delete', text,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2305 args.linewise, vim.visualBlock);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2306 return clipCursorToContent(cm, finalHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2307 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2308 indent: function(cm, args, ranges) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2309 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2310 if (cm.indentMore) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2311 var repeat = (vim.visualMode) ? args.repeat : 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2312 for (var j = 0; j < repeat; j++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2313 if (args.indentRight) cm.indentMore();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2314 else cm.indentLess();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2315 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2316 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2317 var startLine = ranges[0].anchor.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2318 var endLine = vim.visualBlock ?
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2319 ranges[ranges.length - 1].anchor.line :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2320 ranges[0].head.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2321 // In visual mode, n> shifts the selection right n times, instead of
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2322 // shifting n lines right once.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2323 var repeat = (vim.visualMode) ? args.repeat : 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2324 if (args.linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2325 // The only way to delete a newline is to delete until the start of
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2326 // the next line, so in linewise mode evalInput will include the next
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2327 // line. We don't want this in indent, so we go back a line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2328 endLine--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2329 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2330 for (var i = startLine; i <= endLine; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2331 for (var j = 0; j < repeat; j++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2332 cm.indentLine(i, args.indentRight);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2333 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2334 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2335 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2336 return motions.moveToFirstNonWhiteSpaceCharacter(cm, ranges[0].anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2337 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2338 indentAuto: function(cm, _args, ranges) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2339 cm.execCommand("indentAuto");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2340 return motions.moveToFirstNonWhiteSpaceCharacter(cm, ranges[0].anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2341 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2342 changeCase: function(cm, args, ranges, oldAnchor, newHead) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2343 var selections = cm.getSelections();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2344 var swapped = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2345 var toLower = args.toLower;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2346 for (var j = 0; j < selections.length; j++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2347 var toSwap = selections[j];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2348 var text = '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2349 if (toLower === true) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2350 text = toSwap.toLowerCase();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2351 } else if (toLower === false) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2352 text = toSwap.toUpperCase();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2353 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2354 for (var i = 0; i < toSwap.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2355 var character = toSwap.charAt(i);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2356 text += isUpperCase(character) ? character.toLowerCase() :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2357 character.toUpperCase();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2358 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2359 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2360 swapped.push(text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2361 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2362 cm.replaceSelections(swapped);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2363 if (args.shouldMoveCursor){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2364 return newHead;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2365 } else if (!cm.state.vim.visualMode && args.linewise && ranges[0].anchor.line + 1 == ranges[0].head.line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2366 return motions.moveToFirstNonWhiteSpaceCharacter(cm, oldAnchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2367 } else if (args.linewise){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2368 return oldAnchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2369 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2370 return cursorMin(ranges[0].anchor, ranges[0].head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2371 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2372 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2373 yank: function(cm, args, ranges, oldAnchor) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2374 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2375 var text = cm.getSelection();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2376 var endPos = vim.visualMode
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2377 ? cursorMin(vim.sel.anchor, vim.sel.head, ranges[0].head, ranges[0].anchor)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2378 : oldAnchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2379 vimGlobalState.registerController.pushText(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2380 args.registerName, 'yank',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2381 text, args.linewise, vim.visualBlock);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2382 return endPos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2383 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2384 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2385
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2386 function defineOperator(name, fn) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2387 operators[name] = fn;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2388 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2389
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2390 var actions = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2391 jumpListWalk: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2392 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2393 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2394 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2395 var repeat = actionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2396 var forward = actionArgs.forward;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2397 var jumpList = vimGlobalState.jumpList;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2398
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2399 var mark = jumpList.move(cm, forward ? repeat : -repeat);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2400 var markPos = mark ? mark.find() : undefined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2401 markPos = markPos ? markPos : cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2402 cm.setCursor(markPos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2403 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2404 scroll: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2405 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2406 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2407 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2408 var repeat = actionArgs.repeat || 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2409 var lineHeight = cm.defaultTextHeight();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2410 var top = cm.getScrollInfo().top;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2411 var delta = lineHeight * repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2412 var newPos = actionArgs.forward ? top + delta : top - delta;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2413 var cursor = copyCursor(cm.getCursor());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2414 var cursorCoords = cm.charCoords(cursor, 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2415 if (actionArgs.forward) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2416 if (newPos > cursorCoords.top) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2417 cursor.line += (newPos - cursorCoords.top) / lineHeight;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2418 cursor.line = Math.ceil(cursor.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2419 cm.setCursor(cursor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2420 cursorCoords = cm.charCoords(cursor, 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2421 cm.scrollTo(null, cursorCoords.top);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2422 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2423 // Cursor stays within bounds. Just reposition the scroll window.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2424 cm.scrollTo(null, newPos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2425 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2426 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2427 var newBottom = newPos + cm.getScrollInfo().clientHeight;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2428 if (newBottom < cursorCoords.bottom) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2429 cursor.line -= (cursorCoords.bottom - newBottom) / lineHeight;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2430 cursor.line = Math.floor(cursor.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2431 cm.setCursor(cursor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2432 cursorCoords = cm.charCoords(cursor, 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2433 cm.scrollTo(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2434 null, cursorCoords.bottom - cm.getScrollInfo().clientHeight);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2435 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2436 // Cursor stays within bounds. Just reposition the scroll window.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2437 cm.scrollTo(null, newPos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2438 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2439 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2440 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2441 scrollToCursor: function(cm, actionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2442 var lineNum = cm.getCursor().line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2443 var charCoords = cm.charCoords(new Pos(lineNum, 0), 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2444 var height = cm.getScrollInfo().clientHeight;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2445 var y = charCoords.top;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2446 switch (actionArgs.position) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2447 case 'center': y = charCoords.bottom - height / 2;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2448 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2449 case 'bottom':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2450 var lineLastCharPos = new Pos(lineNum, cm.getLine(lineNum).length - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2451 var lineLastCharCoords = cm.charCoords(lineLastCharPos, 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2452 var lineHeight = lineLastCharCoords.bottom - y;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2453 y = y - height + lineHeight;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2454 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2455 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2456 cm.scrollTo(null, y);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2457 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2458 replayMacro: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2459 var registerName = actionArgs.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2460 var repeat = actionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2461 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2462 if (registerName == '@') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2463 registerName = macroModeState.latestRegister;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2464 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2465 macroModeState.latestRegister = registerName;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2466 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2467 while(repeat--){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2468 executeMacroRegister(cm, vim, macroModeState, registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2469 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2470 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2471 enterMacroRecordMode: function(cm, actionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2472 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2473 var registerName = actionArgs.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2474 if (vimGlobalState.registerController.isValidRegister(registerName)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2475 macroModeState.enterMacroRecordMode(cm, registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2476 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2477 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2478 toggleOverwrite: function(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2479 if (!cm.state.overwrite) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2480 cm.toggleOverwrite(true);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2481 cm.setOption('keyMap', 'vim-replace');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2482 CodeMirror.signal(cm, "vim-mode-change", {mode: "replace"});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2483 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2484 cm.toggleOverwrite(false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2485 cm.setOption('keyMap', 'vim-insert');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2486 CodeMirror.signal(cm, "vim-mode-change", {mode: "insert"});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2487 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2488 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2489 enterInsertMode: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2490 if (cm.getOption('readOnly')) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2491 vim.insertMode = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2492 vim.insertModeRepeat = actionArgs && actionArgs.repeat || 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2493 var insertAt = (actionArgs) ? actionArgs.insertAt : null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2494 var sel = vim.sel;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2495 var head = actionArgs.head || cm.getCursor('head');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2496 var height = cm.listSelections().length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2497 if (insertAt == 'eol') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2498 head = new Pos(head.line, lineLength(cm, head.line));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2499 } else if (insertAt == 'bol') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2500 head = new Pos(head.line, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2501 } else if (insertAt == 'charAfter') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2502 head = offsetCursor(head, 0, 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2503 } else if (insertAt == 'firstNonBlank') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2504 head = motions.moveToFirstNonWhiteSpaceCharacter(cm, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2505 } else if (insertAt == 'startOfSelectedArea') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2506 if (!vim.visualMode)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2507 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2508 if (!vim.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2509 if (sel.head.line < sel.anchor.line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2510 head = sel.head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2511 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2512 head = new Pos(sel.anchor.line, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2513 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2514 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2515 head = new Pos(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2516 Math.min(sel.head.line, sel.anchor.line),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2517 Math.min(sel.head.ch, sel.anchor.ch));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2518 height = Math.abs(sel.head.line - sel.anchor.line) + 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2519 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2520 } else if (insertAt == 'endOfSelectedArea') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2521 if (!vim.visualMode)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2522 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2523 if (!vim.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2524 if (sel.head.line >= sel.anchor.line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2525 head = offsetCursor(sel.head, 0, 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2526 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2527 head = new Pos(sel.anchor.line, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2528 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2529 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2530 head = new Pos(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2531 Math.min(sel.head.line, sel.anchor.line),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2532 Math.max(sel.head.ch, sel.anchor.ch) + 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2533 height = Math.abs(sel.head.line - sel.anchor.line) + 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2534 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2535 } else if (insertAt == 'inplace') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2536 if (vim.visualMode){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2537 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2538 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2539 } else if (insertAt == 'lastEdit') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2540 head = getLastEditPos(cm) || head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2541 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2542 cm.setOption('disableInput', false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2543 if (actionArgs && actionArgs.replace) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2544 // Handle Replace-mode as a special case of insert mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2545 cm.toggleOverwrite(true);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2546 cm.setOption('keyMap', 'vim-replace');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2547 CodeMirror.signal(cm, "vim-mode-change", {mode: "replace"});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2548 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2549 cm.toggleOverwrite(false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2550 cm.setOption('keyMap', 'vim-insert');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2551 CodeMirror.signal(cm, "vim-mode-change", {mode: "insert"});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2552 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2553 if (!vimGlobalState.macroModeState.isPlaying) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2554 // Only record if not replaying.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2555 cm.on('change', onChange);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2556 CodeMirror.on(cm.getInputField(), 'keydown', onKeyEventTargetKeyDown);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2557 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2558 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2559 exitVisualMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2560 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2561 selectForInsert(cm, head, height);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2562 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2563 toggleVisualMode: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2564 var repeat = actionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2565 var anchor = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2566 var head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2567 // TODO: The repeat should actually select number of characters/lines
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2568 // equal to the repeat times the size of the previous visual
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2569 // operation.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2570 if (!vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2571 // Entering visual mode
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2572 vim.visualMode = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2573 vim.visualLine = !!actionArgs.linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2574 vim.visualBlock = !!actionArgs.blockwise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2575 head = clipCursorToContent(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2576 cm, new Pos(anchor.line, anchor.ch + repeat - 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2577 vim.sel = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2578 anchor: anchor,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2579 head: head
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2580 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2581 CodeMirror.signal(cm, "vim-mode-change", {mode: "visual", subMode: vim.visualLine ? "linewise" : vim.visualBlock ? "blockwise" : ""});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2582 updateCmSelection(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2583 updateMark(cm, vim, '<', cursorMin(anchor, head));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2584 updateMark(cm, vim, '>', cursorMax(anchor, head));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2585 } else if (vim.visualLine ^ actionArgs.linewise ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2586 vim.visualBlock ^ actionArgs.blockwise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2587 // Toggling between modes
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2588 vim.visualLine = !!actionArgs.linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2589 vim.visualBlock = !!actionArgs.blockwise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2590 CodeMirror.signal(cm, "vim-mode-change", {mode: "visual", subMode: vim.visualLine ? "linewise" : vim.visualBlock ? "blockwise" : ""});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2591 updateCmSelection(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2592 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2593 exitVisualMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2594 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2595 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2596 reselectLastSelection: function(cm, _actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2597 var lastSelection = vim.lastSelection;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2598 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2599 updateLastSelection(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2600 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2601 if (lastSelection) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2602 var anchor = lastSelection.anchorMark.find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2603 var head = lastSelection.headMark.find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2604 if (!anchor || !head) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2605 // If the marks have been destroyed due to edits, do nothing.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2606 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2607 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2608 vim.sel = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2609 anchor: anchor,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2610 head: head
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2611 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2612 vim.visualMode = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2613 vim.visualLine = lastSelection.visualLine;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2614 vim.visualBlock = lastSelection.visualBlock;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2615 updateCmSelection(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2616 updateMark(cm, vim, '<', cursorMin(anchor, head));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2617 updateMark(cm, vim, '>', cursorMax(anchor, head));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2618 CodeMirror.signal(cm, 'vim-mode-change', {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2619 mode: 'visual',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2620 subMode: vim.visualLine ? 'linewise' :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2621 vim.visualBlock ? 'blockwise' : ''});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2622 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2623 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2624 joinLines: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2625 var curStart, curEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2626 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2627 curStart = cm.getCursor('anchor');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2628 curEnd = cm.getCursor('head');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2629 if (cursorIsBefore(curEnd, curStart)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2630 var tmp = curEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2631 curEnd = curStart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2632 curStart = tmp;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2633 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2634 curEnd.ch = lineLength(cm, curEnd.line) - 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2635 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2636 // Repeat is the number of lines to join. Minimum 2 lines.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2637 var repeat = Math.max(actionArgs.repeat, 2);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2638 curStart = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2639 curEnd = clipCursorToContent(cm, new Pos(curStart.line + repeat - 1,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2640 Infinity));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2641 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2642 var finalCh = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2643 for (var i = curStart.line; i < curEnd.line; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2644 finalCh = lineLength(cm, curStart.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2645 var tmp = new Pos(curStart.line + 1,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2646 lineLength(cm, curStart.line + 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2647 var text = cm.getRange(curStart, tmp);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2648 text = actionArgs.keepSpaces
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2649 ? text.replace(/\n\r?/g, '')
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2650 : text.replace(/\n\s*/g, ' ');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2651 cm.replaceRange(text, curStart, tmp);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2652 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2653 var curFinalPos = new Pos(curStart.line, finalCh);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2654 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2655 exitVisualMode(cm, false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2656 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2657 cm.setCursor(curFinalPos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2658 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2659 newLineAndEnterInsertMode: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2660 vim.insertMode = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2661 var insertAt = copyCursor(cm.getCursor());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2662 if (insertAt.line === cm.firstLine() && !actionArgs.after) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2663 // Special case for inserting newline before start of document.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2664 cm.replaceRange('\n', new Pos(cm.firstLine(), 0));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2665 cm.setCursor(cm.firstLine(), 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2666 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2667 insertAt.line = (actionArgs.after) ? insertAt.line :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2668 insertAt.line - 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2669 insertAt.ch = lineLength(cm, insertAt.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2670 cm.setCursor(insertAt);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2671 var newlineFn = CodeMirror.commands.newlineAndIndentContinueComment ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2672 CodeMirror.commands.newlineAndIndent;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2673 newlineFn(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2674 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2675 this.enterInsertMode(cm, { repeat: actionArgs.repeat }, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2676 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2677 paste: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2678 var cur = copyCursor(cm.getCursor());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2679 var register = vimGlobalState.registerController.getRegister(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2680 actionArgs.registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2681 var text = register.toString();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2682 if (!text) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2683 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2684 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2685 if (actionArgs.matchIndent) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2686 var tabSize = cm.getOption("tabSize");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2687 // length that considers tabs and tabSize
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2688 var whitespaceLength = function(str) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2689 var tabs = (str.split("\t").length - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2690 var spaces = (str.split(" ").length - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2691 return tabs * tabSize + spaces * 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2692 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2693 var currentLine = cm.getLine(cm.getCursor().line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2694 var indent = whitespaceLength(currentLine.match(/^\s*/)[0]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2695 // chomp last newline b/c don't want it to match /^\s*/gm
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2696 var chompedText = text.replace(/\n$/, '');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2697 var wasChomped = text !== chompedText;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2698 var firstIndent = whitespaceLength(text.match(/^\s*/)[0]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2699 var text = chompedText.replace(/^\s*/gm, function(wspace) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2700 var newIndent = indent + (whitespaceLength(wspace) - firstIndent);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2701 if (newIndent < 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2702 return "";
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2703 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2704 else if (cm.getOption("indentWithTabs")) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2705 var quotient = Math.floor(newIndent / tabSize);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2706 return Array(quotient + 1).join('\t');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2707 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2708 else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2709 return Array(newIndent + 1).join(' ');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2710 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2711 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2712 text += wasChomped ? "\n" : "";
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2713 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2714 if (actionArgs.repeat > 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2715 var text = Array(actionArgs.repeat + 1).join(text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2716 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2717 var linewise = register.linewise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2718 var blockwise = register.blockwise;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2719 if (blockwise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2720 text = text.split('\n');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2721 if (linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2722 text.pop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2723 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2724 for (var i = 0; i < text.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2725 text[i] = (text[i] == '') ? ' ' : text[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2726 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2727 cur.ch += actionArgs.after ? 1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2728 cur.ch = Math.min(lineLength(cm, cur.line), cur.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2729 } else if (linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2730 if(vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2731 text = vim.visualLine ? text.slice(0, -1) : '\n' + text.slice(0, text.length - 1) + '\n';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2732 } else if (actionArgs.after) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2733 // Move the newline at the end to the start instead, and paste just
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2734 // before the newline character of the line we are on right now.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2735 text = '\n' + text.slice(0, text.length - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2736 cur.ch = lineLength(cm, cur.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2737 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2738 cur.ch = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2739 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2740 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2741 cur.ch += actionArgs.after ? 1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2742 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2743 var curPosFinal;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2744 var idx;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2745 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2746 // save the pasted text for reselection if the need arises
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2747 vim.lastPastedText = text;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2748 var lastSelectionCurEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2749 var selectedArea = getSelectedAreaRange(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2750 var selectionStart = selectedArea[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2751 var selectionEnd = selectedArea[1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2752 var selectedText = cm.getSelection();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2753 var selections = cm.listSelections();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2754 var emptyStrings = new Array(selections.length).join('1').split('1');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2755 // save the curEnd marker before it get cleared due to cm.replaceRange.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2756 if (vim.lastSelection) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2757 lastSelectionCurEnd = vim.lastSelection.headMark.find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2758 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2759 // push the previously selected text to unnamed register
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2760 vimGlobalState.registerController.unnamedRegister.setText(selectedText);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2761 if (blockwise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2762 // first delete the selected text
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2763 cm.replaceSelections(emptyStrings);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2764 // Set new selections as per the block length of the yanked text
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2765 selectionEnd = new Pos(selectionStart.line + text.length-1, selectionStart.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2766 cm.setCursor(selectionStart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2767 selectBlock(cm, selectionEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2768 cm.replaceSelections(text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2769 curPosFinal = selectionStart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2770 } else if (vim.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2771 cm.replaceSelections(emptyStrings);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2772 cm.setCursor(selectionStart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2773 cm.replaceRange(text, selectionStart, selectionStart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2774 curPosFinal = selectionStart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2775 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2776 cm.replaceRange(text, selectionStart, selectionEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2777 curPosFinal = cm.posFromIndex(cm.indexFromPos(selectionStart) + text.length - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2778 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2779 // restore the the curEnd marker
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2780 if(lastSelectionCurEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2781 vim.lastSelection.headMark = cm.setBookmark(lastSelectionCurEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2782 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2783 if (linewise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2784 curPosFinal.ch=0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2785 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2786 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2787 if (blockwise) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2788 cm.setCursor(cur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2789 for (var i = 0; i < text.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2790 var line = cur.line+i;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2791 if (line > cm.lastLine()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2792 cm.replaceRange('\n', new Pos(line, 0));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2793 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2794 var lastCh = lineLength(cm, line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2795 if (lastCh < cur.ch) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2796 extendLineToColumn(cm, line, cur.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2797 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2798 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2799 cm.setCursor(cur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2800 selectBlock(cm, new Pos(cur.line + text.length-1, cur.ch));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2801 cm.replaceSelections(text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2802 curPosFinal = cur;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2803 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2804 cm.replaceRange(text, cur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2805 // Now fine tune the cursor to where we want it.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2806 if (linewise && actionArgs.after) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2807 curPosFinal = new Pos(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2808 cur.line + 1,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2809 findFirstNonWhiteSpaceCharacter(cm.getLine(cur.line + 1)));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2810 } else if (linewise && !actionArgs.after) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2811 curPosFinal = new Pos(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2812 cur.line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2813 findFirstNonWhiteSpaceCharacter(cm.getLine(cur.line)));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2814 } else if (!linewise && actionArgs.after) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2815 idx = cm.indexFromPos(cur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2816 curPosFinal = cm.posFromIndex(idx + text.length - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2817 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2818 idx = cm.indexFromPos(cur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2819 curPosFinal = cm.posFromIndex(idx + text.length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2820 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2821 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2822 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2823 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2824 exitVisualMode(cm, false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2825 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2826 cm.setCursor(curPosFinal);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2827 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2828 undo: function(cm, actionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2829 cm.operation(function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2830 repeatFn(cm, CodeMirror.commands.undo, actionArgs.repeat)();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2831 cm.setCursor(cm.getCursor('anchor'));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2832 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2833 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2834 redo: function(cm, actionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2835 repeatFn(cm, CodeMirror.commands.redo, actionArgs.repeat)();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2836 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2837 setRegister: function(_cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2838 vim.inputState.registerName = actionArgs.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2839 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2840 setMark: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2841 var markName = actionArgs.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2842 updateMark(cm, vim, markName, cm.getCursor());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2843 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2844 replace: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2845 var replaceWith = actionArgs.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2846 var curStart = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2847 var replaceTo;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2848 var curEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2849 var selections = cm.listSelections();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2850 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2851 curStart = cm.getCursor('start');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2852 curEnd = cm.getCursor('end');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2853 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2854 var line = cm.getLine(curStart.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2855 replaceTo = curStart.ch + actionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2856 if (replaceTo > line.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2857 replaceTo=line.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2858 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2859 curEnd = new Pos(curStart.line, replaceTo);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2860 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2861 if (replaceWith=='\n') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2862 if (!vim.visualMode) cm.replaceRange('', curStart, curEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2863 // special case, where vim help says to replace by just one line-break
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2864 (CodeMirror.commands.newlineAndIndentContinueComment || CodeMirror.commands.newlineAndIndent)(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2865 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2866 var replaceWithStr = cm.getRange(curStart, curEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2867 //replace all characters in range by selected, but keep linebreaks
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2868 replaceWithStr = replaceWithStr.replace(/[^\n]/g, replaceWith);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2869 if (vim.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2870 // Tabs are split in visua block before replacing
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2871 var spaces = new Array(cm.getOption("tabSize")+1).join(' ');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2872 replaceWithStr = cm.getSelection();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2873 replaceWithStr = replaceWithStr.replace(/\t/g, spaces).replace(/[^\n]/g, replaceWith).split('\n');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2874 cm.replaceSelections(replaceWithStr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2875 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2876 cm.replaceRange(replaceWithStr, curStart, curEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2877 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2878 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2879 curStart = cursorIsBefore(selections[0].anchor, selections[0].head) ?
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2880 selections[0].anchor : selections[0].head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2881 cm.setCursor(curStart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2882 exitVisualMode(cm, false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2883 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2884 cm.setCursor(offsetCursor(curEnd, 0, -1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2885 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2886 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2887 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2888 incrementNumberToken: function(cm, actionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2889 var cur = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2890 var lineStr = cm.getLine(cur.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2891 var re = /(-?)(?:(0x)([\da-f]+)|(0b|0|)(\d+))/gi;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2892 var match;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2893 var start;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2894 var end;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2895 var numberStr;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2896 while ((match = re.exec(lineStr)) !== null) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2897 start = match.index;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2898 end = start + match[0].length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2899 if (cur.ch < end)break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2900 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2901 if (!actionArgs.backtrack && (end <= cur.ch))return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2902 if (match) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2903 var baseStr = match[2] || match[4];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2904 var digits = match[3] || match[5];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2905 var increment = actionArgs.increase ? 1 : -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2906 var base = {'0b': 2, '0': 8, '': 10, '0x': 16}[baseStr.toLowerCase()];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2907 var number = parseInt(match[1] + digits, base) + (increment * actionArgs.repeat);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2908 numberStr = number.toString(base);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2909 var zeroPadding = baseStr ? new Array(digits.length - numberStr.length + 1 + match[1].length).join('0') : '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2910 if (numberStr.charAt(0) === '-') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2911 numberStr = '-' + baseStr + zeroPadding + numberStr.substr(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2912 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2913 numberStr = baseStr + zeroPadding + numberStr;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2914 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2915 var from = new Pos(cur.line, start);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2916 var to = new Pos(cur.line, end);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2917 cm.replaceRange(numberStr, from, to);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2918 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2919 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2920 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2921 cm.setCursor(new Pos(cur.line, start + numberStr.length - 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2922 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2923 repeatLastEdit: function(cm, actionArgs, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2924 var lastEditInputState = vim.lastEditInputState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2925 if (!lastEditInputState) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2926 var repeat = actionArgs.repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2927 if (repeat && actionArgs.repeatIsExplicit) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2928 vim.lastEditInputState.repeatOverride = repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2929 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2930 repeat = vim.lastEditInputState.repeatOverride || repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2931 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2932 repeatLastEdit(cm, vim, repeat, false /** repeatForInsert */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2933 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2934 indent: function(cm, actionArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2935 cm.indentLine(cm.getCursor().line, actionArgs.indentRight);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2936 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2937 exitInsertMode: exitInsertMode
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2938 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2939
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2940 function defineAction(name, fn) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2941 actions[name] = fn;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2942 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2943
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2944 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2945 * Below are miscellaneous utility functions used by vim.js
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2946 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2947
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2948 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2949 * Clips cursor to ensure that line is within the buffer's range
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2950 * If includeLineBreak is true, then allow cur.ch == lineLength.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2951 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2952 function clipCursorToContent(cm, cur) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2953 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2954 var includeLineBreak = vim.insertMode || vim.visualMode;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2955 var line = Math.min(Math.max(cm.firstLine(), cur.line), cm.lastLine() );
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2956 var maxCh = lineLength(cm, line) - 1 + !!includeLineBreak;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2957 var ch = Math.min(Math.max(0, cur.ch), maxCh);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2958 return new Pos(line, ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2959 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2960 function copyArgs(args) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2961 var ret = {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2962 for (var prop in args) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2963 if (args.hasOwnProperty(prop)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2964 ret[prop] = args[prop];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2965 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2966 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2967 return ret;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2968 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2969 function offsetCursor(cur, offsetLine, offsetCh) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2970 if (typeof offsetLine === 'object') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2971 offsetCh = offsetLine.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2972 offsetLine = offsetLine.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2973 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2974 return new Pos(cur.line + offsetLine, cur.ch + offsetCh);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2975 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2976 function commandMatches(keys, keyMap, context, inputState) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2977 // Partial matches are not applied. They inform the key handler
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2978 // that the current key sequence is a subsequence of a valid key
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2979 // sequence, so that the key buffer is not cleared.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2980 var match, partial = [], full = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2981 for (var i = 0; i < keyMap.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2982 var command = keyMap[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2983 if (context == 'insert' && command.context != 'insert' ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2984 command.context && command.context != context ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2985 inputState.operator && command.type == 'action' ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2986 !(match = commandMatch(keys, command.keys))) { continue; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2987 if (match == 'partial') { partial.push(command); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2988 if (match == 'full') { full.push(command); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2989 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2990 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2991 partial: partial.length && partial,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2992 full: full.length && full
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2993 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2994 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2995 function commandMatch(pressed, mapped) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2996 if (mapped.slice(-11) == '<character>') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2997 // Last character matches anything.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2998 var prefixLen = mapped.length - 11;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2999 var pressedPrefix = pressed.slice(0, prefixLen);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3000 var mappedPrefix = mapped.slice(0, prefixLen);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3001 return pressedPrefix == mappedPrefix && pressed.length > prefixLen ? 'full' :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3002 mappedPrefix.indexOf(pressedPrefix) == 0 ? 'partial' : false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3003 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3004 return pressed == mapped ? 'full' :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3005 mapped.indexOf(pressed) == 0 ? 'partial' : false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3006 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3007 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3008 function lastChar(keys) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3009 var match = /^.*(<[^>]+>)$/.exec(keys);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3010 var selectedCharacter = match ? match[1] : keys.slice(-1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3011 if (selectedCharacter.length > 1){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3012 switch(selectedCharacter){
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3013 case '<CR>':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3014 selectedCharacter='\n';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3015 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3016 case '<Space>':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3017 selectedCharacter=' ';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3018 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3019 default:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3020 selectedCharacter='';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3021 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3022 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3023 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3024 return selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3025 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3026 function repeatFn(cm, fn, repeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3027 return function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3028 for (var i = 0; i < repeat; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3029 fn(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3030 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3031 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3032 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3033 function copyCursor(cur) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3034 return new Pos(cur.line, cur.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3035 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3036 function cursorEqual(cur1, cur2) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3037 return cur1.ch == cur2.ch && cur1.line == cur2.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3038 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3039 function cursorIsBefore(cur1, cur2) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3040 if (cur1.line < cur2.line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3041 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3042 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3043 if (cur1.line == cur2.line && cur1.ch < cur2.ch) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3044 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3045 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3046 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3047 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3048 function cursorMin(cur1, cur2) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3049 if (arguments.length > 2) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3050 cur2 = cursorMin.apply(undefined, Array.prototype.slice.call(arguments, 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3051 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3052 return cursorIsBefore(cur1, cur2) ? cur1 : cur2;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3053 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3054 function cursorMax(cur1, cur2) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3055 if (arguments.length > 2) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3056 cur2 = cursorMax.apply(undefined, Array.prototype.slice.call(arguments, 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3057 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3058 return cursorIsBefore(cur1, cur2) ? cur2 : cur1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3059 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3060 function cursorIsBetween(cur1, cur2, cur3) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3061 // returns true if cur2 is between cur1 and cur3.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3062 var cur1before2 = cursorIsBefore(cur1, cur2);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3063 var cur2before3 = cursorIsBefore(cur2, cur3);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3064 return cur1before2 && cur2before3;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3065 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3066 function lineLength(cm, lineNum) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3067 return cm.getLine(lineNum).length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3068 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3069 function trim(s) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3070 if (s.trim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3071 return s.trim();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3072 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3073 return s.replace(/^\s+|\s+$/g, '');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3074 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3075 function escapeRegex(s) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3076 return s.replace(/([.?*+$\[\]\/\\(){}|\-])/g, '\\$1');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3077 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3078 function extendLineToColumn(cm, lineNum, column) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3079 var endCh = lineLength(cm, lineNum);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3080 var spaces = new Array(column-endCh+1).join(' ');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3081 cm.setCursor(new Pos(lineNum, endCh));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3082 cm.replaceRange(spaces, cm.getCursor());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3083 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3084 // This functions selects a rectangular block
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3085 // of text with selectionEnd as any of its corner
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3086 // Height of block:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3087 // Difference in selectionEnd.line and first/last selection.line
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3088 // Width of the block:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3089 // Distance between selectionEnd.ch and any(first considered here) selection.ch
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3090 function selectBlock(cm, selectionEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3091 var selections = [], ranges = cm.listSelections();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3092 var head = copyCursor(cm.clipPos(selectionEnd));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3093 var isClipped = !cursorEqual(selectionEnd, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3094 var curHead = cm.getCursor('head');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3095 var primIndex = getIndex(ranges, curHead);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3096 var wasClipped = cursorEqual(ranges[primIndex].head, ranges[primIndex].anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3097 var max = ranges.length - 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3098 var index = max - primIndex > primIndex ? max : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3099 var base = ranges[index].anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3100
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3101 var firstLine = Math.min(base.line, head.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3102 var lastLine = Math.max(base.line, head.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3103 var baseCh = base.ch, headCh = head.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3104
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3105 var dir = ranges[index].head.ch - baseCh;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3106 var newDir = headCh - baseCh;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3107 if (dir > 0 && newDir <= 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3108 baseCh++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3109 if (!isClipped) { headCh--; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3110 } else if (dir < 0 && newDir >= 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3111 baseCh--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3112 if (!wasClipped) { headCh++; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3113 } else if (dir < 0 && newDir == -1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3114 baseCh--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3115 headCh++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3116 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3117 for (var line = firstLine; line <= lastLine; line++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3118 var range = {anchor: new Pos(line, baseCh), head: new Pos(line, headCh)};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3119 selections.push(range);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3120 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3121 cm.setSelections(selections);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3122 selectionEnd.ch = headCh;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3123 base.ch = baseCh;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3124 return base;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3125 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3126 function selectForInsert(cm, head, height) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3127 var sel = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3128 for (var i = 0; i < height; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3129 var lineHead = offsetCursor(head, i, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3130 sel.push({anchor: lineHead, head: lineHead});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3131 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3132 cm.setSelections(sel, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3133 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3134 // getIndex returns the index of the cursor in the selections.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3135 function getIndex(ranges, cursor, end) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3136 for (var i = 0; i < ranges.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3137 var atAnchor = end != 'head' && cursorEqual(ranges[i].anchor, cursor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3138 var atHead = end != 'anchor' && cursorEqual(ranges[i].head, cursor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3139 if (atAnchor || atHead) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3140 return i;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3141 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3142 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3143 return -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3144 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3145 function getSelectedAreaRange(cm, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3146 var lastSelection = vim.lastSelection;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3147 var getCurrentSelectedAreaRange = function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3148 var selections = cm.listSelections();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3149 var start = selections[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3150 var end = selections[selections.length-1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3151 var selectionStart = cursorIsBefore(start.anchor, start.head) ? start.anchor : start.head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3152 var selectionEnd = cursorIsBefore(end.anchor, end.head) ? end.head : end.anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3153 return [selectionStart, selectionEnd];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3154 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3155 var getLastSelectedAreaRange = function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3156 var selectionStart = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3157 var selectionEnd = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3158 var block = lastSelection.visualBlock;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3159 if (block) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3160 var width = block.width;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3161 var height = block.height;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3162 selectionEnd = new Pos(selectionStart.line + height, selectionStart.ch + width);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3163 var selections = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3164 // selectBlock creates a 'proper' rectangular block.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3165 // We do not want that in all cases, so we manually set selections.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3166 for (var i = selectionStart.line; i < selectionEnd.line; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3167 var anchor = new Pos(i, selectionStart.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3168 var head = new Pos(i, selectionEnd.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3169 var range = {anchor: anchor, head: head};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3170 selections.push(range);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3171 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3172 cm.setSelections(selections);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3173 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3174 var start = lastSelection.anchorMark.find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3175 var end = lastSelection.headMark.find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3176 var line = end.line - start.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3177 var ch = end.ch - start.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3178 selectionEnd = {line: selectionEnd.line + line, ch: line ? selectionEnd.ch : ch + selectionEnd.ch};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3179 if (lastSelection.visualLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3180 selectionStart = new Pos(selectionStart.line, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3181 selectionEnd = new Pos(selectionEnd.line, lineLength(cm, selectionEnd.line));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3182 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3183 cm.setSelection(selectionStart, selectionEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3184 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3185 return [selectionStart, selectionEnd];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3186 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3187 if (!vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3188 // In case of replaying the action.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3189 return getLastSelectedAreaRange();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3190 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3191 return getCurrentSelectedAreaRange();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3192 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3193 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3194 // Updates the previous selection with the current selection's values. This
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3195 // should only be called in visual mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3196 function updateLastSelection(cm, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3197 var anchor = vim.sel.anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3198 var head = vim.sel.head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3199 // To accommodate the effect of lastPastedText in the last selection
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3200 if (vim.lastPastedText) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3201 head = cm.posFromIndex(cm.indexFromPos(anchor) + vim.lastPastedText.length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3202 vim.lastPastedText = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3203 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3204 vim.lastSelection = {'anchorMark': cm.setBookmark(anchor),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3205 'headMark': cm.setBookmark(head),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3206 'anchor': copyCursor(anchor),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3207 'head': copyCursor(head),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3208 'visualMode': vim.visualMode,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3209 'visualLine': vim.visualLine,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3210 'visualBlock': vim.visualBlock};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3211 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3212 function expandSelection(cm, start, end) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3213 var sel = cm.state.vim.sel;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3214 var head = sel.head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3215 var anchor = sel.anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3216 var tmp;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3217 if (cursorIsBefore(end, start)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3218 tmp = end;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3219 end = start;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3220 start = tmp;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3221 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3222 if (cursorIsBefore(head, anchor)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3223 head = cursorMin(start, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3224 anchor = cursorMax(anchor, end);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3225 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3226 anchor = cursorMin(start, anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3227 head = cursorMax(head, end);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3228 head = offsetCursor(head, 0, -1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3229 if (head.ch == -1 && head.line != cm.firstLine()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3230 head = new Pos(head.line - 1, lineLength(cm, head.line - 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3231 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3232 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3233 return [anchor, head];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3234 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3235 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3236 * Updates the CodeMirror selection to match the provided vim selection.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3237 * If no arguments are given, it uses the current vim selection state.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3238 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3239 function updateCmSelection(cm, sel, mode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3240 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3241 sel = sel || vim.sel;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3242 var mode = mode ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3243 vim.visualLine ? 'line' : vim.visualBlock ? 'block' : 'char';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3244 var cmSel = makeCmSelection(cm, sel, mode);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3245 cm.setSelections(cmSel.ranges, cmSel.primary);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3246 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3247 function makeCmSelection(cm, sel, mode, exclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3248 var head = copyCursor(sel.head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3249 var anchor = copyCursor(sel.anchor);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3250 if (mode == 'char') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3251 var headOffset = !exclusive && !cursorIsBefore(sel.head, sel.anchor) ? 1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3252 var anchorOffset = cursorIsBefore(sel.head, sel.anchor) ? 1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3253 head = offsetCursor(sel.head, 0, headOffset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3254 anchor = offsetCursor(sel.anchor, 0, anchorOffset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3255 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3256 ranges: [{anchor: anchor, head: head}],
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3257 primary: 0
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3258 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3259 } else if (mode == 'line') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3260 if (!cursorIsBefore(sel.head, sel.anchor)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3261 anchor.ch = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3262
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3263 var lastLine = cm.lastLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3264 if (head.line > lastLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3265 head.line = lastLine;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3266 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3267 head.ch = lineLength(cm, head.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3268 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3269 head.ch = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3270 anchor.ch = lineLength(cm, anchor.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3271 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3272 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3273 ranges: [{anchor: anchor, head: head}],
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3274 primary: 0
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3275 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3276 } else if (mode == 'block') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3277 var top = Math.min(anchor.line, head.line),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3278 fromCh = anchor.ch,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3279 bottom = Math.max(anchor.line, head.line),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3280 toCh = head.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3281 if (fromCh < toCh) { toCh += 1; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3282 else { fromCh += 1; } var height = bottom - top + 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3283 var primary = head.line == top ? 0 : height - 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3284 var ranges = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3285 for (var i = 0; i < height; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3286 ranges.push({
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3287 anchor: new Pos(top + i, fromCh),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3288 head: new Pos(top + i, toCh)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3289 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3290 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3291 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3292 ranges: ranges,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3293 primary: primary
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3294 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3295 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3296 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3297 function getHead(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3298 var cur = cm.getCursor('head');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3299 if (cm.getSelection().length == 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3300 // Small corner case when only 1 character is selected. The "real"
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3301 // head is the left of head and anchor.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3302 cur = cursorMin(cur, cm.getCursor('anchor'));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3303 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3304 return cur;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3305 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3306
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3307 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3308 * If moveHead is set to false, the CodeMirror selection will not be
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3309 * touched. The caller assumes the responsibility of putting the cursor
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3310 * in the right place.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3311 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3312 function exitVisualMode(cm, moveHead) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3313 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3314 if (moveHead !== false) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3315 cm.setCursor(clipCursorToContent(cm, vim.sel.head));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3316 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3317 updateLastSelection(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3318 vim.visualMode = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3319 vim.visualLine = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3320 vim.visualBlock = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3321 if (!vim.insertMode) CodeMirror.signal(cm, "vim-mode-change", {mode: "normal"});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3322 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3323
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3324 // Remove any trailing newlines from the selection. For
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3325 // example, with the caret at the start of the last word on the line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3326 // 'dw' should word, but not the newline, while 'w' should advance the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3327 // caret to the first character of the next line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3328 function clipToLine(cm, curStart, curEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3329 var selection = cm.getRange(curStart, curEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3330 // Only clip if the selection ends with trailing newline + whitespace
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3331 if (/\n\s*$/.test(selection)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3332 var lines = selection.split('\n');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3333 // We know this is all whitespace.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3334 lines.pop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3335
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3336 // Cases:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3337 // 1. Last word is an empty line - do not clip the trailing '\n'
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3338 // 2. Last word is not an empty line - clip the trailing '\n'
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3339 var line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3340 // Find the line containing the last word, and clip all whitespace up
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3341 // to it.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3342 for (var line = lines.pop(); lines.length > 0 && line && isWhiteSpaceString(line); line = lines.pop()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3343 curEnd.line--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3344 curEnd.ch = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3345 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3346 // If the last word is not an empty line, clip an additional newline
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3347 if (line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3348 curEnd.line--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3349 curEnd.ch = lineLength(cm, curEnd.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3350 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3351 curEnd.ch = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3352 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3353 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3354 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3355
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3356 // Expand the selection to line ends.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3357 function expandSelectionToLine(_cm, curStart, curEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3358 curStart.ch = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3359 curEnd.ch = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3360 curEnd.line++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3361 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3362
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3363 function findFirstNonWhiteSpaceCharacter(text) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3364 if (!text) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3365 return 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3366 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3367 var firstNonWS = text.search(/\S/);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3368 return firstNonWS == -1 ? text.length : firstNonWS;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3369 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3370
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3371 function expandWordUnderCursor(cm, inclusive, _forward, bigWord, noSymbol) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3372 var cur = getHead(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3373 var line = cm.getLine(cur.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3374 var idx = cur.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3375
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3376 // Seek to first word or non-whitespace character, depending on if
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3377 // noSymbol is true.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3378 var test = noSymbol ? wordCharTest[0] : bigWordCharTest [0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3379 while (!test(line.charAt(idx))) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3380 idx++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3381 if (idx >= line.length) { return null; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3382 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3383
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3384 if (bigWord) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3385 test = bigWordCharTest[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3386 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3387 test = wordCharTest[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3388 if (!test(line.charAt(idx))) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3389 test = wordCharTest[1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3390 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3391 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3392
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3393 var end = idx, start = idx;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3394 while (test(line.charAt(end)) && end < line.length) { end++; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3395 while (test(line.charAt(start)) && start >= 0) { start--; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3396 start++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3397
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3398 if (inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3399 // If present, include all whitespace after word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3400 // Otherwise, include all whitespace before word, except indentation.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3401 var wordEnd = end;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3402 while (/\s/.test(line.charAt(end)) && end < line.length) { end++; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3403 if (wordEnd == end) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3404 var wordStart = start;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3405 while (/\s/.test(line.charAt(start - 1)) && start > 0) { start--; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3406 if (!start) { start = wordStart; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3407 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3408 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3409 return { start: new Pos(cur.line, start), end: new Pos(cur.line, end) };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3410 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3411
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3412 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3413 * Depends on the following:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3414 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3415 * - editor mode should be htmlmixedmode / xml
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3416 * - mode/xml/xml.js should be loaded
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3417 * - addon/fold/xml-fold.js should be loaded
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3418 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3419 * If any of the above requirements are not true, this function noops.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3420 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3421 * This is _NOT_ a 100% accurate implementation of vim tag text objects.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3422 * The following caveats apply (based off cursory testing, I'm sure there
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3423 * are other discrepancies):
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3424 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3425 * - Does not work inside comments:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3426 * ```
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3427 * <!-- <div>broken</div> -->
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3428 * ```
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3429 * - Does not work when tags have different cases:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3430 * ```
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3431 * <div>broken</DIV>
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3432 * ```
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3433 * - Does not work when cursor is inside a broken tag:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3434 * ```
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3435 * <div><brok><en></div>
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3436 * ```
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3437 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3438 function expandTagUnderCursor(cm, head, inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3439 var cur = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3440 if (!CodeMirror.findMatchingTag || !CodeMirror.findEnclosingTag) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3441 return { start: cur, end: cur };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3442 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3443
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3444 var tags = CodeMirror.findMatchingTag(cm, head) || CodeMirror.findEnclosingTag(cm, head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3445 if (!tags || !tags.open || !tags.close) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3446 return { start: cur, end: cur };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3447 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3448
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3449 if (inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3450 return { start: tags.open.from, end: tags.close.to };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3451 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3452 return { start: tags.open.to, end: tags.close.from };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3453 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3454
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3455 function recordJumpPosition(cm, oldCur, newCur) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3456 if (!cursorEqual(oldCur, newCur)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3457 vimGlobalState.jumpList.add(cm, oldCur, newCur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3458 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3459 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3460
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3461 function recordLastCharacterSearch(increment, args) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3462 vimGlobalState.lastCharacterSearch.increment = increment;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3463 vimGlobalState.lastCharacterSearch.forward = args.forward;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3464 vimGlobalState.lastCharacterSearch.selectedCharacter = args.selectedCharacter;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3465 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3466
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3467 var symbolToMode = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3468 '(': 'bracket', ')': 'bracket', '{': 'bracket', '}': 'bracket',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3469 '[': 'section', ']': 'section',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3470 '*': 'comment', '/': 'comment',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3471 'm': 'method', 'M': 'method',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3472 '#': 'preprocess'
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3473 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3474 var findSymbolModes = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3475 bracket: {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3476 isComplete: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3477 if (state.nextCh === state.symb) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3478 state.depth++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3479 if (state.depth >= 1)return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3480 } else if (state.nextCh === state.reverseSymb) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3481 state.depth--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3482 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3483 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3484 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3485 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3486 section: {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3487 init: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3488 state.curMoveThrough = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3489 state.symb = (state.forward ? ']' : '[') === state.symb ? '{' : '}';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3490 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3491 isComplete: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3492 return state.index === 0 && state.nextCh === state.symb;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3493 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3494 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3495 comment: {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3496 isComplete: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3497 var found = state.lastCh === '*' && state.nextCh === '/';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3498 state.lastCh = state.nextCh;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3499 return found;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3500 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3501 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3502 // TODO: The original Vim implementation only operates on level 1 and 2.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3503 // The current implementation doesn't check for code block level and
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3504 // therefore it operates on any levels.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3505 method: {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3506 init: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3507 state.symb = (state.symb === 'm' ? '{' : '}');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3508 state.reverseSymb = state.symb === '{' ? '}' : '{';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3509 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3510 isComplete: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3511 if (state.nextCh === state.symb)return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3512 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3513 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3514 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3515 preprocess: {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3516 init: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3517 state.index = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3518 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3519 isComplete: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3520 if (state.nextCh === '#') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3521 var token = state.lineText.match(/^#(\w+)/)[1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3522 if (token === 'endif') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3523 if (state.forward && state.depth === 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3524 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3525 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3526 state.depth++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3527 } else if (token === 'if') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3528 if (!state.forward && state.depth === 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3529 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3530 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3531 state.depth--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3532 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3533 if (token === 'else' && state.depth === 0)return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3534 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3535 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3536 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3537 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3538 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3539 function findSymbol(cm, repeat, forward, symb) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3540 var cur = copyCursor(cm.getCursor());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3541 var increment = forward ? 1 : -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3542 var endLine = forward ? cm.lineCount() : -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3543 var curCh = cur.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3544 var line = cur.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3545 var lineText = cm.getLine(line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3546 var state = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3547 lineText: lineText,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3548 nextCh: lineText.charAt(curCh),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3549 lastCh: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3550 index: curCh,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3551 symb: symb,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3552 reverseSymb: (forward ? { ')': '(', '}': '{' } : { '(': ')', '{': '}' })[symb],
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3553 forward: forward,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3554 depth: 0,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3555 curMoveThrough: false
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3556 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3557 var mode = symbolToMode[symb];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3558 if (!mode)return cur;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3559 var init = findSymbolModes[mode].init;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3560 var isComplete = findSymbolModes[mode].isComplete;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3561 if (init) { init(state); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3562 while (line !== endLine && repeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3563 state.index += increment;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3564 state.nextCh = state.lineText.charAt(state.index);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3565 if (!state.nextCh) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3566 line += increment;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3567 state.lineText = cm.getLine(line) || '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3568 if (increment > 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3569 state.index = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3570 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3571 var lineLen = state.lineText.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3572 state.index = (lineLen > 0) ? (lineLen-1) : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3573 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3574 state.nextCh = state.lineText.charAt(state.index);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3575 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3576 if (isComplete(state)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3577 cur.line = line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3578 cur.ch = state.index;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3579 repeat--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3580 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3581 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3582 if (state.nextCh || state.curMoveThrough) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3583 return new Pos(line, state.index);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3584 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3585 return cur;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3586 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3587
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3588 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3589 * Returns the boundaries of the next word. If the cursor in the middle of
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3590 * the word, then returns the boundaries of the current word, starting at
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3591 * the cursor. If the cursor is at the start/end of a word, and we are going
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3592 * forward/backward, respectively, find the boundaries of the next word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3593 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3594 * @param {CodeMirror} cm CodeMirror object.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3595 * @param {Cursor} cur The cursor position.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3596 * @param {boolean} forward True to search forward. False to search
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3597 * backward.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3598 * @param {boolean} bigWord True if punctuation count as part of the word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3599 * False if only [a-zA-Z0-9] characters count as part of the word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3600 * @param {boolean} emptyLineIsWord True if empty lines should be treated
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3601 * as words.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3602 * @return {Object{from:number, to:number, line: number}} The boundaries of
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3603 * the word, or null if there are no more words.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3604 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3605 function findWord(cm, cur, forward, bigWord, emptyLineIsWord) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3606 var lineNum = cur.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3607 var pos = cur.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3608 var line = cm.getLine(lineNum);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3609 var dir = forward ? 1 : -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3610 var charTests = bigWord ? bigWordCharTest: wordCharTest;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3611
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3612 if (emptyLineIsWord && line == '') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3613 lineNum += dir;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3614 line = cm.getLine(lineNum);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3615 if (!isLine(cm, lineNum)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3616 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3617 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3618 pos = (forward) ? 0 : line.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3619 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3620
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3621 while (true) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3622 if (emptyLineIsWord && line == '') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3623 return { from: 0, to: 0, line: lineNum };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3624 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3625 var stop = (dir > 0) ? line.length : -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3626 var wordStart = stop, wordEnd = stop;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3627 // Find bounds of next word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3628 while (pos != stop) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3629 var foundWord = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3630 for (var i = 0; i < charTests.length && !foundWord; ++i) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3631 if (charTests[i](line.charAt(pos))) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3632 wordStart = pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3633 // Advance to end of word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3634 while (pos != stop && charTests[i](line.charAt(pos))) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3635 pos += dir;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3636 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3637 wordEnd = pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3638 foundWord = wordStart != wordEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3639 if (wordStart == cur.ch && lineNum == cur.line &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3640 wordEnd == wordStart + dir) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3641 // We started at the end of a word. Find the next one.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3642 continue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3643 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3644 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3645 from: Math.min(wordStart, wordEnd + 1),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3646 to: Math.max(wordStart, wordEnd),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3647 line: lineNum };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3648 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3649 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3650 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3651 if (!foundWord) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3652 pos += dir;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3653 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3654 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3655 // Advance to next/prev line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3656 lineNum += dir;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3657 if (!isLine(cm, lineNum)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3658 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3659 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3660 line = cm.getLine(lineNum);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3661 pos = (dir > 0) ? 0 : line.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3662 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3663 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3664
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3665 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3666 * @param {CodeMirror} cm CodeMirror object.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3667 * @param {Pos} cur The position to start from.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3668 * @param {int} repeat Number of words to move past.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3669 * @param {boolean} forward True to search forward. False to search
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3670 * backward.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3671 * @param {boolean} wordEnd True to move to end of word. False to move to
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3672 * beginning of word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3673 * @param {boolean} bigWord True if punctuation count as part of the word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3674 * False if only alphabet characters count as part of the word.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3675 * @return {Cursor} The position the cursor should move to.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3676 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3677 function moveToWord(cm, cur, repeat, forward, wordEnd, bigWord) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3678 var curStart = copyCursor(cur);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3679 var words = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3680 if (forward && !wordEnd || !forward && wordEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3681 repeat++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3682 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3683 // For 'e', empty lines are not considered words, go figure.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3684 var emptyLineIsWord = !(forward && wordEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3685 for (var i = 0; i < repeat; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3686 var word = findWord(cm, cur, forward, bigWord, emptyLineIsWord);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3687 if (!word) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3688 var eodCh = lineLength(cm, cm.lastLine());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3689 words.push(forward
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3690 ? {line: cm.lastLine(), from: eodCh, to: eodCh}
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3691 : {line: 0, from: 0, to: 0});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3692 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3693 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3694 words.push(word);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3695 cur = new Pos(word.line, forward ? (word.to - 1) : word.from);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3696 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3697 var shortCircuit = words.length != repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3698 var firstWord = words[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3699 var lastWord = words.pop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3700 if (forward && !wordEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3701 // w
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3702 if (!shortCircuit && (firstWord.from != curStart.ch || firstWord.line != curStart.line)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3703 // We did not start in the middle of a word. Discard the extra word at the end.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3704 lastWord = words.pop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3705 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3706 return new Pos(lastWord.line, lastWord.from);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3707 } else if (forward && wordEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3708 return new Pos(lastWord.line, lastWord.to - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3709 } else if (!forward && wordEnd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3710 // ge
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3711 if (!shortCircuit && (firstWord.to != curStart.ch || firstWord.line != curStart.line)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3712 // We did not start in the middle of a word. Discard the extra word at the end.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3713 lastWord = words.pop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3714 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3715 return new Pos(lastWord.line, lastWord.to);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3716 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3717 // b
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3718 return new Pos(lastWord.line, lastWord.from);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3719 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3720 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3721
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3722 function moveToEol(cm, head, motionArgs, vim, keepHPos) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3723 var cur = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3724 var retval= new Pos(cur.line + motionArgs.repeat - 1, Infinity);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3725 var end=cm.clipPos(retval);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3726 end.ch--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3727 if (!keepHPos) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3728 vim.lastHPos = Infinity;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3729 vim.lastHSPos = cm.charCoords(end,'div').left;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3730 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3731 return retval;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3732 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3733
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3734 function moveToCharacter(cm, repeat, forward, character) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3735 var cur = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3736 var start = cur.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3737 var idx;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3738 for (var i = 0; i < repeat; i ++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3739 var line = cm.getLine(cur.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3740 idx = charIdxInLine(start, line, character, forward, true);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3741 if (idx == -1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3742 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3743 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3744 start = idx;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3745 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3746 return new Pos(cm.getCursor().line, idx);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3747 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3748
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3749 function moveToColumn(cm, repeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3750 // repeat is always >= 1, so repeat - 1 always corresponds
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3751 // to the column we want to go to.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3752 var line = cm.getCursor().line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3753 return clipCursorToContent(cm, new Pos(line, repeat - 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3754 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3755
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3756 function updateMark(cm, vim, markName, pos) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3757 if (!inArray(markName, validMarks)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3758 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3759 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3760 if (vim.marks[markName]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3761 vim.marks[markName].clear();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3762 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3763 vim.marks[markName] = cm.setBookmark(pos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3764 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3765
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3766 function charIdxInLine(start, line, character, forward, includeChar) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3767 // Search for char in line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3768 // motion_options: {forward, includeChar}
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3769 // If includeChar = true, include it too.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3770 // If forward = true, search forward, else search backwards.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3771 // If char is not found on this line, do nothing
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3772 var idx;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3773 if (forward) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3774 idx = line.indexOf(character, start + 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3775 if (idx != -1 && !includeChar) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3776 idx -= 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3777 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3778 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3779 idx = line.lastIndexOf(character, start - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3780 if (idx != -1 && !includeChar) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3781 idx += 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3782 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3783 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3784 return idx;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3785 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3786
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3787 function findParagraph(cm, head, repeat, dir, inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3788 var line = head.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3789 var min = cm.firstLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3790 var max = cm.lastLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3791 var start, end, i = line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3792 function isEmpty(i) { return !cm.getLine(i); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3793 function isBoundary(i, dir, any) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3794 if (any) { return isEmpty(i) != isEmpty(i + dir); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3795 return !isEmpty(i) && isEmpty(i + dir);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3796 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3797 if (dir) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3798 while (min <= i && i <= max && repeat > 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3799 if (isBoundary(i, dir)) { repeat--; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3800 i += dir;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3801 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3802 return new Pos(i, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3803 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3804
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3805 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3806 if (vim.visualLine && isBoundary(line, 1, true)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3807 var anchor = vim.sel.anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3808 if (isBoundary(anchor.line, -1, true)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3809 if (!inclusive || anchor.line != line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3810 line += 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3811 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3812 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3813 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3814 var startState = isEmpty(line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3815 for (i = line; i <= max && repeat; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3816 if (isBoundary(i, 1, true)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3817 if (!inclusive || isEmpty(i) != startState) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3818 repeat--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3819 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3820 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3821 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3822 end = new Pos(i, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3823 // select boundary before paragraph for the last one
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3824 if (i > max && !startState) { startState = true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3825 else { inclusive = false; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3826 for (i = line; i > min; i--) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3827 if (!inclusive || isEmpty(i) == startState || i == line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3828 if (isBoundary(i, -1, true)) { break; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3829 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3830 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3831 start = new Pos(i, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3832 return { start: start, end: end };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3833 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3834 function getSentence(cm, cur, repeat, dir, inclusive /*includes whitespace*/) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3835 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3836 Takes an index object
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3837 {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3838 line: the line string,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3839 ln: line number,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3840 pos: index in line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3841 dir: direction of traversal (-1 or 1)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3842 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3843 and modifies the pos member to represent the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3844 next valid position or sets the line to null if there are
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3845 no more valid positions.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3846 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3847 function nextChar(curr) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3848 if (curr.pos + curr.dir < 0 || curr.pos + curr.dir >= curr.line.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3849 curr.line = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3850 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3851 else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3852 curr.pos += curr.dir;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3853 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3854 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3855 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3856 Performs one iteration of traversal in forward direction
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3857 Returns an index object of the new location
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3858 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3859 function forward(cm, ln, pos, dir) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3860 var line = cm.getLine(ln);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3861
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3862 var curr = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3863 line: line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3864 ln: ln,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3865 pos: pos,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3866 dir: dir,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3867 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3868
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3869 if (curr.line === "") {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3870 return { ln: curr.ln, pos: curr.pos };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3871 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3872
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3873 var lastSentencePos = curr.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3874
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3875 // Move one step to skip character we start on
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3876 nextChar(curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3877
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3878 while (curr.line !== null) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3879 lastSentencePos = curr.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3880 if (isEndOfSentenceSymbol(curr.line[curr.pos])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3881 if (!inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3882 return { ln: curr.ln, pos: curr.pos + 1 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3883 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3884 nextChar(curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3885 while (curr.line !== null ) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3886 if (isWhiteSpaceString(curr.line[curr.pos])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3887 lastSentencePos = curr.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3888 nextChar(curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3889 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3890 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3891 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3892 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3893 return { ln: curr.ln, pos: lastSentencePos + 1, };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3894 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3895 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3896 nextChar(curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3897 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3898 return { ln: curr.ln, pos: lastSentencePos + 1 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3899 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3900
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3901 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3902 Performs one iteration of traversal in reverse direction
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3903 Returns an index object of the new location
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3904 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3905 function reverse(cm, ln, pos, dir) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3906 var line = cm.getLine(ln);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3907
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3908 var curr = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3909 line: line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3910 ln: ln,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3911 pos: pos,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3912 dir: dir,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3913 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3914
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3915 if (curr.line === "") {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3916 return { ln: curr.ln, pos: curr.pos };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3917 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3918
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3919 var lastSentencePos = curr.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3920
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3921 // Move one step to skip character we start on
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3922 nextChar(curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3923
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3924 while (curr.line !== null) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3925 if (!isWhiteSpaceString(curr.line[curr.pos]) && !isEndOfSentenceSymbol(curr.line[curr.pos])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3926 lastSentencePos = curr.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3927 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3928
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3929 else if (isEndOfSentenceSymbol(curr.line[curr.pos]) ) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3930 if (!inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3931 return { ln: curr.ln, pos: lastSentencePos };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3932 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3933 if (isWhiteSpaceString(curr.line[curr.pos + 1])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3934 return { ln: curr.ln, pos: curr.pos + 1, };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3935 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3936 return {ln: curr.ln, pos: lastSentencePos};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3937 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3938 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3939 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3940
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3941 nextChar(curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3942 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3943 curr.line = line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3944 if (inclusive && isWhiteSpaceString(curr.line[curr.pos])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3945 return { ln: curr.ln, pos: curr.pos };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3946 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3947 return { ln: curr.ln, pos: lastSentencePos };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3948 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3949
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3950 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3951
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3952 var curr_index = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3953 ln: cur.line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3954 pos: cur.ch,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3955 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3956
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3957 while (repeat > 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3958 if (dir < 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3959 curr_index = reverse(cm, curr_index.ln, curr_index.pos, dir);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3960 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3961 else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3962 curr_index = forward(cm, curr_index.ln, curr_index.pos, dir);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3963 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3964 repeat--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3965 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3966
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3967 return new Pos(curr_index.ln, curr_index.pos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3968 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3969
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3970 function findSentence(cm, cur, repeat, dir) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3971
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3972 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3973 Takes an index object
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3974 {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3975 line: the line string,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3976 ln: line number,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3977 pos: index in line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3978 dir: direction of traversal (-1 or 1)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3979 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3980 and modifies the line, ln, and pos members to represent the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3981 next valid position or sets them to null if there are
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3982 no more valid positions.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3983 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3984 function nextChar(cm, idx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3985 if (idx.pos + idx.dir < 0 || idx.pos + idx.dir >= idx.line.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3986 idx.ln += idx.dir;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3987 if (!isLine(cm, idx.ln)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3988 idx.line = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3989 idx.ln = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3990 idx.pos = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3991 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3992 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3993 idx.line = cm.getLine(idx.ln);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3994 idx.pos = (idx.dir > 0) ? 0 : idx.line.length - 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3995 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3996 else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3997 idx.pos += idx.dir;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3998 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3999 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4000
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4001 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4002 Performs one iteration of traversal in forward direction
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4003 Returns an index object of the new location
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4004 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4005 function forward(cm, ln, pos, dir) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4006 var line = cm.getLine(ln);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4007 var stop = (line === "");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4008
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4009 var curr = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4010 line: line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4011 ln: ln,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4012 pos: pos,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4013 dir: dir,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4014 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4015
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4016 var last_valid = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4017 ln: curr.ln,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4018 pos: curr.pos,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4019 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4020
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4021 var skip_empty_lines = (curr.line === "");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4022
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4023 // Move one step to skip character we start on
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4024 nextChar(cm, curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4025
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4026 while (curr.line !== null) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4027 last_valid.ln = curr.ln;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4028 last_valid.pos = curr.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4029
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4030 if (curr.line === "" && !skip_empty_lines) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4031 return { ln: curr.ln, pos: curr.pos, };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4032 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4033 else if (stop && curr.line !== "" && !isWhiteSpaceString(curr.line[curr.pos])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4034 return { ln: curr.ln, pos: curr.pos, };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4035 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4036 else if (isEndOfSentenceSymbol(curr.line[curr.pos])
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4037 && !stop
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4038 && (curr.pos === curr.line.length - 1
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4039 || isWhiteSpaceString(curr.line[curr.pos + 1]))) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4040 stop = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4041 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4042
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4043 nextChar(cm, curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4044 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4045
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4046 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4047 Set the position to the last non whitespace character on the last
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4048 valid line in the case that we reach the end of the document.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4049 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4050 var line = cm.getLine(last_valid.ln);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4051 last_valid.pos = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4052 for(var i = line.length - 1; i >= 0; --i) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4053 if (!isWhiteSpaceString(line[i])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4054 last_valid.pos = i;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4055 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4056 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4057 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4058
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4059 return last_valid;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4060
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4061 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4062
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4063 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4064 Performs one iteration of traversal in reverse direction
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4065 Returns an index object of the new location
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4066 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4067 function reverse(cm, ln, pos, dir) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4068 var line = cm.getLine(ln);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4069
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4070 var curr = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4071 line: line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4072 ln: ln,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4073 pos: pos,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4074 dir: dir,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4075 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4076
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4077 var last_valid = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4078 ln: curr.ln,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4079 pos: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4080 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4081
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4082 var skip_empty_lines = (curr.line === "");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4083
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4084 // Move one step to skip character we start on
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4085 nextChar(cm, curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4086
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4087 while (curr.line !== null) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4088
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4089 if (curr.line === "" && !skip_empty_lines) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4090 if (last_valid.pos !== null) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4091 return last_valid;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4092 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4093 else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4094 return { ln: curr.ln, pos: curr.pos };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4095 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4096 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4097 else if (isEndOfSentenceSymbol(curr.line[curr.pos])
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4098 && last_valid.pos !== null
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4099 && !(curr.ln === last_valid.ln && curr.pos + 1 === last_valid.pos)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4100 return last_valid;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4101 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4102 else if (curr.line !== "" && !isWhiteSpaceString(curr.line[curr.pos])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4103 skip_empty_lines = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4104 last_valid = { ln: curr.ln, pos: curr.pos };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4105 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4106
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4107 nextChar(cm, curr);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4108 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4109
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4110 /*
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4111 Set the position to the first non whitespace character on the last
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4112 valid line in the case that we reach the beginning of the document.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4113 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4114 var line = cm.getLine(last_valid.ln);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4115 last_valid.pos = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4116 for(var i = 0; i < line.length; ++i) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4117 if (!isWhiteSpaceString(line[i])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4118 last_valid.pos = i;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4119 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4120 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4121 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4122 return last_valid;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4123 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4124
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4125 var curr_index = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4126 ln: cur.line,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4127 pos: cur.ch,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4128 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4129
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4130 while (repeat > 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4131 if (dir < 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4132 curr_index = reverse(cm, curr_index.ln, curr_index.pos, dir);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4133 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4134 else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4135 curr_index = forward(cm, curr_index.ln, curr_index.pos, dir);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4136 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4137 repeat--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4138 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4139
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4140 return new Pos(curr_index.ln, curr_index.pos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4141 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4142
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4143 // TODO: perhaps this finagling of start and end positions belongs
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4144 // in codemirror/replaceRange?
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4145 function selectCompanionObject(cm, head, symb, inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4146 var cur = head, start, end;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4147
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4148 var bracketRegexp = ({
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4149 '(': /[()]/, ')': /[()]/,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4150 '[': /[[\]]/, ']': /[[\]]/,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4151 '{': /[{}]/, '}': /[{}]/,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4152 '<': /[<>]/, '>': /[<>]/})[symb];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4153 var openSym = ({
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4154 '(': '(', ')': '(',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4155 '[': '[', ']': '[',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4156 '{': '{', '}': '{',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4157 '<': '<', '>': '<'})[symb];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4158 var curChar = cm.getLine(cur.line).charAt(cur.ch);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4159 // Due to the behavior of scanForBracket, we need to add an offset if the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4160 // cursor is on a matching open bracket.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4161 var offset = curChar === openSym ? 1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4162
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4163 start = cm.scanForBracket(new Pos(cur.line, cur.ch + offset), -1, undefined, {'bracketRegex': bracketRegexp});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4164 end = cm.scanForBracket(new Pos(cur.line, cur.ch + offset), 1, undefined, {'bracketRegex': bracketRegexp});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4165
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4166 if (!start || !end) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4167 return { start: cur, end: cur };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4168 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4169
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4170 start = start.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4171 end = end.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4172
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4173 if ((start.line == end.line && start.ch > end.ch)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4174 || (start.line > end.line)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4175 var tmp = start;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4176 start = end;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4177 end = tmp;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4178 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4179
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4180 if (inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4181 end.ch += 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4182 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4183 start.ch += 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4184 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4185
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4186 return { start: start, end: end };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4187 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4188
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4189 // Takes in a symbol and a cursor and tries to simulate text objects that
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4190 // have identical opening and closing symbols
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4191 // TODO support across multiple lines
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4192 function findBeginningAndEnd(cm, head, symb, inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4193 var cur = copyCursor(head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4194 var line = cm.getLine(cur.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4195 var chars = line.split('');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4196 var start, end, i, len;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4197 var firstIndex = chars.indexOf(symb);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4198
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4199 // the decision tree is to always look backwards for the beginning first,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4200 // but if the cursor is in front of the first instance of the symb,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4201 // then move the cursor forward
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4202 if (cur.ch < firstIndex) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4203 cur.ch = firstIndex;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4204 // Why is this line even here???
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4205 // cm.setCursor(cur.line, firstIndex+1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4206 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4207 // otherwise if the cursor is currently on the closing symbol
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4208 else if (firstIndex < cur.ch && chars[cur.ch] == symb) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4209 end = cur.ch; // assign end to the current cursor
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4210 --cur.ch; // make sure to look backwards
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4211 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4212
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4213 // if we're currently on the symbol, we've got a start
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4214 if (chars[cur.ch] == symb && !end) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4215 start = cur.ch + 1; // assign start to ahead of the cursor
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4216 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4217 // go backwards to find the start
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4218 for (i = cur.ch; i > -1 && !start; i--) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4219 if (chars[i] == symb) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4220 start = i + 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4221 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4222 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4223 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4224
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4225 // look forwards for the end symbol
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4226 if (start && !end) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4227 for (i = start, len = chars.length; i < len && !end; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4228 if (chars[i] == symb) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4229 end = i;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4230 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4231 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4232 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4233
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4234 // nothing found
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4235 if (!start || !end) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4236 return { start: cur, end: cur };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4237 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4238
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4239 // include the symbols
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4240 if (inclusive) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4241 --start; ++end;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4242 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4243
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4244 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4245 start: new Pos(cur.line, start),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4246 end: new Pos(cur.line, end)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4247 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4248 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4249
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4250 // Search functions
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4251 defineOption('pcre', true, 'boolean');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4252 function SearchState() {}
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4253 SearchState.prototype = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4254 getQuery: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4255 return vimGlobalState.query;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4256 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4257 setQuery: function(query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4258 vimGlobalState.query = query;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4259 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4260 getOverlay: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4261 return this.searchOverlay;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4262 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4263 setOverlay: function(overlay) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4264 this.searchOverlay = overlay;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4265 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4266 isReversed: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4267 return vimGlobalState.isReversed;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4268 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4269 setReversed: function(reversed) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4270 vimGlobalState.isReversed = reversed;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4271 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4272 getScrollbarAnnotate: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4273 return this.annotate;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4274 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4275 setScrollbarAnnotate: function(annotate) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4276 this.annotate = annotate;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4277 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4278 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4279 function getSearchState(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4280 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4281 return vim.searchState_ || (vim.searchState_ = new SearchState());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4282 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4283 function splitBySlash(argString) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4284 return splitBySeparator(argString, '/');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4285 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4286
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4287 function findUnescapedSlashes(argString) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4288 return findUnescapedSeparators(argString, '/');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4289 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4290
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4291 function splitBySeparator(argString, separator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4292 var slashes = findUnescapedSeparators(argString, separator) || [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4293 if (!slashes.length) return [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4294 var tokens = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4295 // in case of strings like foo/bar
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4296 if (slashes[0] !== 0) return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4297 for (var i = 0; i < slashes.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4298 if (typeof slashes[i] == 'number')
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4299 tokens.push(argString.substring(slashes[i] + 1, slashes[i+1]));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4300 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4301 return tokens;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4302 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4303
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4304 function findUnescapedSeparators(str, separator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4305 if (!separator)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4306 separator = '/';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4307
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4308 var escapeNextChar = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4309 var slashes = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4310 for (var i = 0; i < str.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4311 var c = str.charAt(i);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4312 if (!escapeNextChar && c == separator) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4313 slashes.push(i);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4314 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4315 escapeNextChar = !escapeNextChar && (c == '\\');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4316 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4317 return slashes;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4318 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4319
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4320 // Translates a search string from ex (vim) syntax into javascript form.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4321 function translateRegex(str) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4322 // When these match, add a '\' if unescaped or remove one if escaped.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4323 var specials = '|(){';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4324 // Remove, but never add, a '\' for these.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4325 var unescape = '}';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4326 var escapeNextChar = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4327 var out = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4328 for (var i = -1; i < str.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4329 var c = str.charAt(i) || '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4330 var n = str.charAt(i+1) || '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4331 var specialComesNext = (n && specials.indexOf(n) != -1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4332 if (escapeNextChar) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4333 if (c !== '\\' || !specialComesNext) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4334 out.push(c);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4335 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4336 escapeNextChar = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4337 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4338 if (c === '\\') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4339 escapeNextChar = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4340 // Treat the unescape list as special for removing, but not adding '\'.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4341 if (n && unescape.indexOf(n) != -1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4342 specialComesNext = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4343 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4344 // Not passing this test means removing a '\'.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4345 if (!specialComesNext || n === '\\') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4346 out.push(c);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4347 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4348 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4349 out.push(c);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4350 if (specialComesNext && n !== '\\') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4351 out.push('\\');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4352 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4353 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4354 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4355 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4356 return out.join('');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4357 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4358
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4359 // Translates the replace part of a search and replace from ex (vim) syntax into
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4360 // javascript form. Similar to translateRegex, but additionally fixes back references
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4361 // (translates '\[0..9]' to '$[0..9]') and follows different rules for escaping '$'.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4362 var charUnescapes = {'\\n': '\n', '\\r': '\r', '\\t': '\t'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4363 function translateRegexReplace(str) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4364 var escapeNextChar = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4365 var out = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4366 for (var i = -1; i < str.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4367 var c = str.charAt(i) || '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4368 var n = str.charAt(i+1) || '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4369 if (charUnescapes[c + n]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4370 out.push(charUnescapes[c+n]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4371 i++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4372 } else if (escapeNextChar) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4373 // At any point in the loop, escapeNextChar is true if the previous
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4374 // character was a '\' and was not escaped.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4375 out.push(c);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4376 escapeNextChar = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4377 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4378 if (c === '\\') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4379 escapeNextChar = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4380 if ((isNumber(n) || n === '$')) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4381 out.push('$');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4382 } else if (n !== '/' && n !== '\\') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4383 out.push('\\');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4384 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4385 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4386 if (c === '$') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4387 out.push('$');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4388 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4389 out.push(c);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4390 if (n === '/') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4391 out.push('\\');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4392 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4393 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4394 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4395 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4396 return out.join('');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4397 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4398
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4399 // Unescape \ and / in the replace part, for PCRE mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4400 var unescapes = {'\\/': '/', '\\\\': '\\', '\\n': '\n', '\\r': '\r', '\\t': '\t', '\\&':'&'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4401 function unescapeRegexReplace(str) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4402 var stream = new CodeMirror.StringStream(str);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4403 var output = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4404 while (!stream.eol()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4405 // Search for \.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4406 while (stream.peek() && stream.peek() != '\\') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4407 output.push(stream.next());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4408 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4409 var matched = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4410 for (var matcher in unescapes) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4411 if (stream.match(matcher, true)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4412 matched = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4413 output.push(unescapes[matcher]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4414 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4415 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4416 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4417 if (!matched) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4418 // Don't change anything
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4419 output.push(stream.next());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4420 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4421 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4422 return output.join('');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4423 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4424
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4425 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4426 * Extract the regular expression from the query and return a Regexp object.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4427 * Returns null if the query is blank.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4428 * If ignoreCase is passed in, the Regexp object will have the 'i' flag set.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4429 * If smartCase is passed in, and the query contains upper case letters,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4430 * then ignoreCase is overridden, and the 'i' flag will not be set.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4431 * If the query contains the /i in the flag part of the regular expression,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4432 * then both ignoreCase and smartCase are ignored, and 'i' will be passed
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4433 * through to the Regex object.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4434 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4435 function parseQuery(query, ignoreCase, smartCase) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4436 // First update the last search register
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4437 var lastSearchRegister = vimGlobalState.registerController.getRegister('/');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4438 lastSearchRegister.setText(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4439 // Check if the query is already a regex.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4440 if (query instanceof RegExp) { return query; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4441 // First try to extract regex + flags from the input. If no flags found,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4442 // extract just the regex. IE does not accept flags directly defined in
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4443 // the regex string in the form /regex/flags
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4444 var slashes = findUnescapedSlashes(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4445 var regexPart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4446 var forceIgnoreCase;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4447 if (!slashes.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4448 // Query looks like 'regexp'
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4449 regexPart = query;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4450 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4451 // Query looks like 'regexp/...'
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4452 regexPart = query.substring(0, slashes[0]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4453 var flagsPart = query.substring(slashes[0]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4454 forceIgnoreCase = (flagsPart.indexOf('i') != -1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4455 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4456 if (!regexPart) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4457 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4458 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4459 if (!getOption('pcre')) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4460 regexPart = translateRegex(regexPart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4461 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4462 if (smartCase) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4463 ignoreCase = (/^[^A-Z]*$/).test(regexPart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4464 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4465 var regexp = new RegExp(regexPart,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4466 (ignoreCase || forceIgnoreCase) ? 'im' : 'm');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4467 return regexp;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4468 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4469
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4470 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4471 * dom - Document Object Manipulator
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4472 * Usage:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4473 * dom('<tag>'|<node>[, ...{<attributes>|<$styles>}|<child-node>|'<text>'])
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4474 * Examples:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4475 * dom('div', {id:'xyz'}, dom('p', 'CM rocks!', {$color:'red'}))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4476 * dom(document.head, dom('script', 'alert("hello!")'))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4477 * Not supported:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4478 * dom('p', ['arrays are objects'], Error('objects specify attributes'))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4479 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4480 function dom(n) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4481 if (typeof n === 'string') n = document.createElement(n);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4482 for (var a, i = 1; i < arguments.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4483 if (!(a = arguments[i])) continue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4484 if (typeof a !== 'object') a = document.createTextNode(a);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4485 if (a.nodeType) n.appendChild(a);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4486 else for (var key in a) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4487 if (!Object.prototype.hasOwnProperty.call(a, key)) continue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4488 if (key[0] === '$') n.style[key.slice(1)] = a[key];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4489 else n.setAttribute(key, a[key]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4490 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4491 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4492 return n;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4493 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4494
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4495 function showConfirm(cm, template) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4496 var pre = dom('div', {$color: 'red', $whiteSpace: 'pre', class: 'cm-vim-message'}, template);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4497 if (cm.openNotification) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4498 cm.openNotification(pre, {bottom: true, duration: 5000});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4499 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4500 alert(pre.innerText);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4501 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4502 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4503
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4504 function makePrompt(prefix, desc) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4505 return dom(document.createDocumentFragment(),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4506 dom('span', {$fontFamily: 'monospace', $whiteSpace: 'pre'},
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4507 prefix,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4508 dom('input', {type: 'text', autocorrect: 'off',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4509 autocapitalize: 'off', spellcheck: 'false'})),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4510 desc && dom('span', {$color: '#888'}, desc));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4511 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4512
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4513 function showPrompt(cm, options) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4514 var template = makePrompt(options.prefix, options.desc);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4515 if (cm.openDialog) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4516 cm.openDialog(template, options.onClose, {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4517 onKeyDown: options.onKeyDown, onKeyUp: options.onKeyUp,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4518 bottom: true, selectValueOnOpen: false, value: options.value
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4519 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4520 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4521 else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4522 var shortText = '';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4523 if (typeof options.prefix != "string" && options.prefix) shortText += options.prefix.textContent;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4524 if (options.desc) shortText += " " + options.desc;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4525 options.onClose(prompt(shortText, ''));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4526 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4527 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4528
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4529 function regexEqual(r1, r2) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4530 if (r1 instanceof RegExp && r2 instanceof RegExp) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4531 var props = ['global', 'multiline', 'ignoreCase', 'source'];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4532 for (var i = 0; i < props.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4533 var prop = props[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4534 if (r1[prop] !== r2[prop]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4535 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4536 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4537 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4538 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4539 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4540 return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4541 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4542 // Returns true if the query is valid.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4543 function updateSearchQuery(cm, rawQuery, ignoreCase, smartCase) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4544 if (!rawQuery) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4545 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4546 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4547 var state = getSearchState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4548 var query = parseQuery(rawQuery, !!ignoreCase, !!smartCase);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4549 if (!query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4550 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4551 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4552 highlightSearchMatches(cm, query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4553 if (regexEqual(query, state.getQuery())) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4554 return query;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4555 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4556 state.setQuery(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4557 return query;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4558 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4559 function searchOverlay(query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4560 if (query.source.charAt(0) == '^') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4561 var matchSol = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4562 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4563 return {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4564 token: function(stream) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4565 if (matchSol && !stream.sol()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4566 stream.skipToEnd();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4567 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4568 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4569 var match = stream.match(query, false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4570 if (match) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4571 if (match[0].length == 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4572 // Matched empty string, skip to next.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4573 stream.next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4574 return 'searching';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4575 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4576 if (!stream.sol()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4577 // Backtrack 1 to match \b
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4578 stream.backUp(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4579 if (!query.exec(stream.next() + match[0])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4580 stream.next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4581 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4582 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4583 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4584 stream.match(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4585 return 'searching';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4586 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4587 while (!stream.eol()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4588 stream.next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4589 if (stream.match(query, false)) break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4590 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4591 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4592 query: query
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4593 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4594 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4595 var highlightTimeout = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4596 function highlightSearchMatches(cm, query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4597 clearTimeout(highlightTimeout);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4598 highlightTimeout = setTimeout(function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4599 if (!cm.state.vim) return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4600 var searchState = getSearchState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4601 var overlay = searchState.getOverlay();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4602 if (!overlay || query != overlay.query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4603 if (overlay) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4604 cm.removeOverlay(overlay);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4605 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4606 overlay = searchOverlay(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4607 cm.addOverlay(overlay);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4608 if (cm.showMatchesOnScrollbar) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4609 if (searchState.getScrollbarAnnotate()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4610 searchState.getScrollbarAnnotate().clear();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4611 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4612 searchState.setScrollbarAnnotate(cm.showMatchesOnScrollbar(query));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4613 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4614 searchState.setOverlay(overlay);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4615 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4616 }, 50);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4617 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4618 function findNext(cm, prev, query, repeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4619 if (repeat === undefined) { repeat = 1; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4620 return cm.operation(function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4621 var pos = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4622 var cursor = cm.getSearchCursor(query, pos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4623 for (var i = 0; i < repeat; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4624 var found = cursor.find(prev);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4625 if (i == 0 && found && cursorEqual(cursor.from(), pos)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4626 var lastEndPos = prev ? cursor.from() : cursor.to();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4627 found = cursor.find(prev);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4628 if (found && !found[0] && cursorEqual(cursor.from(), lastEndPos)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4629 if (cm.getLine(lastEndPos.line).length == lastEndPos.ch)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4630 found = cursor.find(prev);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4631 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4632 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4633 if (!found) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4634 // SearchCursor may have returned null because it hit EOF, wrap
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4635 // around and try again.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4636 cursor = cm.getSearchCursor(query,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4637 (prev) ? new Pos(cm.lastLine()) : new Pos(cm.firstLine(), 0) );
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4638 if (!cursor.find(prev)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4639 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4640 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4641 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4642 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4643 return cursor.from();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4644 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4645 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4646 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4647 * Pretty much the same as `findNext`, except for the following differences:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4648 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4649 * 1. Before starting the search, move to the previous search. This way if our cursor is
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4650 * already inside a match, we should return the current match.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4651 * 2. Rather than only returning the cursor's from, we return the cursor's from and to as a tuple.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4652 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4653 function findNextFromAndToInclusive(cm, prev, query, repeat, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4654 if (repeat === undefined) { repeat = 1; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4655 return cm.operation(function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4656 var pos = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4657 var cursor = cm.getSearchCursor(query, pos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4658
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4659 // Go back one result to ensure that if the cursor is currently a match, we keep it.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4660 var found = cursor.find(!prev);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4661
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4662 // If we haven't moved, go back one more (similar to if i==0 logic in findNext).
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4663 if (!vim.visualMode && found && cursorEqual(cursor.from(), pos)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4664 cursor.find(!prev);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4665 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4666
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4667 for (var i = 0; i < repeat; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4668 found = cursor.find(prev);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4669 if (!found) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4670 // SearchCursor may have returned null because it hit EOF, wrap
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4671 // around and try again.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4672 cursor = cm.getSearchCursor(query,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4673 (prev) ? new Pos(cm.lastLine()) : new Pos(cm.firstLine(), 0) );
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4674 if (!cursor.find(prev)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4675 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4676 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4677 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4678 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4679 return [cursor.from(), cursor.to()];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4680 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4681 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4682 function clearSearchHighlight(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4683 var state = getSearchState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4684 cm.removeOverlay(getSearchState(cm).getOverlay());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4685 state.setOverlay(null);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4686 if (state.getScrollbarAnnotate()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4687 state.getScrollbarAnnotate().clear();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4688 state.setScrollbarAnnotate(null);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4689 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4690 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4691 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4692 * Check if pos is in the specified range, INCLUSIVE.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4693 * Range can be specified with 1 or 2 arguments.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4694 * If the first range argument is an array, treat it as an array of line
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4695 * numbers. Match pos against any of the lines.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4696 * If the first range argument is a number,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4697 * if there is only 1 range argument, check if pos has the same line
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4698 * number
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4699 * if there are 2 range arguments, then check if pos is in between the two
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4700 * range arguments.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4701 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4702 function isInRange(pos, start, end) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4703 if (typeof pos != 'number') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4704 // Assume it is a cursor position. Get the line number.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4705 pos = pos.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4706 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4707 if (start instanceof Array) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4708 return inArray(pos, start);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4709 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4710 if (typeof end == 'number') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4711 return (pos >= start && pos <= end);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4712 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4713 return pos == start;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4714 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4715 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4716 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4717 function getUserVisibleLines(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4718 var scrollInfo = cm.getScrollInfo();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4719 var occludeToleranceTop = 6;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4720 var occludeToleranceBottom = 10;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4721 var from = cm.coordsChar({left:0, top: occludeToleranceTop + scrollInfo.top}, 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4722 var bottomY = scrollInfo.clientHeight - occludeToleranceBottom + scrollInfo.top;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4723 var to = cm.coordsChar({left:0, top: bottomY}, 'local');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4724 return {top: from.line, bottom: to.line};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4725 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4726
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4727 function getMarkPos(cm, vim, markName) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4728 if (markName == '\'' || markName == '`') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4729 return vimGlobalState.jumpList.find(cm, -1) || new Pos(0, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4730 } else if (markName == '.') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4731 return getLastEditPos(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4732 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4733
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4734 var mark = vim.marks[markName];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4735 return mark && mark.find();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4736 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4737
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4738 function getLastEditPos(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4739 var done = cm.doc.history.done;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4740 for (var i = done.length; i--;) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4741 if (done[i].changes) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4742 return copyCursor(done[i].changes[0].to);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4743 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4744 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4745 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4746
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4747 var ExCommandDispatcher = function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4748 this.buildCommandMap_();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4749 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4750 ExCommandDispatcher.prototype = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4751 processCommand: function(cm, input, opt_params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4752 var that = this;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4753 cm.operation(function () {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4754 cm.curOp.isVimOp = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4755 that._processCommand(cm, input, opt_params);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4756 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4757 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4758 _processCommand: function(cm, input, opt_params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4759 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4760 var commandHistoryRegister = vimGlobalState.registerController.getRegister(':');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4761 var previousCommand = commandHistoryRegister.toString();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4762 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4763 exitVisualMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4764 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4765 var inputStream = new CodeMirror.StringStream(input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4766 // update ": with the latest command whether valid or invalid
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4767 commandHistoryRegister.setText(input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4768 var params = opt_params || {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4769 params.input = input;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4770 try {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4771 this.parseInput_(cm, inputStream, params);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4772 } catch(e) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4773 showConfirm(cm, e.toString());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4774 throw e;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4775 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4776 var command;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4777 var commandName;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4778 if (!params.commandName) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4779 // If only a line range is defined, move to the line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4780 if (params.line !== undefined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4781 commandName = 'move';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4782 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4783 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4784 command = this.matchCommand_(params.commandName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4785 if (command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4786 commandName = command.name;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4787 if (command.excludeFromCommandHistory) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4788 commandHistoryRegister.setText(previousCommand);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4789 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4790 this.parseCommandArgs_(inputStream, params, command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4791 if (command.type == 'exToKey') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4792 // Handle Ex to Key mapping.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4793 for (var i = 0; i < command.toKeys.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4794 vimApi.handleKey(cm, command.toKeys[i], 'mapping');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4795 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4796 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4797 } else if (command.type == 'exToEx') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4798 // Handle Ex to Ex mapping.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4799 this.processCommand(cm, command.toInput);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4800 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4801 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4802 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4803 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4804 if (!commandName) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4805 showConfirm(cm, 'Not an editor command ":' + input + '"');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4806 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4807 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4808 try {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4809 exCommands[commandName](cm, params);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4810 // Possibly asynchronous commands (e.g. substitute, which might have a
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4811 // user confirmation), are responsible for calling the callback when
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4812 // done. All others have it taken care of for them here.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4813 if ((!command || !command.possiblyAsync) && params.callback) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4814 params.callback();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4815 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4816 } catch(e) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4817 showConfirm(cm, e.toString());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4818 throw e;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4819 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4820 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4821 parseInput_: function(cm, inputStream, result) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4822 inputStream.eatWhile(':');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4823 // Parse range.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4824 if (inputStream.eat('%')) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4825 result.line = cm.firstLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4826 result.lineEnd = cm.lastLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4827 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4828 result.line = this.parseLineSpec_(cm, inputStream);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4829 if (result.line !== undefined && inputStream.eat(',')) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4830 result.lineEnd = this.parseLineSpec_(cm, inputStream);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4831 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4832 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4833
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4834 // Parse command name.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4835 var commandMatch = inputStream.match(/^(\w+|!!|@@|[!#&*<=>@~])/);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4836 if (commandMatch) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4837 result.commandName = commandMatch[1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4838 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4839 result.commandName = inputStream.match(/.*/)[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4840 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4841
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4842 return result;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4843 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4844 parseLineSpec_: function(cm, inputStream) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4845 var numberMatch = inputStream.match(/^(\d+)/);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4846 if (numberMatch) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4847 // Absolute line number plus offset (N+M or N-M) is probably a typo,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4848 // not something the user actually wanted. (NB: vim does allow this.)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4849 return parseInt(numberMatch[1], 10) - 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4850 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4851 switch (inputStream.next()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4852 case '.':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4853 return this.parseLineSpecOffset_(inputStream, cm.getCursor().line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4854 case '$':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4855 return this.parseLineSpecOffset_(inputStream, cm.lastLine());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4856 case '\'':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4857 var markName = inputStream.next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4858 var markPos = getMarkPos(cm, cm.state.vim, markName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4859 if (!markPos) throw new Error('Mark not set');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4860 return this.parseLineSpecOffset_(inputStream, markPos.line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4861 case '-':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4862 case '+':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4863 inputStream.backUp(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4864 // Offset is relative to current line if not otherwise specified.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4865 return this.parseLineSpecOffset_(inputStream, cm.getCursor().line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4866 default:
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4867 inputStream.backUp(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4868 return undefined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4869 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4870 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4871 parseLineSpecOffset_: function(inputStream, line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4872 var offsetMatch = inputStream.match(/^([+-])?(\d+)/);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4873 if (offsetMatch) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4874 var offset = parseInt(offsetMatch[2], 10);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4875 if (offsetMatch[1] == "-") {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4876 line -= offset;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4877 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4878 line += offset;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4879 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4880 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4881 return line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4882 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4883 parseCommandArgs_: function(inputStream, params, command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4884 if (inputStream.eol()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4885 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4886 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4887 params.argString = inputStream.match(/.*/)[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4888 // Parse command-line arguments
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4889 var delim = command.argDelimiter || /\s+/;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4890 var args = trim(params.argString).split(delim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4891 if (args.length && args[0]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4892 params.args = args;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4893 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4894 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4895 matchCommand_: function(commandName) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4896 // Return the command in the command map that matches the shortest
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4897 // prefix of the passed in command name. The match is guaranteed to be
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4898 // unambiguous if the defaultExCommandMap's shortNames are set up
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4899 // correctly. (see @code{defaultExCommandMap}).
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4900 for (var i = commandName.length; i > 0; i--) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4901 var prefix = commandName.substring(0, i);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4902 if (this.commandMap_[prefix]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4903 var command = this.commandMap_[prefix];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4904 if (command.name.indexOf(commandName) === 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4905 return command;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4906 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4907 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4908 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4909 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4910 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4911 buildCommandMap_: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4912 this.commandMap_ = {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4913 for (var i = 0; i < defaultExCommandMap.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4914 var command = defaultExCommandMap[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4915 var key = command.shortName || command.name;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4916 this.commandMap_[key] = command;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4917 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4918 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4919 map: function(lhs, rhs, ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4920 if (lhs != ':' && lhs.charAt(0) == ':') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4921 if (ctx) { throw Error('Mode not supported for ex mappings'); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4922 var commandName = lhs.substring(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4923 if (rhs != ':' && rhs.charAt(0) == ':') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4924 // Ex to Ex mapping
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4925 this.commandMap_[commandName] = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4926 name: commandName,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4927 type: 'exToEx',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4928 toInput: rhs.substring(1),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4929 user: true
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4930 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4931 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4932 // Ex to key mapping
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4933 this.commandMap_[commandName] = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4934 name: commandName,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4935 type: 'exToKey',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4936 toKeys: rhs,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4937 user: true
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4938 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4939 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4940 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4941 if (rhs != ':' && rhs.charAt(0) == ':') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4942 // Key to Ex mapping.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4943 var mapping = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4944 keys: lhs,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4945 type: 'keyToEx',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4946 exArgs: { input: rhs.substring(1) }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4947 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4948 if (ctx) { mapping.context = ctx; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4949 defaultKeymap.unshift(mapping);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4950 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4951 // Key to key mapping
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4952 var mapping = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4953 keys: lhs,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4954 type: 'keyToKey',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4955 toKeys: rhs
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4956 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4957 if (ctx) { mapping.context = ctx; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4958 defaultKeymap.unshift(mapping);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4959 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4960 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4961 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4962 unmap: function(lhs, ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4963 if (lhs != ':' && lhs.charAt(0) == ':') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4964 // Ex to Ex or Ex to key mapping
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4965 if (ctx) { throw Error('Mode not supported for ex mappings'); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4966 var commandName = lhs.substring(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4967 if (this.commandMap_[commandName] && this.commandMap_[commandName].user) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4968 delete this.commandMap_[commandName];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4969 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4970 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4971 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4972 // Key to Ex or key to key mapping
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4973 var keys = lhs;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4974 for (var i = 0; i < defaultKeymap.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4975 if (keys == defaultKeymap[i].keys
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4976 && defaultKeymap[i].context === ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4977 defaultKeymap.splice(i, 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4978 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4979 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4980 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4981 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4982 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4983 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4984
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4985 var exCommands = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4986 colorscheme: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4987 if (!params.args || params.args.length < 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4988 showConfirm(cm, cm.getOption('theme'));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4989 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4990 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4991 cm.setOption('theme', params.args[0]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4992 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4993 map: function(cm, params, ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4994 var mapArgs = params.args;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4995 if (!mapArgs || mapArgs.length < 2) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4996 if (cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4997 showConfirm(cm, 'Invalid mapping: ' + params.input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4998 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4999 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5000 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5001 exCommandDispatcher.map(mapArgs[0], mapArgs[1], ctx);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5002 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5003 imap: function(cm, params) { this.map(cm, params, 'insert'); },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5004 nmap: function(cm, params) { this.map(cm, params, 'normal'); },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5005 vmap: function(cm, params) { this.map(cm, params, 'visual'); },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5006 unmap: function(cm, params, ctx) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5007 var mapArgs = params.args;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5008 if (!mapArgs || mapArgs.length < 1 || !exCommandDispatcher.unmap(mapArgs[0], ctx)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5009 if (cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5010 showConfirm(cm, 'No such mapping: ' + params.input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5011 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5012 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5013 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5014 move: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5015 commandDispatcher.processCommand(cm, cm.state.vim, {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5016 type: 'motion',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5017 motion: 'moveToLineOrEdgeOfDocument',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5018 motionArgs: { forward: false, explicitRepeat: true,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5019 linewise: true },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5020 repeatOverride: params.line+1});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5021 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5022 set: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5023 var setArgs = params.args;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5024 // Options passed through to the setOption/getOption calls. May be passed in by the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5025 // local/global versions of the set command
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5026 var setCfg = params.setCfg || {};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5027 if (!setArgs || setArgs.length < 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5028 if (cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5029 showConfirm(cm, 'Invalid mapping: ' + params.input);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5030 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5031 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5032 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5033 var expr = setArgs[0].split('=');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5034 var optionName = expr[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5035 var value = expr[1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5036 var forceGet = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5037
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5038 if (optionName.charAt(optionName.length - 1) == '?') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5039 // If post-fixed with ?, then the set is actually a get.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5040 if (value) { throw Error('Trailing characters: ' + params.argString); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5041 optionName = optionName.substring(0, optionName.length - 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5042 forceGet = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5043 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5044 if (value === undefined && optionName.substring(0, 2) == 'no') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5045 // To set boolean options to false, the option name is prefixed with
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5046 // 'no'.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5047 optionName = optionName.substring(2);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5048 value = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5049 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5050
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5051 var optionIsBoolean = options[optionName] && options[optionName].type == 'boolean';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5052 if (optionIsBoolean && value == undefined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5053 // Calling set with a boolean option sets it to true.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5054 value = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5055 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5056 // If no value is provided, then we assume this is a get.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5057 if (!optionIsBoolean && value === undefined || forceGet) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5058 var oldValue = getOption(optionName, cm, setCfg);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5059 if (oldValue instanceof Error) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5060 showConfirm(cm, oldValue.message);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5061 } else if (oldValue === true || oldValue === false) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5062 showConfirm(cm, ' ' + (oldValue ? '' : 'no') + optionName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5063 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5064 showConfirm(cm, ' ' + optionName + '=' + oldValue);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5065 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5066 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5067 var setOptionReturn = setOption(optionName, value, cm, setCfg);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5068 if (setOptionReturn instanceof Error) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5069 showConfirm(cm, setOptionReturn.message);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5070 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5071 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5072 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5073 setlocal: function (cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5074 // setCfg is passed through to setOption
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5075 params.setCfg = {scope: 'local'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5076 this.set(cm, params);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5077 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5078 setglobal: function (cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5079 // setCfg is passed through to setOption
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5080 params.setCfg = {scope: 'global'};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5081 this.set(cm, params);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5082 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5083 registers: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5084 var regArgs = params.args;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5085 var registers = vimGlobalState.registerController.registers;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5086 var regInfo = '----------Registers----------\n\n';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5087 if (!regArgs) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5088 for (var registerName in registers) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5089 var text = registers[registerName].toString();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5090 if (text.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5091 regInfo += '"' + registerName + ' ' + text + '\n';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5092 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5093 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5094 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5095 var registerName;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5096 regArgs = regArgs.join('');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5097 for (var i = 0; i < regArgs.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5098 registerName = regArgs.charAt(i);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5099 if (!vimGlobalState.registerController.isValidRegister(registerName)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5100 continue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5101 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5102 var register = registers[registerName] || new Register();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5103 regInfo += '"' + registerName + ' ' + register.toString() + '\n';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5104 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5105 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5106 showConfirm(cm, regInfo);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5107 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5108 sort: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5109 var reverse, ignoreCase, unique, number, pattern;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5110 function parseArgs() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5111 if (params.argString) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5112 var args = new CodeMirror.StringStream(params.argString);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5113 if (args.eat('!')) { reverse = true; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5114 if (args.eol()) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5115 if (!args.eatSpace()) { return 'Invalid arguments'; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5116 var opts = args.match(/([dinuox]+)?\s*(\/.+\/)?\s*/);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5117 if (!opts && !args.eol()) { return 'Invalid arguments'; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5118 if (opts[1]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5119 ignoreCase = opts[1].indexOf('i') != -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5120 unique = opts[1].indexOf('u') != -1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5121 var decimal = opts[1].indexOf('d') != -1 || opts[1].indexOf('n') != -1 && 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5122 var hex = opts[1].indexOf('x') != -1 && 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5123 var octal = opts[1].indexOf('o') != -1 && 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5124 if (decimal + hex + octal > 1) { return 'Invalid arguments'; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5125 number = decimal && 'decimal' || hex && 'hex' || octal && 'octal';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5126 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5127 if (opts[2]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5128 pattern = new RegExp(opts[2].substr(1, opts[2].length - 2), ignoreCase ? 'i' : '');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5129 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5130 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5131 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5132 var err = parseArgs();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5133 if (err) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5134 showConfirm(cm, err + ': ' + params.argString);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5135 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5136 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5137 var lineStart = params.line || cm.firstLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5138 var lineEnd = params.lineEnd || params.line || cm.lastLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5139 if (lineStart == lineEnd) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5140 var curStart = new Pos(lineStart, 0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5141 var curEnd = new Pos(lineEnd, lineLength(cm, lineEnd));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5142 var text = cm.getRange(curStart, curEnd).split('\n');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5143 var numberRegex = pattern ? pattern :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5144 (number == 'decimal') ? /(-?)([\d]+)/ :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5145 (number == 'hex') ? /(-?)(?:0x)?([0-9a-f]+)/i :
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5146 (number == 'octal') ? /([0-7]+)/ : null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5147 var radix = (number == 'decimal') ? 10 : (number == 'hex') ? 16 : (number == 'octal') ? 8 : null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5148 var numPart = [], textPart = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5149 if (number || pattern) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5150 for (var i = 0; i < text.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5151 var matchPart = pattern ? text[i].match(pattern) : null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5152 if (matchPart && matchPart[0] != '') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5153 numPart.push(matchPart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5154 } else if (!pattern && numberRegex.exec(text[i])) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5155 numPart.push(text[i]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5156 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5157 textPart.push(text[i]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5158 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5159 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5160 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5161 textPart = text;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5162 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5163 function compareFn(a, b) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5164 if (reverse) { var tmp; tmp = a; a = b; b = tmp; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5165 if (ignoreCase) { a = a.toLowerCase(); b = b.toLowerCase(); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5166 var anum = number && numberRegex.exec(a);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5167 var bnum = number && numberRegex.exec(b);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5168 if (!anum) { return a < b ? -1 : 1; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5169 anum = parseInt((anum[1] + anum[2]).toLowerCase(), radix);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5170 bnum = parseInt((bnum[1] + bnum[2]).toLowerCase(), radix);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5171 return anum - bnum;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5172 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5173 function comparePatternFn(a, b) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5174 if (reverse) { var tmp; tmp = a; a = b; b = tmp; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5175 if (ignoreCase) { a[0] = a[0].toLowerCase(); b[0] = b[0].toLowerCase(); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5176 return (a[0] < b[0]) ? -1 : 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5177 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5178 numPart.sort(pattern ? comparePatternFn : compareFn);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5179 if (pattern) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5180 for (var i = 0; i < numPart.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5181 numPart[i] = numPart[i].input;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5182 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5183 } else if (!number) { textPart.sort(compareFn); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5184 text = (!reverse) ? textPart.concat(numPart) : numPart.concat(textPart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5185 if (unique) { // Remove duplicate lines
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5186 var textOld = text;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5187 var lastLine;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5188 text = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5189 for (var i = 0; i < textOld.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5190 if (textOld[i] != lastLine) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5191 text.push(textOld[i]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5192 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5193 lastLine = textOld[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5194 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5195 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5196 cm.replaceRange(text.join('\n'), curStart, curEnd);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5197 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5198 vglobal: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5199 // global inspects params.commandName
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5200 this.global(cm, params);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5201 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5202 global: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5203 // a global command is of the form
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5204 // :[range]g/pattern/[cmd]
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5205 // argString holds the string /pattern/[cmd]
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5206 var argString = params.argString;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5207 if (!argString) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5208 showConfirm(cm, 'Regular Expression missing from global');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5209 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5210 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5211 var inverted = params.commandName[0] === 'v';
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5212 // range is specified here
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5213 var lineStart = (params.line !== undefined) ? params.line : cm.firstLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5214 var lineEnd = params.lineEnd || params.line || cm.lastLine();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5215 // get the tokens from argString
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5216 var tokens = splitBySlash(argString);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5217 var regexPart = argString, cmd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5218 if (tokens.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5219 regexPart = tokens[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5220 cmd = tokens.slice(1, tokens.length).join('/');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5221 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5222 if (regexPart) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5223 // If regex part is empty, then use the previous query. Otherwise
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5224 // use the regex part as the new query.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5225 try {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5226 updateSearchQuery(cm, regexPart, true /** ignoreCase */,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5227 true /** smartCase */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5228 } catch (e) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5229 showConfirm(cm, 'Invalid regex: ' + regexPart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5230 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5231 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5232 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5233 // now that we have the regexPart, search for regex matches in the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5234 // specified range of lines
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5235 var query = getSearchState(cm).getQuery();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5236 var matchedLines = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5237 for (var i = lineStart; i <= lineEnd; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5238 var line = cm.getLineHandle(i);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5239 var matched = query.test(line.text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5240 if (matched !== inverted) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5241 matchedLines.push(cmd ? line : line.text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5242 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5243 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5244 // if there is no [cmd], just display the list of matched lines
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5245 if (!cmd) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5246 showConfirm(cm, matchedLines.join('\n'));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5247 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5248 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5249 var index = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5250 var nextCommand = function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5251 if (index < matchedLines.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5252 var line = matchedLines[index++];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5253 var lineNum = cm.getLineNumber(line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5254 if (lineNum == null) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5255 nextCommand();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5256 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5257 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5258 var command = (lineNum + 1) + cmd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5259 exCommandDispatcher.processCommand(cm, command, {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5260 callback: nextCommand
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5261 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5262 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5263 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5264 nextCommand();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5265 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5266 substitute: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5267 if (!cm.getSearchCursor) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5268 throw new Error('Search feature not available. Requires searchcursor.js or ' +
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5269 'any other getSearchCursor implementation.');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5270 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5271 var argString = params.argString;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5272 var tokens = argString ? splitBySeparator(argString, argString[0]) : [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5273 var regexPart, replacePart = '', trailing, flagsPart, count;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5274 var confirm = false; // Whether to confirm each replace.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5275 var global = false; // True to replace all instances on a line, false to replace only 1.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5276 if (tokens.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5277 regexPart = tokens[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5278 if (getOption('pcre') && regexPart !== '') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5279 regexPart = new RegExp(regexPart).source; //normalize not escaped characters
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5280 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5281 replacePart = tokens[1];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5282 if (replacePart !== undefined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5283 if (getOption('pcre')) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5284 replacePart = unescapeRegexReplace(replacePart.replace(/([^\\])&/g,"$1$$&"));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5285 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5286 replacePart = translateRegexReplace(replacePart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5287 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5288 vimGlobalState.lastSubstituteReplacePart = replacePart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5289 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5290 trailing = tokens[2] ? tokens[2].split(' ') : [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5291 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5292 // either the argString is empty or its of the form ' hello/world'
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5293 // actually splitBySlash returns a list of tokens
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5294 // only if the string starts with a '/'
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5295 if (argString && argString.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5296 showConfirm(cm, 'Substitutions should be of the form ' +
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5297 ':s/pattern/replace/');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5298 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5299 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5300 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5301 // After the 3rd slash, we can have flags followed by a space followed
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5302 // by count.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5303 if (trailing) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5304 flagsPart = trailing[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5305 count = parseInt(trailing[1]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5306 if (flagsPart) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5307 if (flagsPart.indexOf('c') != -1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5308 confirm = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5309 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5310 if (flagsPart.indexOf('g') != -1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5311 global = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5312 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5313 if (getOption('pcre')) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5314 regexPart = regexPart + '/' + flagsPart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5315 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5316 regexPart = regexPart.replace(/\//g, "\\/") + '/' + flagsPart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5317 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5318 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5319 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5320 if (regexPart) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5321 // If regex part is empty, then use the previous query. Otherwise use
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5322 // the regex part as the new query.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5323 try {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5324 updateSearchQuery(cm, regexPart, true /** ignoreCase */,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5325 true /** smartCase */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5326 } catch (e) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5327 showConfirm(cm, 'Invalid regex: ' + regexPart);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5328 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5329 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5330 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5331 replacePart = replacePart || vimGlobalState.lastSubstituteReplacePart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5332 if (replacePart === undefined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5333 showConfirm(cm, 'No previous substitute regular expression');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5334 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5335 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5336 var state = getSearchState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5337 var query = state.getQuery();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5338 var lineStart = (params.line !== undefined) ? params.line : cm.getCursor().line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5339 var lineEnd = params.lineEnd || lineStart;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5340 if (lineStart == cm.firstLine() && lineEnd == cm.lastLine()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5341 lineEnd = Infinity;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5342 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5343 if (count) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5344 lineStart = lineEnd;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5345 lineEnd = lineStart + count - 1;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5346 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5347 var startPos = clipCursorToContent(cm, new Pos(lineStart, 0));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5348 var cursor = cm.getSearchCursor(query, startPos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5349 doReplace(cm, confirm, global, lineStart, lineEnd, cursor, query, replacePart, params.callback);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5350 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5351 redo: CodeMirror.commands.redo,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5352 undo: CodeMirror.commands.undo,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5353 write: function(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5354 if (CodeMirror.commands.save) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5355 // If a save command is defined, call it.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5356 CodeMirror.commands.save(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5357 } else if (cm.save) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5358 // Saves to text area if no save command is defined and cm.save() is available.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5359 cm.save();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5360 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5361 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5362 nohlsearch: function(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5363 clearSearchHighlight(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5364 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5365 yank: function (cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5366 var cur = copyCursor(cm.getCursor());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5367 var line = cur.line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5368 var lineText = cm.getLine(line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5369 vimGlobalState.registerController.pushText(
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5370 '0', 'yank', lineText, true, true);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5371 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5372 delmarks: function(cm, params) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5373 if (!params.argString || !trim(params.argString)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5374 showConfirm(cm, 'Argument required');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5375 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5376 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5377
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5378 var state = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5379 var stream = new CodeMirror.StringStream(trim(params.argString));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5380 while (!stream.eol()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5381 stream.eatSpace();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5382
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5383 // Record the streams position at the beginning of the loop for use
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5384 // in error messages.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5385 var count = stream.pos;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5386
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5387 if (!stream.match(/[a-zA-Z]/, false)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5388 showConfirm(cm, 'Invalid argument: ' + params.argString.substring(count));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5389 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5390 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5391
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5392 var sym = stream.next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5393 // Check if this symbol is part of a range
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5394 if (stream.match('-', true)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5395 // This symbol is part of a range.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5396
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5397 // The range must terminate at an alphabetic character.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5398 if (!stream.match(/[a-zA-Z]/, false)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5399 showConfirm(cm, 'Invalid argument: ' + params.argString.substring(count));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5400 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5401 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5402
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5403 var startMark = sym;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5404 var finishMark = stream.next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5405 // The range must terminate at an alphabetic character which
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5406 // shares the same case as the start of the range.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5407 if (isLowerCase(startMark) && isLowerCase(finishMark) ||
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5408 isUpperCase(startMark) && isUpperCase(finishMark)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5409 var start = startMark.charCodeAt(0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5410 var finish = finishMark.charCodeAt(0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5411 if (start >= finish) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5412 showConfirm(cm, 'Invalid argument: ' + params.argString.substring(count));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5413 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5414 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5415
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5416 // Because marks are always ASCII values, and we have
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5417 // determined that they are the same case, we can use
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5418 // their char codes to iterate through the defined range.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5419 for (var j = 0; j <= finish - start; j++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5420 var mark = String.fromCharCode(start + j);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5421 delete state.marks[mark];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5422 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5423 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5424 showConfirm(cm, 'Invalid argument: ' + startMark + '-');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5425 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5426 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5427 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5428 // This symbol is a valid mark, and is not part of a range.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5429 delete state.marks[sym];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5430 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5431 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5432 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5433 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5434
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5435 var exCommandDispatcher = new ExCommandDispatcher();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5436
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5437 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5438 * @param {CodeMirror} cm CodeMirror instance we are in.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5439 * @param {boolean} confirm Whether to confirm each replace.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5440 * @param {Cursor} lineStart Line to start replacing from.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5441 * @param {Cursor} lineEnd Line to stop replacing at.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5442 * @param {RegExp} query Query for performing matches with.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5443 * @param {string} replaceWith Text to replace matches with. May contain $1,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5444 * $2, etc for replacing captured groups using JavaScript replace.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5445 * @param {function()} callback A callback for when the replace is done.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5446 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5447 function doReplace(cm, confirm, global, lineStart, lineEnd, searchCursor, query,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5448 replaceWith, callback) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5449 // Set up all the functions.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5450 cm.state.vim.exMode = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5451 var done = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5452 var lastPos, modifiedLineNumber, joined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5453 function replaceAll() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5454 cm.operation(function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5455 while (!done) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5456 replace();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5457 next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5458 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5459 stop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5460 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5461 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5462 function replace() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5463 var text = cm.getRange(searchCursor.from(), searchCursor.to());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5464 var newText = text.replace(query, replaceWith);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5465 var unmodifiedLineNumber = searchCursor.to().line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5466 searchCursor.replace(newText);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5467 modifiedLineNumber = searchCursor.to().line;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5468 lineEnd += modifiedLineNumber - unmodifiedLineNumber;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5469 joined = modifiedLineNumber < unmodifiedLineNumber;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5470 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5471 function findNextValidMatch() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5472 var lastMatchTo = lastPos && copyCursor(searchCursor.to());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5473 var match = searchCursor.findNext();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5474 if (match && !match[0] && lastMatchTo && cursorEqual(searchCursor.from(), lastMatchTo)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5475 match = searchCursor.findNext();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5476 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5477 return match;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5478 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5479 function next() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5480 // The below only loops to skip over multiple occurrences on the same
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5481 // line when 'global' is not true.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5482 while(findNextValidMatch() &&
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5483 isInRange(searchCursor.from(), lineStart, lineEnd)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5484 if (!global && searchCursor.from().line == modifiedLineNumber && !joined) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5485 continue;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5486 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5487 cm.scrollIntoView(searchCursor.from(), 30);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5488 cm.setSelection(searchCursor.from(), searchCursor.to());
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5489 lastPos = searchCursor.from();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5490 done = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5491 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5492 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5493 done = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5494 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5495 function stop(close) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5496 if (close) { close(); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5497 cm.focus();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5498 if (lastPos) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5499 cm.setCursor(lastPos);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5500 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5501 vim.exMode = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5502 vim.lastHPos = vim.lastHSPos = lastPos.ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5503 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5504 if (callback) { callback(); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5505 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5506 function onPromptKeyDown(e, _value, close) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5507 // Swallow all keys.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5508 CodeMirror.e_stop(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5509 var keyName = CodeMirror.keyName(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5510 switch (keyName) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5511 case 'Y':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5512 replace(); next(); break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5513 case 'N':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5514 next(); break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5515 case 'A':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5516 // replaceAll contains a call to close of its own. We don't want it
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5517 // to fire too early or multiple times.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5518 var savedCallback = callback;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5519 callback = undefined;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5520 cm.operation(replaceAll);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5521 callback = savedCallback;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5522 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5523 case 'L':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5524 replace();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5525 // fall through and exit.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5526 case 'Q':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5527 case 'Esc':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5528 case 'Ctrl-C':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5529 case 'Ctrl-[':
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5530 stop(close);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5531 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5532 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5533 if (done) { stop(close); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5534 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5535 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5536
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5537 // Actually do replace.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5538 next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5539 if (done) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5540 showConfirm(cm, 'No matches for ' + query.source);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5541 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5542 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5543 if (!confirm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5544 replaceAll();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5545 if (callback) { callback(); }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5546 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5547 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5548 showPrompt(cm, {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5549 prefix: dom('span', 'replace with ', dom('strong', replaceWith), ' (y/n/a/q/l)'),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5550 onKeyDown: onPromptKeyDown
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5551 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5552 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5553
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5554 CodeMirror.keyMap.vim = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5555 attach: attachVimMap,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5556 detach: detachVimMap,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5557 call: cmKey
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5558 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5559
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5560 function exitInsertMode(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5561 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5562 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5563 var insertModeChangeRegister = vimGlobalState.registerController.getRegister('.');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5564 var isPlaying = macroModeState.isPlaying;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5565 var lastChange = macroModeState.lastInsertModeChanges;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5566 if (!isPlaying) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5567 cm.off('change', onChange);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5568 CodeMirror.off(cm.getInputField(), 'keydown', onKeyEventTargetKeyDown);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5569 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5570 if (!isPlaying && vim.insertModeRepeat > 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5571 // Perform insert mode repeat for commands like 3,a and 3,o.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5572 repeatLastEdit(cm, vim, vim.insertModeRepeat - 1,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5573 true /** repeatForInsert */);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5574 vim.lastEditInputState.repeatOverride = vim.insertModeRepeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5575 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5576 delete vim.insertModeRepeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5577 vim.insertMode = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5578 cm.setCursor(cm.getCursor().line, cm.getCursor().ch-1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5579 cm.setOption('keyMap', 'vim');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5580 cm.setOption('disableInput', true);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5581 cm.toggleOverwrite(false); // exit replace mode if we were in it.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5582 // update the ". register before exiting insert mode
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5583 insertModeChangeRegister.setText(lastChange.changes.join(''));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5584 CodeMirror.signal(cm, "vim-mode-change", {mode: "normal"});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5585 if (macroModeState.isRecording) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5586 logInsertModeChange(macroModeState);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5587 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5588 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5589
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5590 function _mapCommand(command) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5591 defaultKeymap.unshift(command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5592 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5593
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5594 function mapCommand(keys, type, name, args, extra) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5595 var command = {keys: keys, type: type};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5596 command[type] = name;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5597 command[type + "Args"] = args;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5598 for (var key in extra)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5599 command[key] = extra[key];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5600 _mapCommand(command);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5601 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5602
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5603 // The timeout in milliseconds for the two-character ESC keymap should be
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5604 // adjusted according to your typing speed to prevent false positives.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5605 defineOption('insertModeEscKeysTimeout', 200, 'number');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5606
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5607 CodeMirror.keyMap['vim-insert'] = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5608 // TODO: override navigation keys so that Esc will cancel automatic
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5609 // indentation from o, O, i_<CR>
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5610 fallthrough: ['default'],
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5611 attach: attachVimMap,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5612 detach: detachVimMap,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5613 call: cmKey
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5614 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5615
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5616 CodeMirror.keyMap['vim-replace'] = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5617 'Backspace': 'goCharLeft',
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5618 fallthrough: ['vim-insert'],
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5619 attach: attachVimMap,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5620 detach: detachVimMap,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5621 call: cmKey
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5622 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5623
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5624 function executeMacroRegister(cm, vim, macroModeState, registerName) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5625 var register = vimGlobalState.registerController.getRegister(registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5626 if (registerName == ':') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5627 // Read-only register containing last Ex command.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5628 if (register.keyBuffer[0]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5629 exCommandDispatcher.processCommand(cm, register.keyBuffer[0]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5630 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5631 macroModeState.isPlaying = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5632 return;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5633 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5634 var keyBuffer = register.keyBuffer;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5635 var imc = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5636 macroModeState.isPlaying = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5637 macroModeState.replaySearchQueries = register.searchQueries.slice(0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5638 for (var i = 0; i < keyBuffer.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5639 var text = keyBuffer[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5640 var match, key;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5641 while (text) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5642 // Pull off one command key, which is either a single character
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5643 // or a special sequence wrapped in '<' and '>', e.g. '<Space>'.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5644 match = (/<\w+-.+?>|<\w+>|./).exec(text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5645 key = match[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5646 text = text.substring(match.index + key.length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5647 vimApi.handleKey(cm, key, 'macro');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5648 if (vim.insertMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5649 var changes = register.insertModeChanges[imc++].changes;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5650 vimGlobalState.macroModeState.lastInsertModeChanges.changes =
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5651 changes;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5652 repeatInsertModeChanges(cm, changes, 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5653 exitInsertMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5654 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5655 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5656 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5657 macroModeState.isPlaying = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5658 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5659
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5660 function logKey(macroModeState, key) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5661 if (macroModeState.isPlaying) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5662 var registerName = macroModeState.latestRegister;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5663 var register = vimGlobalState.registerController.getRegister(registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5664 if (register) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5665 register.pushText(key);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5666 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5667 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5668
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5669 function logInsertModeChange(macroModeState) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5670 if (macroModeState.isPlaying) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5671 var registerName = macroModeState.latestRegister;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5672 var register = vimGlobalState.registerController.getRegister(registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5673 if (register && register.pushInsertModeChanges) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5674 register.pushInsertModeChanges(macroModeState.lastInsertModeChanges);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5675 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5676 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5677
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5678 function logSearchQuery(macroModeState, query) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5679 if (macroModeState.isPlaying) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5680 var registerName = macroModeState.latestRegister;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5681 var register = vimGlobalState.registerController.getRegister(registerName);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5682 if (register && register.pushSearchQuery) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5683 register.pushSearchQuery(query);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5684 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5685 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5686
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5687 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5688 * Listens for changes made in insert mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5689 * Should only be active in insert mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5690 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5691 function onChange(cm, changeObj) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5692 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5693 var lastChange = macroModeState.lastInsertModeChanges;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5694 if (!macroModeState.isPlaying) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5695 while(changeObj) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5696 lastChange.expectCursorActivityForChange = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5697 if (lastChange.ignoreCount > 1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5698 lastChange.ignoreCount--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5699 } else if (changeObj.origin == '+input' || changeObj.origin == 'paste'
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5700 || changeObj.origin === undefined /* only in testing */) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5701 var selectionCount = cm.listSelections().length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5702 if (selectionCount > 1)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5703 lastChange.ignoreCount = selectionCount;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5704 var text = changeObj.text.join('\n');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5705 if (lastChange.maybeReset) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5706 lastChange.changes = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5707 lastChange.maybeReset = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5708 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5709 if (text) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5710 if (cm.state.overwrite && !/\n/.test(text)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5711 lastChange.changes.push([text]);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5712 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5713 lastChange.changes.push(text);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5714 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5715 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5716 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5717 // Change objects may be chained with next.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5718 changeObj = changeObj.next;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5719 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5720 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5721 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5722
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5723 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5724 * Listens for any kind of cursor activity on CodeMirror.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5725 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5726 function onCursorActivity(cm) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5727 var vim = cm.state.vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5728 if (vim.insertMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5729 // Tracking cursor activity in insert mode (for macro support).
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5730 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5731 if (macroModeState.isPlaying) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5732 var lastChange = macroModeState.lastInsertModeChanges;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5733 if (lastChange.expectCursorActivityForChange) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5734 lastChange.expectCursorActivityForChange = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5735 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5736 // Cursor moved outside the context of an edit. Reset the change.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5737 lastChange.maybeReset = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5738 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5739 } else if (!cm.curOp.isVimOp) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5740 handleExternalSelection(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5741 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5742 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5743 function handleExternalSelection(cm, vim) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5744 var anchor = cm.getCursor('anchor');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5745 var head = cm.getCursor('head');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5746 // Enter or exit visual mode to match mouse selection.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5747 if (vim.visualMode && !cm.somethingSelected()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5748 exitVisualMode(cm, false);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5749 } else if (!vim.visualMode && !vim.insertMode && cm.somethingSelected()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5750 vim.visualMode = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5751 vim.visualLine = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5752 CodeMirror.signal(cm, "vim-mode-change", {mode: "visual"});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5753 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5754 if (vim.visualMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5755 // Bind CodeMirror selection model to vim selection model.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5756 // Mouse selections are considered visual characterwise.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5757 var headOffset = !cursorIsBefore(head, anchor) ? -1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5758 var anchorOffset = cursorIsBefore(head, anchor) ? -1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5759 head = offsetCursor(head, 0, headOffset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5760 anchor = offsetCursor(anchor, 0, anchorOffset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5761 vim.sel = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5762 anchor: anchor,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5763 head: head
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5764 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5765 updateMark(cm, vim, '<', cursorMin(head, anchor));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5766 updateMark(cm, vim, '>', cursorMax(head, anchor));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5767 } else if (!vim.insertMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5768 // Reset lastHPos if selection was modified by something outside of vim mode e.g. by mouse.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5769 vim.lastHPos = cm.getCursor().ch;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5770 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5771 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5772
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5773 /** Wrapper for special keys pressed in insert mode */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5774 function InsertModeKey(keyName) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5775 this.keyName = keyName;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5776 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5777
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5778 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5779 * Handles raw key down events from the text area.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5780 * - Should only be active in insert mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5781 * - For recording deletes in insert mode.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5782 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5783 function onKeyEventTargetKeyDown(e) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5784 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5785 var lastChange = macroModeState.lastInsertModeChanges;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5786 var keyName = CodeMirror.keyName(e);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5787 if (!keyName) { return; }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5788 function onKeyFound() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5789 if (lastChange.maybeReset) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5790 lastChange.changes = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5791 lastChange.maybeReset = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5792 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5793 lastChange.changes.push(new InsertModeKey(keyName));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5794 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5795 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5796 if (keyName.indexOf('Delete') != -1 || keyName.indexOf('Backspace') != -1) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5797 CodeMirror.lookupKey(keyName, 'vim-insert', onKeyFound);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5798 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5799 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5800
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5801 /**
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5802 * Repeats the last edit, which includes exactly 1 command and at most 1
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5803 * insert. Operator and motion commands are read from lastEditInputState,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5804 * while action commands are read from lastEditActionCommand.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5805 *
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5806 * If repeatForInsert is true, then the function was called by
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5807 * exitInsertMode to repeat the insert mode changes the user just made. The
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5808 * corresponding enterInsertMode call was made with a count.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5809 */
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5810 function repeatLastEdit(cm, vim, repeat, repeatForInsert) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5811 var macroModeState = vimGlobalState.macroModeState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5812 macroModeState.isPlaying = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5813 var isAction = !!vim.lastEditActionCommand;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5814 var cachedInputState = vim.inputState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5815 function repeatCommand() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5816 if (isAction) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5817 commandDispatcher.processAction(cm, vim, vim.lastEditActionCommand);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5818 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5819 commandDispatcher.evalInput(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5820 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5821 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5822 function repeatInsert(repeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5823 if (macroModeState.lastInsertModeChanges.changes.length > 0) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5824 // For some reason, repeat cw in desktop VIM does not repeat
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5825 // insert mode changes. Will conform to that behavior.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5826 repeat = !vim.lastEditActionCommand ? 1 : repeat;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5827 var changeObject = macroModeState.lastInsertModeChanges;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5828 repeatInsertModeChanges(cm, changeObject.changes, repeat);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5829 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5830 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5831 vim.inputState = vim.lastEditInputState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5832 if (isAction && vim.lastEditActionCommand.interlaceInsertRepeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5833 // o and O repeat have to be interlaced with insert repeats so that the
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5834 // insertions appear on separate lines instead of the last line.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5835 for (var i = 0; i < repeat; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5836 repeatCommand();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5837 repeatInsert(1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5838 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5839 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5840 if (!repeatForInsert) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5841 // Hack to get the cursor to end up at the right place. If I is
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5842 // repeated in insert mode repeat, cursor will be 1 insert
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5843 // change set left of where it should be.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5844 repeatCommand();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5845 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5846 repeatInsert(repeat);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5847 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5848 vim.inputState = cachedInputState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5849 if (vim.insertMode && !repeatForInsert) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5850 // Don't exit insert mode twice. If repeatForInsert is set, then we
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5851 // were called by an exitInsertMode call lower on the stack.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5852 exitInsertMode(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5853 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5854 macroModeState.isPlaying = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5855 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5856
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5857 function repeatInsertModeChanges(cm, changes, repeat) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5858 function keyHandler(binding) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5859 if (typeof binding == 'string') {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5860 CodeMirror.commands[binding](cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5861 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5862 binding(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5863 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5864 return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5865 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5866 var head = cm.getCursor('head');
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5867 var visualBlock = vimGlobalState.macroModeState.lastInsertModeChanges.visualBlock;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5868 if (visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5869 // Set up block selection again for repeating the changes.
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5870 selectForInsert(cm, head, visualBlock + 1);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5871 repeat = cm.listSelections().length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5872 cm.setCursor(head);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5873 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5874 for (var i = 0; i < repeat; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5875 if (visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5876 cm.setCursor(offsetCursor(head, i, 0));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5877 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5878 for (var j = 0; j < changes.length; j++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5879 var change = changes[j];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5880 if (change instanceof InsertModeKey) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5881 CodeMirror.lookupKey(change.keyName, 'vim-insert', keyHandler);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5882 } else if (typeof change == "string") {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5883 cm.replaceSelection(change);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5884 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5885 var start = cm.getCursor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5886 var end = offsetCursor(start, 0, change[0].length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5887 cm.replaceRange(change[0], start, end);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5888 cm.setCursor(end);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5889 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5890 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5891 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5892 if (visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5893 cm.setCursor(offsetCursor(head, 0, 1));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5894 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5895 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5896
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5897 // multiselect support
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5898 function cloneVimState(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5899 var n = new state.constructor();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5900 Object.keys(state).forEach(function(key) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5901 var o = state[key];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5902 if (Array.isArray(o))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5903 o = o.slice();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5904 else if (o && typeof o == "object" && o.constructor != Object)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5905 o = cloneVimState(o);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5906 n[key] = o;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5907 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5908 if (state.sel) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5909 n.sel = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5910 head: state.sel.head && copyCursor(state.sel.head),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5911 anchor: state.sel.anchor && copyCursor(state.sel.anchor)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5912 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5913 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5914 return n;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5915 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5916 function multiSelectHandleKey(cm, key, origin) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5917 var isHandled = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5918 var vim = vimApi.maybeInitVimState_(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5919 var visualBlock = vim.visualBlock || vim.wasInVisualBlock;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5920
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5921 var wasMultiselect = cm.isInMultiSelectMode();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5922 if (vim.wasInVisualBlock && !wasMultiselect) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5923 vim.wasInVisualBlock = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5924 } else if (wasMultiselect && vim.visualBlock) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5925 vim.wasInVisualBlock = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5926 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5927
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5928 if (key == '<Esc>' && !vim.insertMode && !vim.visualMode && wasMultiselect && vim.status == "<Esc>") {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5929 // allow editor to exit multiselect
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5930 clearInputState(cm);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5931 } else if (visualBlock || !wasMultiselect || cm.inVirtualSelectionMode) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5932 isHandled = vimApi.handleKey(cm, key, origin);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5933 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5934 var old = cloneVimState(vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5935
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5936 cm.operation(function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5937 cm.curOp.isVimOp = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5938 cm.forEachSelection(function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5939 var head = cm.getCursor("head");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5940 var anchor = cm.getCursor("anchor");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5941 var headOffset = !cursorIsBefore(head, anchor) ? -1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5942 var anchorOffset = cursorIsBefore(head, anchor) ? -1 : 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5943 head = offsetCursor(head, 0, headOffset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5944 anchor = offsetCursor(anchor, 0, anchorOffset);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5945 cm.state.vim.sel.head = head;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5946 cm.state.vim.sel.anchor = anchor;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5947
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5948 isHandled = vimApi.handleKey(cm, key, origin);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5949 if (cm.virtualSelection) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5950 cm.state.vim = cloneVimState(old);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5951 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5952 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5953 if (cm.curOp.cursorActivity && !isHandled)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5954 cm.curOp.cursorActivity = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5955 cm.state.vim = vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5956 }, true);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5957 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5958 // some commands may bring visualMode and selection out of sync
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5959 if (isHandled && !vim.visualMode && !vim.insert && vim.visualMode != cm.somethingSelected()) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5960 handleExternalSelection(cm, vim);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5961 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5962 return isHandled;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5963 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5964 resetVimGlobalState();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5965
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5966 return vimApi;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5967 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5968
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5969 function initVim(CodeMirror5) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5970 CodeMirror5.Vim = initVim$1(CodeMirror5);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5971 return CodeMirror5.Vim;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5972 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5973
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5974
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5975
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5976 CodeMirror.Vim = initVim(CodeMirror);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5977 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5978