前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >JavaScript 中获取光标位置

JavaScript 中获取光标位置

作者头像
越陌度阡
发布于 2020-11-26 04:27:17
发布于 2020-11-26 04:27:17
12.8K01
代码可运行
举报
运行总次数:1
代码可运行

1.概念和原理

DOM中并没有直接获取光标位置的方法,那么我们只能间接来获取光标位置。DOM支持获取光标选中的范围,我们可以以此为切入点,来获取或定位光标的位置,当选取范围起始点和结束点一样时,就是光标插入的位置。

1.1. 术语

anchor:选区起点。

focus:选区终点。

range:选区范围,包含整个节点或节点的一部分。

1.2. Selection

Selection对象表示用户选择的文本范围或插入符号的位置。

Selection选取的节点范围都是块级节点,input和texteare并不能作为Selection的节点。

Selection对象存在于window对象上,可以通过window.getSelection()获取示例。

属性:

anchorNode:选区起点的节点。

anchorOffset:选区的起点位置。

focusNode:选区终点的节点。

focusOffset:选区的终点位置。

isCollapsed:起点和终点是否重叠。

rangeCount:选区包含的range数目。

方法

getRangeAt(index):获取指定的选取范围。

addRange(range):将一个范围添加到Selection对象中。

removeRange():移出指定的范围。

removeAllRanges():移出所有range对象。

collapse(parentNode,offset):将光标移动到parentNode节点的offset位置。

collapseToStart():取消当前选区,并把光标定位在原选区的最开始处,如果此时光标所处的位置是可编辑的,且它获得了焦点,则光标会在原地闪烁。

collapseToEnd():取消当前选区,并将光标定位到原选取的最末位,如果此时光标所处的位置是可编辑的,且它获得了焦点,则光标会在原地闪烁。

extend(node,offset):将终点位置移动到node节点的offset位置。

modify(alter,direction,granularity):通过alter方式(move/extend)来改变光标位置,移动方向为direction(left/right),移动单位为granularity。

containsNode(aNode,aPartlyContained):判断aNode是否包含在Selection中,aPartlyContained为false表示全包含,为true表示只要部分包含即可。

toString():放回当前Selection对象的字符串。

1.3. Range

Range对象表示一个Selection的选择范围,一个Selection可以包含多个Range

获取对象

document.createRange():创建一个Range。

selection.getRangeAt(index):获取指定的Range。

属性

collapsed:判断起始位置是否重合。

endContaniner:range终点节点。

endOffset:range的终点位置。

startContaniner:ranstartge起点节点。

startOffset:range的起点位置。

commonAncestorContainer:包含起始点的节点。

方法

setStart(startNode,startOffset):设置范围在startNode的起始位置为startOffset。

setEnd(endNode,endOffset):设置范围在endNode的起始位置为endOffset。

selectNode(referenceNode):设置range的节点为referenceNode。

selectNodeContents(referenceNode):设置range的内容为referenceNode。

collapse(toStart):向边界点折叠range,即是设置光标位置,toStart默认为false,表示光标定位在节点末尾,true表示光标定位在节点起点。

cloneContents():克隆一个range的内容片段。

deleteContents():删除range的内容片段。

extractContents():将range的内容从文档树移动到文档片段中。

insertNode(newNode):在range的其实位置插入新的节点。

surroundContents(newNode):将range对象的内容移动到新的节点中。

cloneRange():克隆一个range对象。

detach():释放当前range。

1.4. input/textarea

HTML5中,input/textarea 都存在以下属性,不支持IE6/7。

selectionDirection:forward | backward | none,选区方向。

selectionEnd:选区终点位置

selectionStart:选区起点位置

setSelectionRange(selectionStart, selectionEnd, [selectionDirection]):设置获取焦点的输入性元素的选区范围。

2. 获取光标位置

