写在前面:

网上有各种富文本编辑器,微博分享等操作,这些功能非常实用,他们就是使用 range,selection 对象来实现的,这两个对象偏冷门,不涉及编辑器一般用不到,range 对象是对选区的操作,选区是鼠标在页面上按下,然后拖动选择的内容,range 对象就是对这个选区的操作;selection 是当前活动区的操作对象,selection 对象下面可以有多个 range 对象;这篇文章写的是标准的 selection和 range,小于 IE 9 的旧版本不在讨论范围内;

先来看一下 W3C 文档上的 range 对象的基本属性:

collapsed:返回一个用于判断 Range 起始位置和终止位置是否相同的布尔值。

commonAncestorContainer:返回包含 startContainer 和 endContainer 的最深的节点。

endContainer:返回包含 Range 终点的节点。

endOffset:返回 endContainer 中表示Range终点位置的数字。

startContainer:返回包含 Range 开始的节点。

startOffset:返回 startContainer 中表示 Range 起始位置的数字。

注意:range 的这些属性都是只读的,看下面这张图,就明白这些属性是怎么回事了:

collapsed 为 flase,说明起始点和终点不在同一个位置,startOffset 是从0开始的,d 的索引是2,表示鼠标按下的位置,也就是 div 元素的字符开始算的,endOffset 是 4,是从 span 标签开始算的,i 字符在 span 标签立面的索引是4;

注意,他们总是从包裹字符的节点的开始字符计算的;

再来看下 range 的方法和解释:

setStart(node, index):设置 Range 的起点。
setEnd(node, index):设置 Range 的终点。
setStartBefore(node):以其它节点( Node)为基准,设置 Range 的起点。
setStartAfter():以其它节点为基准,设置 Range 的始点。
setEndBefore():以其它节点为基准,设置 Range 的终点。
setEndAfter():以其它节点为基准,设置 Range 的终点。
selectNode():设定一个包含节点和节点内容的 Range。
selectNodeContents():设定一个包含某个节点内容的 Range。
collapse():向指定端点折叠该 Range。

这个解释看了之后一脸蒙圈,我的感受是讲的不清楚,还是上图,加上我们的理解更直观:

setStart setEnd 方法:

JS 操作:

 // 创建新的 range 对象,设置 setStart setEnd
function setStartEnd() {
var box = document.getElementById('box');
if (document.createRange) {
var range = document.createRange();
var start = box.querySelector('span').firstChild;
var end = box.querySelector('p').firstChild; range.setStart(start, 2); // p 标签的第 2 个字符
range.setEnd(end, 3); // span 标签的 第 3 个字符
console.log(range.cloneContents()); // 打印 range 的内容
console.log(range); // 打印 range
}
}

选区的内容和打印 range 的结果:

这个例子可以看出来,startOffset 的位置在第一个 span 标签的 g 字符前面,endOffset 的位置在 p 标签的 p 字符前面;

需要注意的是,如果是跨标签的选区,没有匹配到结束标签时,会折叠选区,把起点的位置设置为结束点的位置;

看下这个例子,把开始节点设置为 p 标签,结束节点为 span 标签,从后面向前面选择:

var start = box.querySelector('p').firstChild; // 开始节点是 p
var end = box.querySelector('span').firstChild; // 结束节点是 span

打印出来的结果:

仔细检查会发现 collapsed: true,并且 startContainer 和 endContainer 是同一个元素,这说明这个 range 选区折叠了,所以操作的时候需要注意;

setStartBefore setEndBefore 方法:

JS 操作:

var box = document.getElementById('box');
if (document.createRange) {
var range = document.createRange();
var spans = box.querySelectorAll('span');
var refStart = spans[0].firstChild;
var refEnd = spans[1].firstChild; range.setStartBefore(refStart); // 设置 p 节点的开始位置为 range 的起点
range.setEndBefore(refEnd); // 设置 span 节点的开始位置为 range 的终点 console.log(range.cloneContents());
console.log(range);
}

这两个方法的作用是以参考节点的开始位置作为参考,来设置 range 的起点和结束点;

setStartAfter,setEndAfter 方法:

JS:

var box = document.getElementById('box');
if (document.createRange) {
var range = document.createRange();
var spans = box.querySelectorAll('span');
var refStart = spans[0].firstChild;
var refEnd = spans[1].firstChild; range.setStartAfter(refStart); // 设置 p 节点的结束位置为 range 的起点
range.setEndAfter(refEnd); // 设置 span 节点的结束位置为 range 的终点 console.log(range.cloneContents());
console.log(range);
}

这两个方法的作用是以参考节点的结束位置作为参考,来设置 range 的起点和结束点;

selectNode 方法:

JS:

var box = document.getElementById('box');
if (document.createRange) {
var range = document.createRange();
var span = box.querySelector('span').firstChild; range.selectNode(span); console.log(range.cloneContents());
console.log(range);
}

selectNode 方法的操作就是 range 选择整个节点的,默认 startOffset 是 0,endOffset 是1;

selectNodeContents 方法:

这个方法和 selectNode 很类似,区别是会默认把结束点的位置移到最后;

var box = document.getElementById('box');
if (document.createRange) {
var range = document.createRange();
var span = box.querySelector('span').firstChild; range.selectNodeContents(span); console.log(range.cloneContents());
console.log(range);
}

collapse 方法是设置折叠选区的,开始位置和结束位置相同,这么解释就清楚多了,作用是多用来设置光标的;

下一篇看一下 selection 对象的基本属性和用法;

