大部分Web应用的富文本内容都是以HTML字符串的形式存储的,通过HTML文档去展示HTML内容自然没有问题。但是,在微信小程序(下文简称为「小程序」)中,应当如何渲染这部分内容呢?

在微信小程序中渲染HTML内容的3种解决方案

wxParse

小程序刚上线那会儿,是无法直接渲染HTML内容的,于是就诞生了一个叫做「wxParse」的库。它的原理就是把HTML代码解析成树结构的数据,再通过小程序的模板把该数据渲染出来。

rich-text

后来,小程序增加了「rich-text」组件用于展示富文本内容。然而,这个组件存在一个极大的限制:组件内屏蔽了所有节点的事件。也就是说,在该组件内,连「预览图片」这样一个简单的功能都无法实现。

web-view

再后来,小程序允许通过「web-view」组件嵌套网页,通过网页展示HTML内容是兼容性最好的解决方案了。然而,因为要多加载一个页面,性能是较差的。

当「WePY」遇上「wxParse」

基于用户体验和功能交互上的考虑,我们抛弃了「rich-text」和「web-view」这两个原生组件,选择了「wxParse」。然而,用着用着却发现,「wxParse」也不能很好地满足需要:

我们的小程序是基于「WePY」框架开发的,而「wxParse」是基于原生的小程序编写的。要想让两者兼容,必须修改「wxParse」的源代码。

「wxParse」只是简单地通过image组件对原img元素的图片进行显示和预览。而在实际使用中,可能会用到云存储的接口对图片进行缩小,达到「用小图显示,用原图预览」的目的。

「wxParse」直接使用小程序的video组件展示视频,但是video组件的层级问题经常导致UI异常(例如把某个固定定位的元素给挡了)。

此外,围观一下「wxParse」的代码仓库可以发现,它已经两年没有迭代了。所以就萌生了基于「WePY」的组件模式重新写一个富文本组件的想法,其成果就是「WePY HTML」项目。

实现过程

解析HTML

首先仍然是要把HTML字符串解析为树结构的数据,我采用的是「特殊字符分隔法」。HTML中的特殊字符是「」,前者为开始符,后者为结束符。

如果待解析内容以开始符开头,则截取开始符到结束符之间的内容作为节点进行解析。

如果待解析内容不以开始符开头,则截取开头到开始符之前(如果开始符不存在,则为末尾)的内容作为纯文本解析。

剩余内容进入下一轮解析,直到无剩余内容为止。

为了形成树结构,解析过程中要维护一个上下文节点(默认为根节点):

如果截取出来的内容是开始标签,则根据匹配出的标签名和属性,在当前上下文节点下创建一个子节点。如果该标签不是自结束标签(br、img等),就把上下文节点设为新节点。

如果截取出来的内容是结束标签,则根据标签名关闭当前上下文节点(把上下文节点设为其父节点)。

如果是纯文本,则在当前上下文节点下创建一个文本节点,上下文节点不变。

上下文(解析前)

解析内容

上下文(解析后)

根节点

div

div

p

p

Hello world

p

p

div

div

根节点

经过上述流程,HTML字符串就被解析为节点树了。

对比

本组件算法

wxParse

parse5

性能

3~6ms

20ms左右

20ms左右

容错性

一般

文件大小(未压缩)

6kb

22kb

接近400kb

可见,在不考虑容错性(产生错误的结果,而非抛出异常)的情况下,本组件的算法与其余两者相比有压倒性的优势,符合小程序「小而快」的需要。而一般情况下,富文本编辑器所生成的代码也不会出现语法错误。因此,即使容错性较差,问题也不大(但这是需要改进的)。

模板渲染

树结构的渲染,必然会涉及到子节点的递归处理。然而,小程序的模板并不支持递归,这下仿佛掉入了一个大坑。

看了一下「wxParse」模板的实现,它采用简单粗暴的方式解决这个问题:通过13个长得几乎一模一样的模板进行嵌套调用(1调用2,2调用3,……,12调用13),也就是说最多可以支持12次嵌套。一般来说,这个深度也足够了。

由于「WePY」框架本身是有构建机制的,所以不必手写十来个几乎一模一样的模板,通过一个构建的插件去生成即可。

{{ item.text }}

以下是对应的构建代码(需要安装「wepy-plugin-replace」):

// wepy.config.js