2.1. 可编辑div获取光标位置

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 获取当前光标位置
const getCursortPosition = function (element) {
    var caretOffset = 0;
    var doc = element.ownerDocument || element.document;
    var win = doc.defaultView || doc.parentWindow;
    var sel;
    // 谷歌、火狐
    if (typeof win.getSelection != "undefined") {
        sel = win.getSelection();
        // 选中的区域
        if (sel.rangeCount > 0) {
            var range = win.getSelection().getRangeAt(0);
            // 克隆一个选中区域
            var preCaretRange = range.cloneRange();
            // 设置选中区域的节点内容为当前节点
            preCaretRange.selectNodeContents(element);
            // 重置选中区域的结束位置
            preCaretRange.setEnd(range.endContainer, range.endOffset);
            caretOffset = preCaretRange.toString().length;
        }
    // IE浏览器
    } else if ((sel = doc.selection) && sel.type != "Control") {
        var textRange = sel.createRange();
        var preCaretTextRange = doc.body.createTextRange();
        preCaretTextRange.moveToElementText(element);
        preCaretTextRange.setEndPoint("EndToEnd", textRange);
        caretOffset = preCaretTextRange.text.length;
    }
    return caretOffset;
}

2.2. input/textarea获取光标位置

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 输入框获取光标
const getPosition = function (element) {
    let cursorPos = 0;
    // IE浏览器
    if (document.selection) {
        var selectRange = document.selection.createRange();
        selectRange.moveStart('character', -element.value.length);
        cursorPos = selectRange.text.length;
    } else if (element.selectionStart || element.selectionStart == '0') {
        cursorPos = element.selectionStart;
    }
    return cursorPos;
}

3. 设置光标位置

3.1. 可编辑div设置光标位置

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 设置光标位置
const setCaretPosition = function (element, pos) {
    var range, selection;
    // Firefox, Chrome, Opera, Safari, IE 9+
    if (document.createRange) {
        // 创建一个选中区域
        range = document.createRange();
        // 选中节点的内容
        range.selectNodeContents(element);
        if (element.innerHTML.length > 0) {
            // 设置光标起始为指定位置
            range.setStart(element.childNodes[0], pos); 
        }
        // 设置选中区域为一个点
        range.collapse(true);       
        // 获取当前选中区域
        selection = window.getSelection();
        // 移除所有的选中范围
        selection.removeAllRanges();
        // 添加新建的范围
        selection.addRange(range);
    //IE 8 and lower
    }else if (document.selection){
        // 创建一个范围(范围与所选内容类似但不可见)
        range = document.body.createTextRange();
        // 选择范围的元素的全部内容
        range.moveToElementText(element);
        // 将范围折叠到终点
        range.collapse(false);
        // 选择范围
        range.select();
    }
}

3.2. input/textarea获取光标位置

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 设置光标位置
function setCaretPosition(textDom, pos) {
    if (textDom.setSelectionRange) {
        // IE Support
        textDom.focus();
        textDom.setSelectionRange(pos, pos);
    } else if (textDom.createTextRange) {
        // Firefox support
        var range = textDom.createTextRange();
        range.collapse(true);
        range.moveEnd('character', pos);
        range.moveStart('character', pos);
        range.select();
    }
}

4. 示例

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<!DOCTYPT html>

<html>

<head>
    <title>JavaScript 中获取光标位置</title>
    <style>
        p {
            display: flex;
            flex-direction: row;
        }

        .btn {
            height: 24px;
            margin: 0 10px;
        }

        .edit-div {
            display: inline-block;
            width: 225px;
            border: 1px solid #decdcd;
        }
    </style>
    <script>

        function getCursortPosition(e) {
            // 获取父级元素
            var eleP = e.target.parentNode;
            var pos = 0;
            if (e.target.nodeName == "DIV") {
                pos = getDivPosition(e.target);
            } else {
                pos = getPosition(e.target);
            }
            var spanEle = (eleP.childNodes)[7];
            spanEle.innerText = pos;
        }

        // 可编辑div获取坐标
        const getDivPosition = function (element) {
            var caretOffset = 0;
            var doc = element.ownerDocument || element.document;
            var win = doc.defaultView || doc.parentWindow;
            var sel;
            // 谷歌、火狐
            if (typeof win.getSelection != "undefined") {
                sel = win.getSelection();
                // 选中的区域
                if (sel.rangeCount > 0) {
                    var range = win.getSelection().getRangeAt(0);
                    // 克隆一个选中区域
                    var preCaretRange = range.cloneRange();
                    // 设置选中区域的节点内容为当前节点
                    preCaretRange.selectNodeContents(element);
                    // 重置选中区域的结束位置
                    preCaretRange.setEnd(range.endContainer, range.endOffset);
                    caretOffset = preCaretRange.toString().length;
                }
                // IE
            } else if ((sel = doc.selection) && sel.type != "Control") {
                var textRange = sel.createRange();
                var preCaretTextRange = doc.body.createTextRange();
                preCaretTextRange.moveToElementText(element);
                preCaretTextRange.setEndPoint("EndToEnd", textRange);
                caretOffset = preCaretTextRange.text.length;
            }
            return caretOffset;
        }

        // 输入框获取光标
        const getPosition = function (element) {
            let cursorPos = 0;
            // IE
            if (document.selection) {
                var selectRange = document.selection.createRange();
                selectRange.moveStart('character', -element.value.length);
                cursorPos = selectRange.text.length;
            } else if (element.selectionStart || element.selectionStart == '0') {
                cursorPos = element.selectionStart;
            }
            return cursorPos;
        }


    </script>
