IM应用中如何计算富文本的高度

背景

在开发IM的项目过程中,经常会有出现一些需要计算DOM高度,然后超出若干行隐藏等需求。很多时候,需要计算高度的DOM元素都是动态生成的,我们无法在数据渲染前获取到它的高度。

如果没有任何交互,我们可以通过CSS来实现这个需求。但是,如果我们需要使用JavaScript来实现一些交互(比如消息渲染时,超过2行显示某个特定按钮等),则只能通过JavaScript来进行实现。我在这里介绍一种通过JavaScript来对元素高度进行计算的方法,希望能够给大家提供一些思路。

技术方案

根据前端的基本常识,在内存中未渲染的DOM元素是无法获取到高度的,因此我们有两个方向来解决这个难题:

  1. 通过字数对行数进行估算。
  2. 将元素渲染后进行高度测算 。

实现方案

以下的实现方案将根据上面所选择的技术方案来进行实现。

通过字数进行估算

方案

此方案无需多言,就是通过字的总数、行高和每一行能够容下的字的个数进行估算等。在项目最开始时,我采用的就是这个方案。具体实现代码较为简单,因此不在本文中提供示例。

优点

此方案实现简单,基本不需要任何技术成本。

缺点

只适用于等宽文字,如果出现富文本(比如有emoji或者图片表情等高度不一致)的情况,则无法适用。如果字体为非等宽字体或者存在\n之类的换行符或者是\t之类的制表符时,估算的准确度也会下降。

在DOM渲染后进行操作

方案

顾名思义,此方案就是先不考虑DOM元素行数逻辑,直接将所有的DOM节点全部渲染到页面中,渲染完成后再对进行后续逻辑判断。获取高度后页面行数计算将在后面统一讲解。

优点

此方案通过直接在实际场景的页面上渲染后进行高度计算,因此计算精准,不存在任何偏差。同时,此方案实现起来也较为简单,只需要将业务逻辑执行时间后延,并不需要开发额外的代码。

缺点

该方案缺点也比较明显,由于是先渲染后处理,因此页面DOM元素会出现重绘和重排,导致页面闪动,从而影响用户的体验。

镜像计算

方案

该方案的灵感来自于上一个方案。因为在实际的页面中进行计算能够保证页面高度计算没有任何误差,因此我们需要一个实际的场景,让浏览器来帮助我们进行高度计算。同时,我们又不能在具体的功能页面中先渲染后计算,因此我们可以直接创建一个与实际页面中一模一样的容器来进行高度计算。这样我们既能够精确计算,又能够不影响用户体验。

具体实现的代码可以参考如下示例:

export default function getLines(element = 'div', style = {}, html = '') {let node = document.createElement(element);//创建一个新容器let length;each(style, (element, index) => {node.style[index] = element;//将传入的style遍历后赋值给新容器});node.innerHTML = html;document.body.appendChild(node);//需要将新容器挂载到DOM中,浏览器才会进行高度计算let height = global.getComputedStyle(node).height;document.body.removeChild(node);//需要将镜像DOM进行移除if (height.indexOf('px') > 0) {length = parseInt(height.split('px')[0]);} else {length = 0;}return length;
}
复制代码

优点

该方案基本上继承了第二个方案的所有优点——精确计算,无误差,并且避免了出现页面闪动的情况。

缺点

此方案仍然存在一些问题,将新容器挂载到document元素上时,可能会引发DOM元素的重新渲染,极低概率会影响页面布局。同时,属性值等需要自己手动传入,而不是利用现成的容器,比较费时费力。

方案再优化

利用现有DOM容器

使用cloneNode方法来对现有的容器进行clone,我们可以省去输入样式的麻烦,同时能够精确保证两个容器完全一致。

隐藏镜像DOM

在实践过程中,在append以后立刻remove镜像DOM节点,不会对页面产生任何影响。如果担心添加时会给页面造成闪动效果,可以给镜像DOM添加上position:fixed;visibility:hidden;z-index:-999;属性,能够让镜像DOM在append到页面时,不会影响当前页面的任何布局。

为什么我们不使用display:none来实现上述效果呢?因为在使用了该属性后,window.getComputedStyle获取的高度将变为auto。同理,如果元素的display属性为inline时,也会出现类似的效果,因此我们需要将display指定为block或者inline-block

理论上我们的容器都应该为块级元素,否则计算高度的意义也就不存在了。因此在容器clone时只需要留意即可,不需要重新指定。

两个优化点经过实践已经证明可行,具体代码就不附上了,如果有需要的可以给我留言~~

通过高度来计算行数

目前,通过高度来计算行数并没有什么比较好的方法,一般是通过line-height两个属性来进行计算。

如果line-height为倍数的话,则还需要font-size属性来确定具体高度。

具体算法为:总高度 / 每一行高度 = 行数

而每一行高度则通过line-height或者line-height* font-size确定。

总结

获取动态元素的高度一直都是IM项目中的一个重要需求,自己在这个方面也踩了许多坑,因此写了这一篇博客来进行记录,同时其他人如果看到了也可以避免一些常见问题。

由于此方案较为繁琐,同时容易留下不少坑,不太推荐使用此方法,还是建议通过产品方案等其他手段规避此方案。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/254716.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

G代码 机器人的CNC实现

  控制铣削工作台和工件的NC程序,通过CAD软件创建,这些NC程序与特定的机器类型相关。 NC程序在笛卡尔坐标系中动作的描述,对于需要确保一个明确的变换轴位置的关节型的机器人来说,缺少附加的状态和旋转信息。传…

IScroll5中文API整理,用法与参考

