【发布时间】:2017-07-26 16:27:03
【问题描述】:
我有这个功能可以在你点击文本时获取光标的位置,它只适用于等宽字符,这很好,但它显然不适用于像中文或日文这样更宽的字符。
function get_char_pos(point) {
var prompt_len = self.find('.prompt').text().length;
var size = get_char_size();
var width = size.width;
var height = size.height;
var offset = self.offset();
var col = Math.floor((point.x - offset.left) / width);
var row = Math.floor((point.y - offset.top) / height);
var lines = get_splited_command_line(command);
var try_pos;
if (row > 0 && lines.length > 1) {
try_pos = col + lines.slice(0, row).reduce(function(sum, line) {
return sum + line.length;
}, 0);
} else {
try_pos = col - prompt_len;
}
// tabs are 4 spaces and newline don't show up in results
var text = command.replace(/\t/g, '\x00\x00\x00\x00').replace(/\n/, '');
var before = text.slice(0, try_pos);
var len = before.replace(/\x00{4}/g, '\t').replace(/\x00+/, '').length;
return len > command.length ? command.length : len;
}
我尝试使用 wcwidth 库创建一个函数(对于更宽的字符返回 2,对于普通字母返回 1),但它不能正常工作,这是带有演示的代码:
var self = $('pre');
var offset = self.offset();
var command = 'チトシタテイトチトシイスチトシタテイトチトシイスチトシタテイトチトシイス\nfoo bar baz\nfoo bar baz\nチトシタテイトチトシイ';
self.html(command);
function get_char_size() {
var span = $('<span> </span>').appendTo(self);
var rect = span[0].getBoundingClientRect();
span.remove();
return rect;
}
var length = wcwidth;
// mock
function get_splited_command_line(string) {
return string.split('\n');
}
function get_char_pos(point) {
var size = get_char_size();
var width = size.width;
var height = size.height;
var offset = self.offset();
var col_count = Math.floor((point.x - offset.left) / width);
var row = Math.floor((point.y - offset.top) / height);
var lines = get_splited_command_line(command);
var line = lines[row];
var col = 0;
var i = col_count;
while (i > 0) {
i -= length(line[col]);
col++;
}
var try_pos;
if (row > 0 && lines.length > 1) {
try_pos = col + lines.slice(0, row).reduce(function(sum, line) {
return sum + length(line);
}, 0);
} else {
try_pos = col;
}
// tabs are 4 spaces and newline don't show up in results
var text = command.replace(/\t/g, '\x00\x00\x00\x00').replace(/\n/, '');
var before = text.slice(0, try_pos);
var len = before.replace(/\x00{4}/g, '\t').replace(/\x00+/, '').length;
var command_len = command.length;
return len > command_len ? command_len : len;
}
self.click(function(e) {
var pos = get_char_pos({
x: e.pageX,
y: e.pageY
});
self.html(command.substring(0, pos-1) + '<span>' +
command[pos] + '</span>' +
command.substring(pos+1));
});
span {
color: red;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<script src="https://rawgit.com/jcubic/leash/master/lib/wcwidth.js"></script>
<pre></pre>
我不能对每个字母使用跨度,因为我将文本分成了 3 个跨度,位于光标之前、光标和光标之后。而且我也有可能在容器中或不在容器中的样式跨度。
感谢任何有关如何修复此功能的帮助。
解决方案:
这是我基于@Will 的代码,我在我的代码中使用了多个元素(由于某种原因,当您单击只有一个字符的元素时,chrome 会出现问题,以防万一焦点不在超过文本的长度):
function get_focus_offset() {
var sel;
if ((sel = window.getSelection()) && (sel.focusNode !== null)) {
return sel.focusOffset;
}
}
function get_char_pos(e) {
var focus = get_focus_offset();
if ($.isNumeric(focus)) {
var node = $(e.target);
// [role="presentation"] is my direct children that have
// siblings that are other nodes with text
var parent = node.closest('[role="presentation"]');
var len = node.text().length;
focus = len === 1 ? 0 : Math.min(focus, len);
return focus + parent.prevUntil('.prompt').text_length() +
node.prevAll().text_length();
} else {
return command.length;
}
}
更新:如果你点击前半部分,它会被正确选择,但是当你点击另一半时,它会选择下一个字符,所以我最终选择了每个元素一个字符的方法.
【问题讨论】:
-
这个的用例是什么?
-
你在看多少个字符?将每个包裹在自己的跨度中是否可行?
-
@charlietfl 我需要这个让 jQuery 终端在点击时移动光标。我已经有了这个工作,但它不适用于更广泛的角色。
-
@J.Chen 这会很困难,因为我还需要在容器中具有颜色和样式的跨度,并且很难区分具有跨度为字母的样式的跨度。
-
顺便问一下,您需要获取有关字符样式的任何信息吗?(颜色、大小、重量等)
标签: javascript jquery