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());