{

plugins: {

replace: {

filter: /\.wxml$/,

config: {

find: /([\W\w]+?)/,

replace(match, tpl) {

let result = '';

// 反正不要钱,直接写个20层嵌套

for (let i = 0; i <= 20; i++) {

result += '\n' + tpl

.replace('wepyhtml-0', 'wepyhtml-' + i)

.replace(//g, () => {

return i === 20 ?

'' :

``;

});

}

return result;

}

}

}

}

}

然而,运行起来后发现,第二层及更深层级的节点都没有渲染出来,说明嵌套失败了。再看一下dist目录下生成的wxml文件可以发现,变量名与组件源代码的并不相同:

「WePY」在生成组件代码时,为了避免组件数据与页面数据的变量名冲突,会根据一定的规则给组件的变量名增加前缀(如上面代码中的「$htmlContent$wepyHtml$」)。

所以在生成嵌套模板时,也必须使用带前缀的变量名。 先在组件代码中增加一个变量「thisIsMe」用于识别前缀:

{{ thisIsMe }}

{{ item.text }}

然后修改构建代码:

replace(match, tpl) {

let result = '';

let prefix = '';

// 匹配 thisIsMe 的前缀

tpl = tpl.replace(/\{\{\s*(\$.*?\$)thisIsMe\s*\}\}/, (match, p) => {

prefix = p;

return '';

});

for (let i = 0; i <= 20; i++) {

result += '\n' + tpl

.replace('wepyhtml-0', 'wepyhtml-' + i)

.replace(//g, () => {

return i === 20 ?

'' :

``;

});

}

return result;

}

至此,渲染问题就解决了。

微信小程序中HTML包含图片

为了节省流量和提高加载速度,展示富文本内容时,一般都会按照所需尺寸对里面的图片进行缩小,点击小图进行预览时才展示原图。

这主要涉及节点属性的修改: 把图片原路径(src属性值)存到自定义属性(例如「data-src」)中,并将其添加到预览图数组。

把图片的src属性值修改为缩小后的图片URL(一般云服务商都有提供此类URL规则)。

点击图片时,使用自定义属性的值进行预览。 为了实现这个需求,本组件在解析节点时提供了一个钩子(onNodeCreate):

onNodeCreate(name, attrs) {

if (name === 'img') {

attrs['data-src'] = attrs.src;

// 预览图数组

this.previewImgs.push(attrs.src);

// 缩图

attrs.src = resizeImg(attrs.src, 640);

}

}

对应的模板和事件处理逻辑如下:

// 点击小图看大图

imgTap(e) {

wepy.previewImage({

current: e.currentTarget.dataset.src,

urls: this.previewImgs

});

}

微信小程序中HTML包含视频

在小程序中,video组件的层级是较高的(且无法降低)。

如果页面设计上存在着可能挡住视频的元素,处理起来就需要一些技巧了: 隐藏video组件,用image组件(视频封面)占位; 点击图片时,让视频全屏播放; 如果退出了全屏,则暂停播放。

相关代码如下:

{

// 点击封面图,播放视频

videoTap(e) {

const nodeId = e.currentTarget.dataset.nodeid;

const context = wepy.createVideoContext('wepyhtml-video-' + nodeId);

context.play();

// 在安卓微信下,如果视频不可见,则调用play()也无法播放

// 需要再调用全屏方法

if (wepy.getSystemInfoSync().platform === 'android') {

context.requestFullScreen();

}

},

// 视频层级较高,为防止遮挡其他特殊定位元素,造成界面异常,

// 强制全屏播放

videoPlay(e) {

wepy.createVideoContext(e.currentTarget.id).requestFullScreen();

},

// 退出全屏则暂停

videoFullscreenChange(e) {

if (!e.detail.fullScreen) {

wepy.createVideoContext(e.currentTarget.id).pause();

}

}

}

开源

最后贴一下「WePY HTML」的项目仓库: https://github.com/beiliao-web-frontend/wepy-html ,具体使用方法见项目内的 README 。

如果你在使用过程中遇到了问题,或者是有好的建议和意见,都可以在 Issues 中提出。

随着微信小程序的不断完善相信用不了多长时间就会有一种更加完美的解决方案,那时我们就不会再改来改去了。更多关于微信小程序开发的文章请点击下面的相关文章

