【问题标题】:How to get character position when click on text in javascript单击javascript中的文本时如何获取字符位置
【发布时间】: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>&nbsp;</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


【解决方案1】:

window.addEventListener('DOMContentLoaded', () => {
    document.querySelectorAll('.charPosition').forEach(el => {
        let characters = el['innerText'].split('');
        el.innerHTML = '';
        characters.forEach(char => {
            let span = document.createElement('span');
            span.innerText = char;
            span.addEventListener('click', function () {
                let position = 0;
                let el = this;
                while (el.previousSibling !== null) {
                    position++;
                    el = el.previousSibling;
                }
                console.log(this.innerHTML + ':' + position);
            });
            el.appendChild(span);
        });
    });
});
&lt;div class="charPosition"&gt;Lorem ipsum dolor sit amet, consectetur adipiscing elit.&lt;/div&gt;

【讨论】:

  • 感谢您的解决方案,但我不能使用跨度,因为我的容器中可能有具有样式的跨度,而且我将文本分为由光标划分的两部分,抱歉没有提到那个。
  • 我最终得到了这个解决方案。
【解决方案2】:

第三次尝试。在那里塞一个管道字符来伪装成一个光标。

https://developer.mozilla.org/en-US/docs/Web/API/Selection

window.addEventListener('DOMContentLoaded', () => {
document.querySelectorAll('.charPosition').forEach(el => {
    let clean, cursor;
    el.addEventListener('click', e => {
        let position = window.getSelection().focusOffset;
        if (cursor && position > cursor)
            position--;
        if (clean)
            el['innerText'] = clean;
        let textnode = el.firstChild['splitText'](position);
        clean = textnode.wholeText;
        cursor = position;
        el.insertBefore(document.createTextNode('|'), textnode);
        el['innerText'] = textnode.wholeText;
    });
});
});
&lt;div class="charPosition"&gt;Lorem ipsum dolor sit amet, consectetur adipiscing elit.&lt;/div&gt;

【讨论】:

  • 不是我,可能是因为您添加了 2 个答案。
  • 你知道focusOffset的跨浏览器解决方案吗?
  • 我添加了第二个答案,因为第一个答案有一些支持,并且更改他们已经支持的消息似乎很奇怪。那好吧。 Selection API 有相当广泛的支持。 caniuse.com/#feat=selection-api IE8 等有一些 polyfills。
  • 您是使用 focusOffset 的第二个解决方案正是我所需要的,谢谢。通过跨浏览器找到了这个要点,用于 focusOffset gist.github.com/tcr/1022198
  • 但是看可以用吗,好像不需要IE8所以只要focusOffset就可以了,再次感谢。
猜你喜欢
  • 2012-12-21
  • 2019-09-02
  • 2012-02-06
  • 1970-01-01
  • 2018-09-10
  • 1970-01-01
  • 2011-02-23
相关资源
最近更新 更多