-
-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathbetter-editor.js
119 lines (108 loc) · 4.38 KB
/
better-editor.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
/*
Allows for better code formatting in text editor
Adds tabs spacing
*/
jQuery(document).ready(function($) {
var tab = " ";
function checkTab(evt) {
var t = evt.target;
var ss = t.selectionStart;
var se = t.selectionEnd;
var key = evt.keyCode || evt.charCode || 0;
var commandPressed = evt.shiftKey;
var numberOfLines = t.value.slice(ss, se).split('\n');
// Shif + Tab - delete tab insertion at the beginning of the line
if (evt.keyCode === 9 && commandPressed) {
evt.preventDefault();
var pre = t.value.slice(0, ss);
// Redefine ss and pre, so it gets the complete initial line as part of the selection
ss = pre.lastIndexOf("\n") + 1;
pre = t.value.slice(0, ss);
var sel = t.value.slice(ss, se).split('\n');
var post = t.value.slice(se, t.value.length);
var newSel = [];
var count = 0;
for (var i = 0; i < sel.length; i++) {
if (sel[i].match(tab)) {
count++;
}
newSel.push(sel[i].replace(tab, ''));
}
var updatedContent = newSel.join('\n');
t.value = pre.concat(updatedContent).concat(post);
t.selectionStart = ss;
t.selectionEnd = se - (tab.length * count);
// This prevents any default behavior. preventDefault() is not enough
return false;
}
// Tab key - insert tab expansion
if (evt.keyCode === 9) {
evt.preventDefault();
// Special case of multi line selection
if (ss != se && t.value.slice(ss, se).indexOf("\n") != -1) {
var pre = t.value.slice(0, ss);
// redefine ss and pre, so it gets the complete initial line as part of the selection
ss = pre.lastIndexOf("\n") + 1;
pre = t.value.slice(0, ss);
var sel = t.value.slice(ss, se).replace(/\n/g, "\n" + tab);
var post = t.value.slice(se, t.value.length);
t.value = pre.concat(tab).concat(sel).concat(post);
t.selectionStart = ss;
t.selectionEnd = se + (tab.length * (numberOfLines.length));
} else {
// "Normal" case (no selection or selection on one line only)
t.value = t.value.slice(0, ss).concat(tab).concat(t.value.slice(ss, t.value.length));
if (ss === se) {
t.selectionStart = t.selectionEnd = ss + tab.length;
} else {
t.selectionStart = ss + tab.length;
t.selectionEnd = se + tab.length;
}
}
} else if (evt.keyCode === 8 && t.value.slice(ss - 2, ss) === tab) {
// Backspace key - delete preceding tab expansion, if exists
evt.preventDefault();
t.value = t.value.slice(0, ss - 2).concat(t.value.slice(ss, t.value.length));
t.selectionStart = t.selectionEnd = ss - tab.length;
} else if (evt.keyCode === 46 && t.value.slice(se, se + 2) === tab) {
// Delete key - delete following tab expansion, if exists
evt.preventDefault();
t.value = t.value.slice(0, ss).concat(t.value.slice(ss + 2, t.value.length));
t.selectionStart = t.selectionEnd = ss;
} else if (evt.keyCode === 37 && t.value.slice(ss - 2, ss) === tab) {
// Left arrow keys - move across the tab in one go
evt.preventDefault();
t.selectionStart = t.selectionEnd = ss - 2;
} else if (evt.keyCode === 39 && t.value.slice(ss, ss + 2) === tab) {
// Right arrow keys - move across the tab in one go
evt.preventDefault();
t.selectionStart = t.selectionEnd = ss + 2;
}
}
var textarea = document.getElementsByTagName('textarea')[0];
textarea.onkeydown = checkTab;
});
function getCaretPosition(editableDiv) {
var caretPos = 0,
sel, range;
if (window.getSelection) {
sel = window.getSelection();
if (sel.rangeCount) {
range = sel.getRangeAt(0);
if (range.commonAncestorContainer.parentNode == editableDiv) {
caretPos = range.endOffset;
}
}
} else if (document.selection && document.selection.createRange) {
range = document.selection.createRange();
if (range.parentElement() == editableDiv) {
var tempEl = document.createElement("span");
editableDiv.insertBefore(tempEl, editableDiv.firstChild);
var tempRange = range.duplicate();
tempRange.moveToElementText(tempEl);
tempRange.setEndPoint("EndToEnd", range);
caretPos = tempRange.text.length;
}
}
return caretPos;
}