You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
252 lines
9.3 KiB
JavaScript
252 lines
9.3 KiB
JavaScript
11 years ago
|
tabIndent = {
|
||
|
version: '0.1.8',
|
||
|
config: {
|
||
|
tab: '\t'
|
||
|
},
|
||
|
events: {
|
||
|
keydown: function(e) {
|
||
|
var tab = tabIndent.config.tab;
|
||
|
var tabWidth = tab.length;
|
||
|
if (e.keyCode === 9) {
|
||
|
e.preventDefault();
|
||
|
var currentStart = this.selectionStart,
|
||
|
currentEnd = this.selectionEnd;
|
||
|
if (e.shiftKey === false) {
|
||
|
// Normal Tab Behaviour
|
||
|
if (!tabIndent.isMultiLine(this)) {
|
||
|
// Add tab before selection, maintain highlighted text selection
|
||
|
this.value = this.value.slice(0, currentStart) + tab + this.value.slice(currentStart);
|
||
|
this.selectionStart = currentStart + tabWidth;
|
||
|
this.selectionEnd = currentEnd + tabWidth;
|
||
|
} else {
|
||
|
// Iterating through the startIndices, if the index falls within selectionStart and selectionEnd, indent it there.
|
||
|
var startIndices = tabIndent.findStartIndices(this),
|
||
|
l = startIndices.length,
|
||
|
newStart = undefined,
|
||
|
newEnd = undefined,
|
||
|
affectedRows = 0;
|
||
|
|
||
|
while(l--) {
|
||
|
var lowerBound = startIndices[l];
|
||
|
if (startIndices[l+1] && currentStart != startIndices[l+1]) lowerBound = startIndices[l+1];
|
||
|
|
||
|
if (lowerBound >= currentStart && startIndices[l] < currentEnd) {
|
||
|
this.value = this.value.slice(0, startIndices[l]) + tab + this.value.slice(startIndices[l]);
|
||
|
|
||
|
newStart = startIndices[l];
|
||
|
if (!newEnd) newEnd = (startIndices[l+1] ? startIndices[l+1] - 1 : 'end');
|
||
|
affectedRows++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.selectionStart = newStart;
|
||
|
this.selectionEnd = (newEnd !== 'end' ? newEnd + (tabWidth * affectedRows) : this.value.length);
|
||
|
}
|
||
|
} else {
|
||
|
// Shift-Tab Behaviour
|
||
|
if (!tabIndent.isMultiLine(this)) {
|
||
|
if (this.value.substr(currentStart - tabWidth, tabWidth) == tab) {
|
||
|
// If there's a tab before the selectionStart, remove it
|
||
|
this.value = this.value.substr(0, currentStart - tabWidth) + this.value.substr(currentStart);
|
||
|
this.selectionStart = currentStart - tabWidth;
|
||
|
this.selectionEnd = currentEnd - tabWidth;
|
||
|
} else if (this.value.substr(currentStart - 1, 1) == "\n" && this.value.substr(currentStart, tabWidth) == tab) {
|
||
|
// However, if the selection is at the start of the line, and the first character is a tab, remove it
|
||
|
this.value = this.value.substring(0, currentStart) + this.value.substr(currentStart + tabWidth);
|
||
|
this.selectionStart = currentStart;
|
||
|
this.selectionEnd = currentEnd - tabWidth;
|
||
|
}
|
||
|
} else {
|
||
|
// Iterating through the startIndices, if the index falls within selectionStart and selectionEnd, remove an indent from that row
|
||
|
var startIndices = tabIndent.findStartIndices(this),
|
||
|
l = startIndices.length,
|
||
|
newStart = undefined,
|
||
|
newEnd = undefined,
|
||
|
affectedRows = 0;
|
||
|
|
||
|
while(l--) {
|
||
|
var lowerBound = startIndices[l];
|
||
|
if (startIndices[l+1] && currentStart != startIndices[l+1]) lowerBound = startIndices[l+1];
|
||
|
|
||
|
if (lowerBound >= currentStart && startIndices[l] < currentEnd) {
|
||
|
if (this.value.substr(startIndices[l], tabWidth) == tab) {
|
||
|
// Remove a tab
|
||
|
this.value = this.value.slice(0, startIndices[l]) + this.value.slice(startIndices[l] + tabWidth);
|
||
|
affectedRows++;
|
||
|
} else {} // Do nothing
|
||
|
|
||
|
newStart = startIndices[l];
|
||
|
if (!newEnd) newEnd = (startIndices[l+1] ? startIndices[l+1] - 1 : 'end');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.selectionStart = newStart;
|
||
|
this.selectionEnd = (newEnd !== 'end' ? newEnd - (affectedRows * tabWidth) : this.value.length);
|
||
|
}
|
||
|
}
|
||
|
} else if (e.keyCode === 27) { // Esc
|
||
|
tabIndent.events.disable(e);
|
||
|
} else if (e.keyCode === 13 && e.shiftKey === false) { // Enter
|
||
|
var self = tabIndent,
|
||
|
cursorPos = this.selectionStart,
|
||
|
startIndices = self.findStartIndices(this),
|
||
|
numStartIndices = startIndices.length,
|
||
|
startIndex = 0,
|
||
|
endIndex = 0,
|
||
|
tabMatch = new RegExp("^" + tab.replace('\t', '\\t').replace(/ /g, '\\s') + "+", 'g'),
|
||
|
lineText = '';
|
||
|
tabs = null;
|
||
|
|
||
|
for(var x=0;x<numStartIndices;x++) {
|
||
|
if (startIndices[x+1] && (cursorPos >= startIndices[x]) && (cursorPos < startIndices[x+1])) {
|
||
|
startIndex = startIndices[x];
|
||
|
endIndex = startIndices[x+1] - 1;
|
||
|
break;
|
||
|
} else {
|
||
|
startIndex = startIndices[numStartIndices-1];
|
||
|
endIndex = this.value.length;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
lineText = this.value.slice(startIndex, endIndex);
|
||
|
tabs = lineText.match(tabMatch);
|
||
|
if (tabs !== null) {
|
||
|
e.preventDefault();
|
||
|
var indentText = tabs[0];
|
||
|
var indentWidth = indentText.length;
|
||
|
var inLinePos = cursorPos - startIndex;
|
||
|
if (indentWidth > inLinePos) {
|
||
|
indentWidth = inLinePos;
|
||
|
indentText = indentText.slice(0, inLinePos);
|
||
|
}
|
||
|
|
||
|
this.value = this.value.slice(0, cursorPos) + "\n" + indentText + this.value.slice(cursorPos);
|
||
|
this.selectionStart = cursorPos + indentWidth + 1;
|
||
|
this.selectionEnd = this.selectionStart;
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
disable: function(e) {
|
||
|
var events = this;
|
||
|
|
||
|
// Temporarily suspend the main tabIndent event
|
||
|
tabIndent.remove(e.target);
|
||
|
},
|
||
|
focus: function() {
|
||
|
var self = tabIndent,
|
||
|
el = this,
|
||
|
delayedRefocus = setTimeout(function() {
|
||
|
var classes = (el.getAttribute('class') || '').split(' '),
|
||
|
contains = classes.indexOf('tabIndent');
|
||
|
|
||
|
el.addEventListener('keydown', self.events.keydown);
|
||
|
el.style.backgroundImage = "url()";
|
||
|
el.style.backgroundPosition = 'top right';
|
||
|
el.style.backgroundRepeat = 'no-repeat';
|
||
|
|
||
|
if (contains !== -1) classes.splice(contains, 1);
|
||
|
classes.push('tabIndent-rendered');
|
||
|
el.setAttribute('class', classes.join(' '));
|
||
|
|
||
|
el.removeEventListener('focus', self.events.keydown);
|
||
|
}, 500);
|
||
|
|
||
|
// If they were just tabbing through the input, let them continue unimpeded
|
||
|
el.addEventListener('blur', function b() {
|
||
|
clearTimeout(delayedRefocus);
|
||
|
el.removeEventListener('blur', b);
|
||
|
});
|
||
|
}
|
||
|
},
|
||
|
render: function(el) {
|
||
|
var self = this;
|
||
|
|
||
|
if (el.nodeName === 'TEXTAREA') {
|
||
|
el.addEventListener('focus', self.events.focus);
|
||
|
|
||
|
el.addEventListener('blur', function b(e) {
|
||
|
self.events.disable(e);
|
||
|
});
|
||
|
}
|
||
|
},
|
||
|
renderAll: function() {
|
||
|
// Find all elements with the tabIndent class
|
||
|
var textareas = document.getElementsByTagName('textarea'),
|
||
|
t = textareas.length,
|
||
|
contains = -1,
|
||
|
classes = [],
|
||
|
el = undefined;
|
||
|
|
||
|
while(t--) {
|
||
|
classes = (textareas[t].getAttribute('class') || '').split(' ');
|
||
|
contains = classes.indexOf('tabIndent');
|
||
|
|
||
|
if (contains !== -1) {
|
||
|
el = textareas[t];
|
||
|
this.render(el);
|
||
|
}
|
||
|
contains = -1;
|
||
|
classes = [];
|
||
|
el = undefined;
|
||
|
}
|
||
|
},
|
||
|
remove: function(el) {
|
||
|
if (el.nodeName === 'TEXTAREA') {
|
||
|
var classes = (el.getAttribute('class') || '').split(' '),
|
||
|
contains = classes.indexOf('tabIndent-rendered');
|
||
|
|
||
|
if (contains !== -1) {
|
||
|
el.removeEventListener('keydown', this.events.keydown);
|
||
|
el.style.backgroundImage = '';
|
||
|
|
||
|
classes.splice(contains, 1);
|
||
|
classes.push('tabIndent');
|
||
|
el.setAttribute('class', (classes.length > 1 ? classes.join(' ') : classes[0]));
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
removeAll: function() {
|
||
|
// Find all elements with the tabIndent class
|
||
|
var textareas = document.getElementsByTagName('textarea'),
|
||
|
t = textareas.length,
|
||
|
contains = -1,
|
||
|
classes = [],
|
||
|
el = undefined;
|
||
|
|
||
|
while(t--) {
|
||
|
classes = (textareas[t].getAttribute('class') || '').split(' ');
|
||
|
contains = classes.indexOf('tabIndent-rendered');
|
||
|
|
||
|
if (contains !== -1) {
|
||
|
el = textareas[t];
|
||
|
this.remove(el);
|
||
|
}
|
||
|
contains = -1;
|
||
|
classes = [];
|
||
|
el = undefined;
|
||
|
}
|
||
|
},
|
||
|
isMultiLine: function(el) {
|
||
|
// Extract the selection
|
||
|
var snippet = el.value.slice(el.selectionStart, el.selectionEnd),
|
||
|
nlRegex = new RegExp(/\n/);
|
||
|
|
||
|
if (nlRegex.test(snippet)) return true;
|
||
|
else return false;
|
||
|
},
|
||
|
findStartIndices: function(el) {
|
||
|
var text = el.value,
|
||
|
startIndices = [],
|
||
|
offset = 0;
|
||
|
|
||
|
while(text.match(/\n/) && text.match(/\n/).length > 0) {
|
||
|
offset = (startIndices.length > 0 ? startIndices[startIndices.length - 1] : 0);
|
||
|
var lineEnd = text.search("\n");
|
||
|
startIndices.push(lineEnd + offset + 1);
|
||
|
text = text.substring(lineEnd + 1);
|
||
|
}
|
||
|
startIndices.unshift(0);
|
||
|
|
||
|
return startIndices;
|
||
|
}
|
||
|
}
|