From e851ea232a114902ea6a8e5cc8f7d34d07969c42 Mon Sep 17 00:00:00 2001 From: Yoshioka Tsuneo Date: Mon, 9 Nov 2015 16:40:04 +0900 Subject: [PATCH] CJK support, Copy and Paste support --- src/term.js | 411 +++++++++++++++++++++++++++++++++++++++++++--------- 1 file changed, 345 insertions(+), 66 deletions(-) diff --git a/src/term.js b/src/term.js index 1b4e93d..232ad0d 100644 --- a/src/term.js +++ b/src/term.js @@ -487,6 +487,10 @@ each(keys(Terminal.defaults), function(key) { Terminal.focus = null; Terminal.prototype.focus = function() { + if (this._textarea) { + this._textarea.focus(); + } + if (Terminal.focus === this) return; if (Terminal.focus) { @@ -540,9 +544,7 @@ Terminal.prototype.initGlobal = function() { Terminal.bindKeys(document); - if (this.isMobile) { - this.fixMobile(document); - } + Terminal.bindCopy(document); if (this.useStyle) { Terminal.insertStyle(document, this.colors[256], this.colors[257]); @@ -560,6 +562,7 @@ Terminal.bindPaste = function(document) { on(window, 'paste', function(ev) { var term = Terminal.focus; if (!term) return; + if (term._textarea) return; if (ev.clipboardData) { term.send(ev.clipboardData.getData('text/plain')); } else if (term.context.clipboardData) { @@ -586,7 +589,7 @@ Terminal.bindKeys = function(document) { || target === Terminal.focus.context || target === Terminal.focus.document || target === Terminal.focus.body - || target === Terminal._textarea + || target === Terminal.focus._textarea || target === Terminal.focus.parent) { return Terminal.focus.keyDown(ev); } @@ -600,7 +603,7 @@ Terminal.bindKeys = function(document) { || target === Terminal.focus.context || target === Terminal.focus.document || target === Terminal.focus.body - || target === Terminal._textarea + || target === Terminal.focus._textarea || target === Terminal.focus.parent) { return Terminal.focus.keyPress(ev); } @@ -613,6 +616,8 @@ Terminal.bindKeys = function(document) { var el = ev.target || ev.srcElement; if (!el) return; + if (!el.parentNode) return; + if (!el.parentNode.parentNode) return; do { if (el === Terminal.focus.element) return; @@ -671,31 +676,68 @@ Terminal.bindCopy = function(document) { * Fix Mobile */ -Terminal.prototype.fixMobile = function(document) { +Terminal.prototype.getTextarea = function(document) { var self = this; var textarea = document.createElement('textarea'); textarea.style.position = 'absolute'; textarea.style.left = '-32000px'; textarea.style.top = '-32000px'; - textarea.style.width = '0px'; - textarea.style.height = '0px'; + textarea.style.width = '100em'; + textarea.style.height = '2em'; + textarea.style.padding = '0'; textarea.style.opacity = '0'; + textarea.style.color = 'inherit'; + textarea.style.font = 'inherit'; + textarea.style.textIndent = '-1em'; /* Hide text cursor on IE */ textarea.style.backgroundColor = 'transparent'; textarea.style.borderStyle = 'none'; textarea.style.outlineStyle = 'none'; textarea.autocapitalize = 'none'; textarea.autocorrect = 'off'; - document.getElementsByTagName('body')[0].appendChild(textarea); + var onInputTimestamp; - Terminal._textarea = textarea; + var onInput = function(ev){ + if(ev.timeStamp && ev.timeStamp === onInputTimestamp){ + return; + } + onInputTimestamp = ev.timeStamp; - setTimeout(function() { - textarea.focus(); - }, 1000); + var value = textarea.textContent || textarea.value; + if (typeof self.select.startPos !== 'undefined'){ + self.select = {}; + self.clearSelectedText(); + self.refresh(0, this.rows - 1); + } + if (!self.compositionStatus) { + textarea.value = ''; + textarea.textContent = ''; + self.send(value); + } + }; - if (this.isAndroid) { + on(textarea, 'compositionstart', function() { + textarea.style.opacity = "1.0"; + textarea.style.textIndent = "0"; + self.compositionStatus = true; + }); + on(textarea, 'compositionend', function(ev) { + textarea.style.opacity = "0.0"; + textarea.style.textIndent = "-1em"; + self.compositionStatus = false; + setTimeout(function(){ + onInput(ev); // for IE that does not trigger 'input' after the IME composition. + }, 1); + }); + + on(textarea, 'keydown', function(){ + var value = textarea.textContent || textarea.value; + }); + + on(textarea, 'input', onInput); + + if (Terminal.isAndroid) { on(textarea, 'change', function() { var value = textarea.textContent || textarea.value; textarea.value = ''; @@ -703,6 +745,7 @@ Terminal.prototype.fixMobile = function(document) { self.send(value + '\r'); }); } + return textarea; }; /** @@ -797,11 +840,154 @@ Terminal.prototype.open = function(parent) { this.element.appendChild(div); this.children.push(div); } + + this._textarea = this.getTextarea(this.document); + this.element.appendChild(this._textarea); + this.parent.appendChild(this.element); + this.select = {}; + // Draw the screen. this.refresh(0, this.rows - 1); + + var updateSelect = function(){ + var startPos = self.select.startPos; + var endPos = self.select.endPos; + + if(endPos.y < startPos.y || (startPos.y == endPos.y && endPos.x < startPos.x)){ + var tmp = startPos; + startPos = endPos; + endPos = tmp; + } + if (self.select.clicks === 2){ + var j = i; + var isMark = function(ch){ + var code = ch.charCodeAt(0); + return (code <= 0x2f) || (0x3a <= code && code <= 0x40) || (0x5b <= code && code < 0x60) || (0x7b <= code && code <= 0x7f); + } + while (startPos.x > 0 && !isMark(self.lines[startPos.y][startPos.x-1][1])){ + startPos.x--; + } + while (endPos.x < self.cols && !isMark(self.lines[endPos.y][endPos.x][1])){ + endPos.x++; + } + }else if(self.select.clicks === 3){ + startPos.x = 0; + endPos.y ++; + endPos.x = 0; + } + + if (startPos.x === endPos.x && startPos.y === endPos.y){ + self.clearSelectedText(); + }else{ + var x2 = self.select.endPos.x; + var y2 = self.select.endPos.y; + x2 --; + if(x2<0){ + y2--; + x2 = self.cols - 1; + } + self.selectText(self.select.startPos.x, x2, self.select.startPos.y, y2); + } + }; + var copySelectToTextarea = function (){ + var textarea = self._textarea; + if (textarea) { + + if (self.select.startPos.x === self.select.endPos.x && self.select.startPos.y === self.select.endPos.y){ + textarea.value = ""; + textarea.select(); + return; + } + + var x2 = self.select.endPos.x; + var y2 = self.select.endPos.y; + x2 --; + if(x2<0){ + y2--; + x2 = self.cols - 1; + } + + var value = self.grabText(self.select.startPos.x, x2, self.select.startPos.y, y2); + textarea.value = value; + textarea.select(); + } + }; + on(this.element, 'mousedown', function(ev) { + + if(ev.button === 2){ + + var r = self.element.getBoundingClientRect(); + + var x = ev.pageX - r.left + self.element.offsetLeft; + var y = ev.pageY - r.top + self.element.offsetTop; + + self._textarea.style.left = x + 'px'; + self._textarea.style.top = y + 'px'; + return; + } + + if (ev.button != 0){ + return; + } + if (navigator.userAgent.indexOf("Trident")){ + /* IE does not hold click number as "detail" property. */ + if (self.select.timer){ + self.select.clicks ++; + clearTimeout(self.select.timer); + self.select.timer = null; + }else{ + self.select.clicks = 1; + } + self.select.timer = setTimeout(function(){ + self.select.timer = null; + }, 600); + }else{ + self.select.clicks = ev.detail; + } + + if (! ev.shiftKey){ + self.clearSelectedText(); + + self.select.startPos = self.getCoords(ev); + self.select.startPos.y += self.ydisp; + } + self.select.endPos = self.getCoords(ev); + self.select.endPos.y += self.ydisp; + updateSelect(); + copySelectToTextarea(); + self.refresh(0, self.rows - 1); + self.select.selecting = true; + }); + on(this.element, 'mousemove', function(ev) { + if(self.select.selecting){ + self.select.endPos = self.getCoords(ev); + self.select.endPos.y += self.ydisp; + updateSelect(); + self.refresh(0, self.rows - 1); + } + }); + on(document, 'mouseup', function(ev) { + if(ev.button === 2){ + + var r = self.element.getBoundingClientRect(); + + var x = ev.pageX - r.left + self.element.offsetLeft; + var y = ev.pageY - r.top + self.element.offsetTop; + + self._textarea.style.left = x - 1 + 'px'; + self._textarea.style.top = y - 1 + 'px'; + return; + } + if(self.select.selecting){ + self.select.selecting = false; + copySelectToTextarea(); + } + }); + + if (!('useEvents' in this.options) || this.options.useEvents) { // Initialize global actions that // need to be taken on the document. @@ -819,9 +1005,6 @@ Terminal.prototype.open = function(parent) { // to focus and paste behavior. on(this.element, 'focus', function() { self.focus(); - if (self.isMobile) { - Terminal._textarea.focus(); - } }); // This causes slightly funky behavior. @@ -871,6 +1054,7 @@ Terminal.prototype.open = function(parent) { // as well as the iPad fix. setTimeout(function() { self.element.focus(); + self.focus(); }, 100); } @@ -887,6 +1071,55 @@ Terminal.prototype.setRawMode = function(value) { this.isRaw = !!value; }; +Terminal.prototype.getCoords = function(ev) { + var x, y, w, h, el; + + var self = this; + + // ignore browsers without pageX for now + if (ev.pageX == null) return; + + x = ev.pageX; + y = ev.pageY; + el = self.element; + + x -= el.clientLeft; + y -= el.clientTop; + + // should probably check offsetParent + // but this is more portable + while (el && el !== self.document.documentElement) { + x -= el.offsetLeft; + y -= el.offsetTop; + el = 'offsetParent' in el + ? el.offsetParent + : el.parentNode; + } + + // convert to cols/rows + w = self.element.clientWidth; + h = self.element.clientHeight; + var cols = Math.floor((x / w) * self.cols); + var rows = Math.floor((y / h) * self.rows); + + // be sure to avoid sending + // bad positions to the program + if (cols < 0) cols = 0; + if (cols > self.cols) cols = self.cols; + if (rows < 0) rows = 0; + if (rows > self.rows) rows = self.rows; + + // xterm sends raw bytes and + // starts at 32 (SP) for each. + //x += 32; + //y += 32; + + return { + x: cols, + y: rows, + }; +} + // XTerm mouse events // http://invisible-island.net/xterm/ctlseqs/ctlseqs.html#Mouse%20Tracking // To better understand these @@ -1299,7 +1532,12 @@ Terminal.prototype.refresh = function(start, end) { , row , parent; + var characterWidth = this.element.clientWidth / this.cols; + var characterHeight = this.element.clientHeight / this.rows; + var focused; + if (end - start >= this.rows / 2) { + focused = (Terminal.focus == this); parent = this.element.parentNode; if (parent) parent.removeChild(this.element); } @@ -1424,8 +1662,12 @@ Terminal.prototype.refresh = function(start, end) { if (ch <= ' ') { out += ' '; } else { - if (isWide(ch)) i++; - out += ch; + if (isWide(ch)) { + i++; + out += '' + ch + ''; + } else { + out += ch; + } } break; } @@ -1444,11 +1686,23 @@ Terminal.prototype.refresh = function(start, end) { this.children[y].innerHTML = out; } - if (parent) parent.appendChild(this.element); - if (start !== end && Terminal.focus == this) - { - this.element.focus(); + if (parent) { + parent.appendChild(this.element); + if (focused) { + this.focus(); + } + } + + if (this._textarea) { + var cursorElement = this.element.querySelector('.terminal-cursor'); + if(cursorElement){ + var cursor_x = cursorElement.offsetLeft; + var cursor_y = cursorElement.offsetTop; + this._textarea.style.left = cursor_x + 'px'; + this._textarea.style.top = cursor_y + 'px'; + } } + }; Terminal.prototype._cursorBlink = function() { @@ -1618,15 +1872,7 @@ Terminal.prototype.write = function(data) { // '\b' case '\x08': if (this.x > 0) { - prev = this.lines[this.y][this.x - 1]; - if(isEastAsian(prev[1])) - { - prev[1] = ''; - } - else - { - this.x--; - } + this.x--; } break; @@ -2923,6 +3169,9 @@ Terminal.prototype.setgCharset = function(g, charset) { Terminal.prototype.keyPress = function(ev) { var key; + if (this._textarea) { + return; + } cancel(ev); @@ -5051,18 +5300,18 @@ Terminal.prototype.copyText = function(text) { }, 1); }; -Terminal.prototype.selectText = function(x1, x2, y1, y2) { - var ox1 - , ox2 - , oy1 - , oy2 - , tmp - , x - , y - , xl - , attr; - +Terminal.prototype.clearSelectedText = function() { if (this._selected) { + var ox1 + , ox2 + , oy1 + , oy2 + , tmp + , x + , y + , xl + , attr; + ox1 = this._selected.x1; ox2 = this._selected.x2; oy1 = this._selected.y1; @@ -5102,9 +5351,22 @@ Terminal.prototype.selectText = function(x1, x2, y1, y2) { } } } + delete this._selected; + } +}; + +Terminal.prototype.selectText = function(x1, x2, y1, y2) { + var tmp + , x + , y + , xl + , attr; + + if (this._selected) { y1 = this._selected.y1; x1 = this._selected.x1; + this.clearSelectedText(); } y1 = Math.max(y1, 0); @@ -5964,30 +6226,47 @@ function indexOf(obj, el) { return -1; } -function isWide(ch) { - ch = ch.charCodeAt(0); - if (ch == 0x02587) return true; - if (ch <= 0x0ff00) return false; - return (ch >= 0x0ff01 && ch <= 0x0ffbe) - || (ch >= 0x0ffc2 && ch <= 0x0ffc7) - || (ch >= 0x0ffca && ch <= 0x0ffcf) - || (ch >= 0x0ffd2 && ch <= 0x0ffd7) - || (ch >= 0x0ffda && ch <= 0x0ffdc) - || (ch >= 0x0ffe0 && ch <= 0x0ffe6) - || (ch >= 0x0ffe8 && ch <= 0x0ffee); -} +/* Ref: https://github.com/ajaxorg/ace/blob/0c66e1eda418477a9efbd0d3ef61698478cc607f/lib/ace/edit_session.js#L2434 */ +function isFullWidth(c) { + if (c < 0x1100) + return false; + return c >= 0x1100 && c <= 0x115F || + c >= 0x11A3 && c <= 0x11A7 || + c >= 0x11FA && c <= 0x11FF || + c >= 0x2329 && c <= 0x232A || + c >= 0x2E80 && c <= 0x2E99 || + c >= 0x2E9B && c <= 0x2EF3 || + c >= 0x2F00 && c <= 0x2FD5 || + c >= 0x2FF0 && c <= 0x2FFB || + c >= 0x3000 && c <= 0x303E || + c >= 0x3041 && c <= 0x3096 || + c >= 0x3099 && c <= 0x30FF || + c >= 0x3105 && c <= 0x312D || + c >= 0x3131 && c <= 0x318E || + c >= 0x3190 && c <= 0x31BA || + c >= 0x31C0 && c <= 0x31E3 || + c >= 0x31F0 && c <= 0x321E || + c >= 0x3220 && c <= 0x3247 || + c >= 0x3250 && c <= 0x32FE || + c >= 0x3300 && c <= 0x4DBF || + c >= 0x4E00 && c <= 0xA48C || + c >= 0xA490 && c <= 0xA4C6 || + c >= 0xA960 && c <= 0xA97C || + c >= 0xAC00 && c <= 0xD7A3 || + c >= 0xD7B0 && c <= 0xD7C6 || + c >= 0xD7CB && c <= 0xD7FB || + c >= 0xF900 && c <= 0xFAFF || + c >= 0xFE10 && c <= 0xFE19 || + c >= 0xFE30 && c <= 0xFE52 || + c >= 0xFE54 && c <= 0xFE66 || + c >= 0xFE68 && c <= 0xFE6B || + c >= 0xFF01 && c <= 0xFF60 || + c >= 0xFFE0 && c <= 0xFFE6; +}; -function isEastAsian(ch) { - ch = ch.charCodeAt(0); - return (ch >= 0x03000 && ch <= 0x030FF) - || (ch >= 0x031F0 && ch <= 0x031FF) - || (ch >= 0x03300 && ch <= 0x04DFF) - || (ch >= 0x04E00 && ch <= 0x09FFF) - || (ch >= 0x0F900 && ch <= 0x0FAFF) - || (ch >= 0x0FF00 && ch <= 0x0FFEF) - || (ch >= 0x20000 && ch <= 0x2A6DF) - || (ch >= 0x2A700 && ch <= 0x2B734) - || (ch >= 0x2F800 && ch <= 0x2FA1F) +function isWide(ch) { + var c = ch.charCodeAt(0); + return isFullWidth(c); } function matchColor(r1, g1, b1) {