微信小程序中嵌套html_在微信小程序中渲染HTML内容3种解决方案及分析与问题解决...相关推荐

  1. 微信小程序中嵌套html_在微信小程序中渲染HTML内容的方法示例

    大部分Web应用的富文本内容都是以HTML字符串的形式存储的,通过HTML文档去展示HTML内容自然没有问题.但是,在微信小程序(下文简称为「小程序」)中,应当如何渲染这部分内容呢? 解决方案 wxP ...

  2. php中嵌套html代码和html代码中嵌套php方式

    php中嵌套html代码和html代码中嵌套php方式 一.总结 拷贝的话直接html代码是极好的方式 1.php中嵌套html代码(本质是原生php):a.原生嵌套<?php .....?&g ...

  3. 微信小程序中嵌套html_微信小程序:web-view嵌套H5实现微信支付功能解决方案及填坑...

    ab7117c7d4947210c39e126a01d23ede.jpg 最近一个多月加班比较严重,偶尔休息一天也是在补睡眠+陪家人,比较长时间没有来进行总结记录了.今天不加班,开始为这段时间做的东西 ...

  4. 微信小程序中嵌套html_微信小程序中使用 web-view 内嵌 H5 时,登录问题的处理方法...

    在微信小程序的开发中,经常遇到需要使用 内嵌 H5 的需求.在这种需求中比较棘手的问题应该就是登录状态的判断了,小程序中的登录状态怎样与H5中的登录状态保持一致? 一般来说,后端开发同事多数会要求我们 ...

  5. 微信里嵌入html5页面,微信小程序web-view嵌套H5实现微信支付功能解决方案

    一. 产品现状 首先,在接入微信支付功能以前,我们的产品情况是这样的: 1 有公众号和app的h5站点及相关配套功能 2 小程序已经有一些基础功能,这些功能没有使用web-view 3 小程序之前的服 ...

  6. 网页中嵌套网页flush_如何修改网页中的内容?

    注意:请勿将本文中提到的技术用于违法行为,因此造成的损失我不承担任何责任. 简介 之前在内测群里聊天,我才发现几乎没有人知道如何修改一个网页中的内容,除了相关领域.可能是因为我已经接触了前端的一些知识 ...

  7. 在js中加html_在HTML文档中嵌入JavaScript的四种方法

    在HTML里嵌入JavaScript 在HTML文档里嵌入客户端JavaScript代码有4中方法: 1.内嵌,放置在标签之间  (少): 2.放置在有 3.放置自HTML事件处理程序中,该事件处理程 ...

  8. Myeclipse学习总结(14)——Eclipse/MyEclipse中Java编译时Java反射机形参没有保留问题2种解决方案

    一.IDE Compiler设置解决 Window -> Preferences -> Java -> Compiler. 选中Store information about met ...

  9. word中如果出现某一行突然文字突然间距增大,两种解决方案

    我们有时候会遇到这种问题.word中突然就有这么一行文字间距拉得特别大,选中"清楚所有格式"都不行:删掉重新打字也还是如此.看段落.字体,间距也没什么问题,和其他几行对照看,都一样 ...

最新文章

  1. python工具是什么-使用Python编写命令行工具有什么好的库?
  2. [学习笔记]圆方树广义圆方树
  3. http请求中的Query String Parameters、Form Data、Request Payload
  4. java web后台_java web 后台那些事
  5. HDU2191 珍惜现在,感恩生活【背包】
  6. mysql安装和下载过程_mysql下载与安装过程
  7. 有什么电脑软件可以测试网速,电脑怎么测试网速(测网速大全)
  8. 运筹系列77:开源线性规划软件clp使用简介
  9. 阿铭Linux_传统IDC 部署网站学习笔记20190122
  10. 计算机图片组合快捷键,电脑高手常用的组合快捷键
  11. GPRS无线MODEM模块上网设置命令
  12. Hard!168 · 吹气球
  13. 工单处理之--docker版java应用增加调试日志的笨方法
  14. 高校智慧运营BRAC方案助力燕山大学打造三方网络共享平台
  15. oracle默认导出dmp路径_Oracle导入导出dmp文件
  16. 去哪儿网——项目管理平台助力研发效率提升
  17. C++ OJ 出现 Wrong Answer的解决方法:如何把输出结果写入到文件中
  18. 解决xcode iOS真机调试正常,模拟器失败问题
  19. CVPR2020:扩展架构以实现高效的视频识别(X3D)
  20. 《进击吧!Blazor!》系列入门教程 第一章 4.数据交互

热门文章

  1. 名茶事典——【凤凰水仙】
  2. Python中的split()函数的用法
  3. 利用msfvenom生成木马文件反弹shell
  4. 高效能TCP通讯基础组件Beetle.Express
  5. 同义词辨析:stand,bear,endure,to…
  6. RelativeSource属性
  7. NYOJ-123-士兵杀敌(四)
  8. 2015年7月1日 课设日志
  9. 程序员面试,必要的注意事项
  10. 金融投资心得(个人领悟篇)