1
/* ***** BEGIN LICENSE BLOCK *****
2
* Distributed under the BSD license:
4
* Copyright (c) 2010, Ajax.org B.V.
7
* Redistribution and use in source and binary forms, with or without
8
* modification, are permitted provided that the following conditions are met:
9
* * Redistributions of source code must retain the above copyright
10
* notice, this list of conditions and the following disclaimer.
11
* * Redistributions in binary form must reproduce the above copyright
12
* notice, this list of conditions and the following disclaimer in the
13
* documentation and/or other materials provided with the distribution.
14
* * Neither the name of Ajax.org B.V. nor the
15
* names of its contributors may be used to endorse or promote products
16
* derived from this software without specific prior written permission.
18
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
19
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21
* DISCLAIMED. IN NO EVENT SHALL AJAX.ORG B.V. BE LIABLE FOR ANY
22
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
25
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
* ***** END LICENSE BLOCK ***** */
31
define('ace/keyboard/emacs', ['require', 'exports', 'module' , 'ace/lib/dom', 'ace/keyboard/hash_handler', 'ace/lib/keys'], function(require, exports, module) {
34
var dom = require("../lib/dom");
36
var screenToTextBlockCoordinates = function(x, y) {
37
var canvasPos = this.scroller.getBoundingClientRect();
40
(x + this.scrollLeft - canvasPos.left - this.$padding) / this.characterWidth
43
(y + this.scrollTop - canvasPos.top) / this.lineHeight
46
return this.session.screenToDocumentPosition(row, col);
49
var HashHandler = require("./hash_handler").HashHandler;
50
exports.handler = new HashHandler();
52
var initialized = false;
56
exports.handler.attach = function(editor) {
59
dom.importCssString('\
60
.emacs-mode .ace_cursor{\
61
border: 2px rgba(50,250,50,0.8) solid!important;\
62
-moz-box-sizing: border-box!important;\
63
-webkit-box-sizing: border-box!important;\
64
box-sizing: border-box!important;\
65
background-color: rgba(0,250,0,0.9);\
68
.emacs-mode .ace_cursor.ace_hidden{\
70
background-color: transparent;\
72
.emacs-mode .ace_overwrite-cursors .ace_cursor {\
74
background-color: transparent;\
75
border-width: 0 0 2px 2px !important;\
77
.emacs-mode .ace_text-layer {\
80
.emacs-mode .ace_cursor-layer {\
85
$formerLongWords = editor.session.$selectLongWords;
86
editor.session.$selectLongWords = true;
87
$formerLineStart = editor.session.$useEmacsStyleLineStart;
88
editor.session.$useEmacsStyleLineStart = true;
90
editor.session.$emacsMark = null;
92
exports.markMode = function() {
93
return editor.session.$emacsMark;
96
exports.setMarkMode = function(p) {
97
editor.session.$emacsMark = p;
100
editor.on("click",$resetMarkMode);
101
editor.on("changeSession",$kbSessionChange);
102
editor.renderer.screenToTextCoordinates = screenToTextBlockCoordinates;
103
editor.setStyle("emacs-mode");
104
editor.commands.addCommands(commands);
105
exports.handler.platform = editor.commands.platform;
108
exports.handler.detach = function(editor) {
109
delete editor.renderer.screenToTextCoordinates;
110
editor.session.$selectLongWords = $formerLongWords;
111
editor.session.$useEmacsStyleLineStart = $formerLineStart;
112
editor.removeEventListener("click",$resetMarkMode);
113
editor.removeEventListener("changeSession",$kbSessionChange);
114
editor.unsetStyle("emacs-mode");
115
editor.commands.removeCommands(commands);
118
var $kbSessionChange = function(e) {
120
e.oldSession.$selectLongWords = $formerLongWords;
121
e.oldSession.$useEmacsStyleLineStart = $formerLineStart;
124
$formerLongWords = e.session.$selectLongWords;
125
e.session.$selectLongWords = true;
126
$formerLineStart = e.session.$useEmacsStyleLineStart;
127
e.session.$useEmacsStyleLineStart = true;
129
if (!e.session.hasOwnProperty('$emacsMark'))
130
e.session.$emacsMark = null;
133
var $resetMarkMode = function(e) {
134
e.editor.session.$emacsMark = null;
137
var keys = require("../lib/keys").KEY_MODS,
138
eMods = {C: "ctrl", S: "shift", M: "alt", CMD: "command"},
139
combinations = ["C-S-M-CMD",
140
"S-M-CMD", "C-M-CMD", "C-S-CMD", "C-S-M",
141
"M-CMD", "S-CMD", "S-M", "C-CMD", "C-M", "C-S",
142
"CMD", "M", "S", "C"];
143
combinations.forEach(function(c) {
145
c.split("-").forEach(function(c) {
146
hashId = hashId | keys[eMods[c]];
148
eMods[hashId] = c.toLowerCase() + "-";
151
exports.handler.bindKey = function(key, command) {
155
var ckb = this.commmandKeyBinding;
156
key.split("|").forEach(function(keyPart) {
157
keyPart = keyPart.toLowerCase();
158
ckb[keyPart] = command;
159
keyPart = keyPart.split(" ")[0];
161
ckb[keyPart] = "null";
166
exports.handler.handleKeyboard = function(data, hashId, key, keyCode) {
168
exports.setMarkMode(null);
170
var str = Array(data.count + 1).join(key);
172
return {command: "insertstring", args: str};
179
var modifier = eMods[hashId];
180
if (modifier == "c-" || data.universalArgument) {
181
var count = parseInt(key[key.length - 1]);
184
return {command: "null"};
187
data.universalArgument = false;
188
if (modifier) key = modifier + key;
189
if (data.keyChain) key = data.keyChain += " " + key;
190
var command = this.commmandKeyBinding[key];
191
data.keyChain = command == "null" ? key : "";
192
if (!command) return;
193
if (command === "null") return {command: "null"};
195
if (command === "universalArgument") {
196
data.universalArgument = true;
197
return {command: "null"};
200
if (typeof command !== "string") {
202
if (command.command) command = command.command;
203
if (command === "goorselect") {
204
command = exports.markMode() ? args[1] : args[0];
209
if (typeof command === "string") {
210
if (command === "insertstring" ||
211
command === "splitline" ||
212
command === "togglecomment") {
213
exports.setMarkMode(null);
215
command = this.commands[command] || data.editor.commands.commands[command];
218
if (!command.readonly && !command.isYank)
219
data.lastCommand = null;
222
var count = data.count;
227
exec: function(editor, args) {
228
for (var i = 0; i < count; i++)
229
command.exec(editor, args);
235
return {command: command, args: args};
238
exports.emacsKeys = {
239
"Up|C-p" : {command: "goorselect", args: ["golineup","selectup"]},
240
"Down|C-n" : {command: "goorselect", args: ["golinedown","selectdown"]},
241
"Left|C-b" : {command: "goorselect", args: ["gotoleft","selectleft"]},
242
"Right|C-f" : {command: "goorselect", args: ["gotoright","selectright"]},
243
"C-Left|M-b" : {command: "goorselect", args: ["gotowordleft","selectwordleft"]},
244
"C-Right|M-f" : {command: "goorselect", args: ["gotowordright","selectwordright"]},
245
"Home|C-a" : {command: "goorselect", args: ["gotolinestart","selecttolinestart"]},
246
"End|C-e" : {command: "goorselect", args: ["gotolineend","selecttolineend"]},
247
"C-Home|S-M-,": {command: "goorselect", args: ["gotostart","selecttostart"]},
248
"C-End|S-M-." : {command: "goorselect", args: ["gotoend","selecttoend"]},
249
"S-Up|S-C-p" : "selectup",
250
"S-Down|S-C-n" : "selectdown",
251
"S-Left|S-C-b" : "selectleft",
252
"S-Right|S-C-f" : "selectright",
253
"S-C-Left|S-M-b" : "selectwordleft",
254
"S-C-Right|S-M-f" : "selectwordright",
255
"S-Home|S-C-a" : "selecttolinestart",
256
"S-End|S-C-e" : "selecttolineend",
257
"S-C-Home" : "selecttostart",
258
"S-C-End" : "selecttoend",
260
"C-l" : "recenterTopBottom",
261
"M-s" : "centerselection",
263
"C-x C-p": "selectall",
264
"C-Down": {command: "goorselect", args: ["gotopagedown","selectpagedown"]},
265
"C-Up": {command: "goorselect", args: ["gotopageup","selectpageup"]},
266
"PageDown|C-v": {command: "goorselect", args: ["gotopagedown","selectpagedown"]},
267
"PageUp|M-v": {command: "goorselect", args: ["gotopageup","selectpageup"]},
268
"S-C-Down": "selectpagedown",
269
"S-C-Up": "selectpageup",
271
"C-r": "findprevious",
273
"M-C-r": "findprevious",
275
"Backspace": "backspace",
277
"Return|C-m": {command: "insertstring", args: "\n"}, // "newline"
280
"M-d|C-Delete": {command: "killWord", args: "right"},
281
"C-Backspace|M-Backspace|M-Delete": {command: "killWord", args: "left"},
284
"C-y|S-Delete": "yank",
286
"C-g": "keyboardQuit",
289
"M-w": "killRingSave",
290
"C-Space": "setMark",
291
"C-x C-x": "exchangePointAndMark",
293
"C-t": "transposeletters",
294
"M-u": "touppercase", // Doesn't work
295
"M-l": "tolowercase",
296
"M-/": "autocomplete", // Doesn't work
297
"C-u": "universalArgument",
299
"M-;": "togglecomment",
301
"C-/|C-x u|S-C--|C-z": "undo",
302
"S-C-/|S-C-x u|C--|S-C-z": "redo", //infinite undo?
303
"C-x r": "selectRectangularRegion"
307
exports.handler.bindKeys(exports.emacsKeys);
309
exports.handler.addCommands({
310
recenterTopBottom: function(editor) {
311
var renderer = editor.renderer;
312
var pos = renderer.$cursorLayer.getPixelPosition();
313
var h = renderer.$size.scrollerHeight - renderer.lineHeight;
314
var scrollTop = renderer.scrollTop;
315
if (Math.abs(pos.top - scrollTop) < 2) {
316
scrollTop = pos.top - h;
317
} else if (Math.abs(pos.top - scrollTop - h * 0.5) < 2) {
320
scrollTop = pos.top - h * 0.5;
322
editor.session.setScrollTop(scrollTop);
324
selectRectangularRegion: function(editor) {
325
editor.multiSelect.toggleBlockSelection();
327
setMark: function(editor) {
328
var markMode = exports.markMode();
330
var cp = editor.getCursorPosition();
331
if (editor.selection.isEmpty() &&
332
markMode.row == cp.row && markMode.column == cp.column) {
333
exports.setMarkMode(null);
337
markMode = editor.getCursorPosition();
338
exports.setMarkMode(markMode);
339
editor.selection.setSelectionAnchor(markMode.row, markMode.column);
342
exchangePointAndMark: {
343
exec: function(editor) {
344
var range = editor.selection.getRange();
345
editor.selection.setSelectionRange(range, !editor.selection.isBackwards());
348
multiselectAction: "forEach"
351
exec: function(editor, dir) {
352
editor.clearSelection();
354
editor.selection.selectWordLeft();
356
editor.selection.selectWordRight();
358
var range = editor.getSelectionRange();
359
var text = editor.session.getTextRange(range);
360
exports.killRing.add(text);
362
editor.session.remove(range);
363
editor.clearSelection();
365
multiselectAction: "forEach"
367
killLine: function(editor) {
368
exports.setMarkMode(null);
369
var pos = editor.getCursorPosition();
370
if (pos.column == 0 &&
371
editor.session.doc.getLine(pos.row).length == 0) {
372
editor.selection.selectLine();
374
editor.clearSelection();
375
editor.selection.selectLineEnd();
377
var range = editor.getSelectionRange();
378
var text = editor.session.getTextRange(range);
379
exports.killRing.add(text);
381
editor.session.remove(range);
382
editor.clearSelection();
384
yank: function(editor) {
385
editor.onPaste(exports.killRing.get());
386
editor.keyBinding.$data.lastCommand = "yank";
388
yankRotate: function(editor) {
389
if (editor.keyBinding.$data.lastCommand != "yank")
392
editor.onPaste(exports.killRing.rotate());
393
editor.keyBinding.$data.lastCommand = "yank";
395
killRegion: function(editor) {
396
exports.killRing.add(editor.getCopyText());
397
editor.commands.byName.cut.exec(editor);
399
killRingSave: function(editor) {
400
exports.killRing.add(editor.getCopyText());
402
keyboardQuit: function(editor) {
403
editor.selection.clearSelection();
404
exports.setMarkMode(null);
408
var commands = exports.handler.commands;
409
commands.yank.isYank = true;
410
commands.yankRotate.isYank = true;
415
str && this.$data.push(str);
416
if (this.$data.length > 30)
420
return this.$data[this.$data.length - 1] || "";
423
if (this.$data.length > 1)
428
this.$data.unshift(this.$data.pop());