</head>

<body>
    <p>
        <label>输入框测试:</label>
        <input type="text" style="width:220px" onclick="getCursortPosition(event);" />
        <span>光标位置:</span>
        <span></span>
    </p>
    <p>
        <label>文本框测试:</label>
        <textarea rows="5" style="width:220px" onclick="getCursortPosition(event);"></textarea>
        <span>光标位置:</span>
        <span></span>
    </p>
    <div>
        <label>可编辑div:</label>
        <div contenteditable="true" class="edit-div" onclick="getCursortPosition(event);"></div>
        <span>光标位置:</span>
        <span></span>
    </div>
</body>

</html>
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2020/03/04 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
在可编辑div中定位光标和设置光标
当我们去点击一个输入框的时候,就会产生一个选中对象 selection,就是我们可以看到的文字变成蓝色的那个区域,selection在火狐浏览器可以直接用 window.getSelection()获取,在HTML里面,selection只有一个的,并且selection是一个区域,你可以想象成一个长方形,它是有开始和结束的。
越陌度阡
2020/11/26
9.8K0
TextRange之插入表情
SNS类或是微博类的产品一般都有一个功能:插入表情,如下所示: 重点:兼容IE与其它主流的浏览器,根据上一次选择的选区范围进行操作。 自己写了一个TextRange(参考了网上的一些例子和代码) /
meteoric
2019/02/25
1.2K0
TextRange之插入表情
在评论输入框中插入表情
最近在做一个后台管理系统,要求可以对前台用户的作品进行评论,而评论要可以输入表情,常规的文字输入框都是用的文本域textarea来做的,但这种输入框只能输入文字,没有办法输入表情图标,这个时候可编辑div就能起到作用了,那么如何在可编辑的div中插入表情呢?
越陌度阡
2020/11/26
4.3K0
造一个 copy-to-clipboard 轮子
用 JS 来复制文本在网页应用里十分常见,比如 github 里复制 remote 地址的功能:
写代码的海怪
2022/03/29
9350
造一个 copy-to-clipboard 轮子
js获取div编辑框,textarea,input text的光标位置,兼容FF和IE
<h<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <title>js获取div编辑框,textarea,input text的光标位置,兼容FF和IE</title> <script src="http:
joshua317
2018/04/10
7.9K0
textarea 在光标处插入文字
效果演示 欢迎访问cssfirefly.cnblogs.com html: <textarea id="text" style="width:500px;height:80px;">欢迎访问http://cssfirefly.cnblogs.com/</textarea> <input type="button" value="插入文字" onclick="insertText(document.getElementById('text'),' NewWord ')"> <input type="but
deepcc
2018/05/16
1.2K0
WEB:Range 与 Selection
为了让开发人员更方便地控制页面,“DOM2 级遍历和范围”模块定义了“范围(Range)接口”。通过范围(Range)可以选择文档中的一个区域,而不必考虑节点的界限。
WEBJ2EE
2019/07/19
1.6K0
WEB:Range 与 Selection
vue.js聊天IM系统|聊天室|群聊
基于vue+vuex+vue-router+webpack2.0+es6+wcPop+iconfont等技术开发的仿微信界面聊天室,之前使用h5开发过一版h5聊天室,实现了微信聊天功能、发送消息/表情,图片、视频预览,打赏、红包等功能。
andy2018
2019/04/05
12.9K2
vue.js聊天IM系统|聊天室|群聊
h5仿微信web端|仿wechat网页聊天实战
《H5+CSS3微信h5微场景实战开发》仿微信H5电脑端聊天场景项目案例、h5仿微信聊天界面
andy2018
2018/06/14
4.2K0
h5仿微信web端|仿wechat网页聊天实战
【留言板】可编辑输入框操作总结
闲暇之余,用于加深自己对基础的了解,徒手撸了一个留言板:输入框。废话少说,进入正题。简陋的效果如下(下载代码): 一、定义需求 可输入文本,以及插入表情。 兼容性:IE与标准浏览器 二、详细设计 根据
sam dragon
2018/01/17
1.5K0
【留言板】可编辑输入框操作总结
【译】JavaScript实现文字剪贴板&amp;React版本
目录 怎么使用JavaScript实现一个剪贴 上面的方法不是很完美我们优化一下 思考一个问题 使用react和typescript重写和优化一下 如何使用copyme 原文参考 写在最前面 有一个简单的需求,用户需要快捷的复制一些相关的信息,然后进行下一步信息的填写。前端这里需要做一个剪贴板方便用户体验。想直接参考 react 使用的可以看 使用react和typescript改写和优化一下 大概设计如下,有多条信息,然后用户可以点击右边的复制 icon 进行快捷的复制。 怎么使用JavaScr
西南_张家辉
2022/09/16
8610
【译】JavaScript实现文字剪贴板&amp;React版本
前端富文本基础及实现
在日常生活中我们会经常接触到各种各样的文档格式和形式,其中富文本在文档格式中扮演了重要角色。对于前端而言,富文本产品也层出不穷,其应用也越来越广。
政采云前端团队
2022/12/01
4.7K1
前端富文本基础及实现
JS魔法堂:判断节点位置关系
一、前言                             在polyfill querySelectorAll 和写弹出窗时都需要判断两个节点间的位置关系,通过jQuery我们可以轻松搞定,但原生JS呢?下面我将整理各种判断方法,以供日后查阅。 二、祖孙关系                         html <div id="ancestor"> <div id="parent"> <div id="son">son</div> </div> </div> <
^_^肥仔John
2018/01/18
5.4K0
你这磨人的小妖精——选中文本并标注的实现过程
但是现在问题来了,我们这是一个现成的react页面,是一个详情页,页面的内容是多个接口返回填进去的:
lhyt
2020/07/21
2K0
富文本编辑器开发简介
富文本编辑器,Rich Text Editor, 简称 RTE, 是一种可内嵌于浏览器,所见即所得的文本编辑器。
江米小枣
2020/06/15
4.6K0
【译】JavaScript实现文字剪贴板&React版本
目录 怎么使用JavaScript实现一个剪贴 上面的方法不是很完美我们优化一下 思考一个问题 使用react和typescript重写和优化一下 如何使用copyme 原文参考 写在最前面 有一
西南_张家辉
2021/02/02
5320
Vue3.x+Vant3仿微信聊天|朋友圈
2021已来临,Vue3开发也要步入正轨了。今天为大家带来的是全新开发的Vue3.x实战聊天室项目Vue3_Chatroom,基于vue3+vant3+vuex+vue-router+v3popup等技术搭建的聊天实例。
andy2018
2021/01/09
4.3K2
Vue3.x+Vant3仿微信聊天|朋友圈
js 实现文本选中与复制
selection api 配合  createRange api 配合 document.execCommand('copy', true)
蓓蕾心晴
2022/10/30
9.3K0
js 实现文本选中与复制
基于react18+arco+vite4桌面版聊天react-webchat
react18-webchat 基于vite4.x+react18创建网页版聊天实战项目。
andy2018
2023/09/12
7852
可编辑DIV设置光标位置
平时我们使用文本框input, textarea时,我们有时候需要得到光标的位置或者说设置光标的位置. 但是文本框有个不足,就是不能满足我们插入图片或者其他的一些要求. 所以有时候我们使用div来代替文本框. 今天为了搞定设置DI的光标位置.头疼啊,终于可以了,记录一下,网上资料不多. 首先要让DIV启用编辑模式. <div contenteditable=true id="divTest"></div> 通过设定contenteditable=true开启div的编辑模式.这样DIV就可以跟文本框一样输入
joshua317
2018/04/10
6.8K0
相关推荐
在可编辑div中定位光标和设置光标
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验