IScroll是移动页面上被使用的一款仿系统滚动插件。IScroll5相对于之前的IScroll4改进了许多,使得大家可以更方便的定制所需的功能了。 做项目的时候正好用到了这个插件,自己做了一下总结,发在这里方便大家学习IScroll5。 官网:htt…

Linux 安装USB摄像头

sudo apt-get updatesudo apt-get install fswebcamsudo apt-get install mplayersudo apt-get install alsamixer安装完毕ls /dev查找设备是否有video0这个设备sudo mplayer tv:// 可以看到摄像内容转载于:https://www.cnblogs.com/smartkeke/p/6820426.html

struct x264_t 维护着CODEC的诸多重要信息

//x264_t结构体维护着CODEC的诸多重要信息struct x264_t{/* encoder parameters ( 编码器参数 )*/x264_param_t param;x264_t *thread[X264_SLICE_MAX];/* bitstream output ( 字节流输出 ) */struct{int i_nal;x264_nal_t nal[X264_NAL_MAX];int i_bitstr…

如何判断一条曲线是否自己相交?

今天看到群里有人在问这个问题,想了一个解决办法。 我们首先作假设,如果一条曲线有交点,那么它就是相交的对吧。可能大家想的都是这样,就开始找各种方法去识别交点。 我们换个角度想一下:是不是我们判断这条曲线是否带…

XML 与网络的数据传输

XML 与网络的数据传输

hdu 5813 Elegant Construction

水题 题意:有n个城市,给你每个城市能到达城市的数量,要你构图,输出有向边,要求无环,输出任意的解 例: Sample Input 332 1 021 143 1 1 0Sample OutputCase #1: Yes21 22 3Case #2: NoCase #3: …

Redis实战笔记

Redis 数据库 一、 概要 1. 特点 用于抽象数据类型的 DSL内存存储基础数据结构 API编码风格避免代码复杂两层 API以优化为乐2. 数据类型 键值对(字符串->字符串)哈希列表(链表)集合:差并交有序集合 列表 集合位图…

内存申请与一级二级指针

1.如果是函数内进行内存申请,很简单,标准用法就可以了: test(){int *array;array(int *)malloc(sizeof(int)*10);//申请10*4 bytes,即10个单位的int内存单元}注意,malloc使用简单,但是注意参数和返回值&…

halcon相机标定及图像矫正(代码)

侵删 1 halcon相机标定和图像矫正 对于相机采集的图片,会由于相机本身和透镜的影响产生形变,通常需要对相机进行标定,获取相机的内参或内外参,然后矫正其畸变。相机畸变主要分为径向畸变和切向畸变,其中径向畸变是由透…

找寻一个邮箱

import java.util.Scanner; import java.util.regex.Matcher; import java.util.regex.Pattern;public class zhengze {public static void main(String[] args) { //1.创建一个正则表达式对象Pattern pPattern.compile("[0-9]{6}"); //2.获得匹配器 String s…

先弄个XML解析器代码抄一抄 慢慢研究 O(∩_∩)O哈哈~

出处:http://bbs.csdn.net/topics/390229172 已经自我放逐好几年了.打算去上班得了.在最后的自由日子里,做点有意义的事吧... 先来下载地址 http://www.kuaipan.cn/file/id_12470514853353274.htm 已经在很多正式,非正式的场合…

紫书 例题8-10 UVa 714 (二分答案)

这道题让最大值最小, 显然是二分答案当题目求的是最大值最小, 最小值最大, 这个时候就要想到二分答案为什么可以二分答案呢, 因为这个时候解是单调性的, 如果简单粗暴一点就全部枚举一遍, 验证答案。但是因…

was not declared in this scope

“was not declared in this scope”是一个错误信息,在编译的时候会遇到。其含义为标识符在其出现的地方是未被定义的。 出现该错误的时候,会同时把未定义的变量名显示出来。比如如下程序: int main(){ printf("%d",i);//这个i是…

函数参数的传递问题(一级指针和二级指针)

函数参数的传递问题(一级指针和二级指针) [转]原以为自己对指针掌握了,却还是对这个问题不太明白。请教! 程序1: void myMalloc(char *s) //我想在函数中分配内存,再返回 { s(char *) malloc(100); } void …

Win7下使用U盘安装linux Ubuntu16.04双系统图文教程

Win7下使用U盘安装linux Ubuntu16.04双系统图文教程 Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的开源GNU/Linux操作系统,Ubuntu 是基于DebianGNU/Linux,支持x86、amd64(即x64)和ppc架构&#xf…

SynchronizationContext

SendOrPostCallback xxx vg > { Text "内部: "vg.ToString(); };dynamic vx new { a SynchronizationContext.Current, b xxx };Thread td new Thread(x >{dynamic tmp x;// SynchronizationContext ds x as SynchronizationContext;for (in…

CoDeSys的前世今生

工作以及网上看到不少人说,CoDeSys和西门子step7,在德国都属于标准过程,牛逼的小朋友都可以用其编程,不知真假,相信无风不起浪,多少有些依据,看看国內清一色的日系编程…

UVALive 7324 ASCII Addition (模拟)

ASCII Addition题目链接: http://acm.hust.edu.cn/vjudge/contest/127407#problem/A Description Nowadays, there are smartphone applications that instantly translate text and even solve math problems if you just point your phone’s camera at them. You…

Eclipse中执行Ant脚本出现Could not find the main class的问题及解

试过了:https://blog.csdn.net/bookroader/article/details/2300337 但是不管用,偶然看到这篇没有直接关系的 https://blog.csdn.net/jiuyueguang/article/details/9350753 联想了一下。项目是JDK1.5,Eclipse是JDK1.8启动,所以在R…