编辑器开发之 Range 范围对象的学习的更多相关文章

  1. ios开发之OC基础-ios开发学习路线图

    本系列的文章主要来自于个人在学习前锋教育-欧阳坚老师的iOS开发教程之OC语言教学视频所做的笔记,边看视频,边记录课程知识点.建议大家先过一遍视频,在看视频的过程中记录知识点关键字,把把握重点,然后再 ...

  2. 编辑器开发之 Selection 对象的学习

    上一篇,介绍了 range 对象的一些属性和方法,了解了一些基本操作,现在来介绍另外一个重要的对象:selection 对象: MDN 的解释是:Selection 对象表示用户选择的文本范围或插入符 ...

  3. 李洪强iOS开发之OC[014] -对象的存储细节

    // //  main.m //  13 - 对象的存储细节 // //  Created by vic fan on 16/7/9. //  Copyright © 2016年 李洪强. All r ...

  4. 李洪强iOS开发之OC[018]对象和方法之间的关系

    // //  main.m //  18 - 对象和方法之间的关系 // //  Created by vic fan on 16/7/14. //  Copyright © 2016年 李洪强. A ...

  5. Winform开发之ADO.NET对象Connection、Command、DataReader、DataAdapter、DataSet和DataTable简介

    ADO.NET技术主要包括Connection.Command.DataReader.DataAdapter.DataSet和DataTable等6个对象,下面对这6个对象进行简单的介绍:(1)Con ...

  6. Android游戏开发之SurfaceView的使用-android学习之旅(五)

    SurfaceView和View的区别 View是在ui主线程中,直接响应用户的操作,以及任务的分发,但是任务比较复杂会出现阻塞. SurfaceView则不会出现这种问题,以为它直接从内存等取得图像 ...

  7. ios开发之OC基础-类和对象

    本系列的文章主要来自于个人在学习前锋教育-欧阳坚老师的iOS开发教程之OC语言教学视频所做的笔记,边看视频,边记录课程知识点.建议大家先过一遍视频,在看视频的过程中记录知识点关键字,把把握重点,然后再 ...

  8. 高效开发之SASS篇 灵异留白事件——图片下方无故留白 你会用::before、::after吗 link 与 @import之对比 学习前端前必知的——HTTP协议详解 深入了解——CSS3新增属性 菜鸟进阶——grunt $(#form :input)与$(#form input)的区别

    高效开发之SASS篇   作为通往前端大神之路的普通的一只学鸟,最近接触了一样稍微高逼格一点的神器,特与大家分享~ 他是谁? 作为前端开发人员,你肯定对css很熟悉,但是你知道css可以自定义吗?大家 ...

  9. ArcGIS Engine开发之旅09--几何对象和空间参考

    原文:ArcGIS Engine开发之旅09--几何对象和空间参考 1.Geometry  Geometry 是 GIS 中使用最为广泛的对象集之一,用户在创建.删除.编辑和进行地理分析的时候,就是处 ...

随机推荐

  1. 在线OJ使用总结(acm)

    赛码网OJ规则 用readLine()代替read_line() 用readLine()代替read_line() 用readLine()代替read_line() 用readLine()代替read ...

  2. PHP与XML技术

    XML的概述 XML(eXtensibleMarkup Language),扩展性标记语言,它是用来描述其他语言的语言.它允许用户设计自己的标记.XML是由W3C(WorldWide 月发布的一种标准 ...

  3. sql server 高可用镜像

    一.什么是数据库镜像 基本软件的高可用性解决方案 快速的故障转移恢复(3秒转移),低硬件成本 基于数据库级别的实现 二.数据库镜像中的服务器角色 主体服务器 承载主体数据库 接受用户连接和事务处理请求 ...

  4. http_server.go

    , fmt.Sprintf("%s: closing %s", proto, listener.Addr())) }

  5. 基于ELK5.1(ElasticSearch, Logstash, Kibana)的一次整合测试

    前言开源实时日志分析ELK平台(ElasticSearch, Logstash, Kibana组成),能很方便的帮我们收集日志,进行集中化的管理,并且能很方便的进行日志的统计和检索,下面基于ELK的最 ...

  6. 将外部dwg图纸中指定带属性的块插入到当前图纸中

    static void InsertBlock() { //获取要插入的块名 TCHAR str[40]; acedGetString(Adesk::kFalse, _T("\n请输入要插入 ...

  7. 【线段树】Bzoj1798 [AHOI2009] 维护序列

    Description 老师交给小可可一个维护数列的任务,现在小可可希望你来帮他完成. 有长为N的数列,不妨设为a1,a2,…,aN .有如下三种操作形式: (1)把数列中的一段数全部乘一个值; (2 ...

  8. 【Unity游戏开发】Lua中的os.date和os.time函数

    一.简介 最近马三在工作中经常使用到了lua 中的 os.date( ) 和 os.time( )函数,不过使用的时候都是不得其解,一般都是看项目里面怎么用,然后我就模仿写一下.今天正好稍微有点空闲时 ...

  9. 【爆料】-《卧龙岗大学毕业证书》UOW一模一样原件

    ☞西悉尼大学毕业证书[微/Q:865121257◆WeChat:CC6669834]UC毕业证书/联系人Alice[查看点击百度快照查看][留信网学历认证&博士&硕士&海归&a ...

  10. .net core Entity Framework 与 EF Core

    重点讲 Entity Framework Core ! (一)Entity Framework 它是适用于.NET 的对象关系映射程序 (ORM),现在的EF6已经是久经沙场,并经历重重磨难,获得一致 ...