热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

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

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

大部分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: /([Ww]+?)/,

replace(match, tpl) {

let result = "";

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

for (let i &#61; 0; i <&#61; 20; i&#43;&#43;) {

result &#43;&#61; "

" &#43; tpl

.replace("wepyhtml-0", "wepyhtml-" &#43; i)

.replace(//g, () &#61;> {

return i &#61;&#61;&#61; 20 ?

"" :

&#96;&#96;;

});

}

return result;

}

}

}

}

}

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

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

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

{{ thisIsMe }}

{{ item.text }}

然后修改构建代码&#xff1a;

replace(match, tpl) {

let result &#61; "";

let prefix &#61; "";

// 匹配 thisIsMe 的前缀

tpl &#61; tpl.replace(/{{s*($.*?$)thisIsMes*}}/, (match, p) &#61;> {

prefix &#61; p;

return "";

});

for (let i &#61; 0; i <&#61; 20; i&#43;&#43;) {

result &#43;&#61; "

" &#43; tpl

.replace("wepyhtml-0", "wepyhtml-" &#43; i)

.replace(//g, () &#61;> {

return i &#61;&#61;&#61; 20 ?

"" :

&#96;&#96;;

});

}

return result;

}

至此&#xff0c;渲染问题就解决了。

微信小程序中HTML包含图片

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

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

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

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

onNodeCreate(name, attrs) {

if (name &#61;&#61;&#61; "img") {

attrs["data-src"] &#61; attrs.src;

// 预览图数组

this.previewImgs.push(attrs.src);

// 缩图

attrs.src &#61; resizeImg(attrs.src, 640);

}

}

对应的模板和事件处理逻辑如下&#xff1a;

// 点击小图看大图

imgTap(e) {

wepy.previewImage({

current: e.currentTarget.dataset.src,

urls: this.previewImgs

});

}

微信小程序中HTML包含视频

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

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

相关代码如下&#xff1a;

{

// 点击封面图&#xff0c;播放视频

videoTap(e) {

const nodeId &#61; e.currentTarget.dataset.nodeid;

const context &#61; wepy.createVideoContext("wepyhtml-video-" &#43; nodeId);

context.play();

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

// 需要再调用全屏方法

if (wepy.getSystemInfoSync().platform &#61;&#61;&#61; "android") {

context.requestFullScreen();

}

},

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

// 强制全屏播放

videoPlay(e) {

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

},

// 退出全屏则暂停

videoFullscreenChange(e) {

if (!e.detail.fullScreen) {

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

}

}

}

开源

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

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

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



推荐阅读
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 本文介绍了如何使用JQuery实现省市二级联动和表单验证。首先,通过change事件监听用户选择的省份,并动态加载对应的城市列表。其次,详细讲解了使用Validation插件进行表单验证的方法,包括内置规则、自定义规则及实时验证功能。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • 作为一名新手,您可能会在初次尝试使用Eclipse进行Struts开发时遇到一些挑战。本文将为您提供详细的指导和解决方案,帮助您克服常见的配置和操作难题。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文介绍了如何利用JavaScript或jQuery来判断网页中的文本框是否处于焦点状态,以及如何检测鼠标是否悬停在指定的HTML元素上。 ... [详细]
  • 本文探讨了Hive中内部表和外部表的区别及其在HDFS上的路径映射,详细解释了两者的创建、加载及删除操作,并提供了查看表详细信息的方法。通过对比这两种表类型,帮助读者理解如何更好地管理和保护数据。 ... [详细]
  • 导航栏样式练习:项目实例解析
    本文详细介绍了如何创建一个具有动态效果的导航栏,包括HTML、CSS和JavaScript代码的实现,并附有详细的说明和效果图。 ... [详细]
  • 深入理解Tornado模板系统
    本文详细介绍了Tornado框架中模板系统的使用方法。Tornado自带的轻量级、高效且灵活的模板语言位于tornado.template模块,支持嵌入Python代码片段,帮助开发者快速构建动态网页。 ... [详细]
  • 本文详细介绍了如何使用 Yii2 的 GridView 组件在列表页面实现数据的直接编辑功能。通过具体的代码示例和步骤,帮助开发者快速掌握这一实用技巧。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • Python自动化处理:从Word文档提取内容并生成带水印的PDF
    本文介绍如何利用Python实现从特定网站下载Word文档,去除水印并添加自定义水印,最终将文档转换为PDF格式。该方法适用于批量处理和自动化需求。 ... [详细]
  • 本文介绍如何通过注册表编辑器自定义和优化Windows文件右键菜单,包括删除不需要的菜单项、添加绿色版或非安装版软件以及将特定应用程序(如Sublime Text)添加到右键菜单中。 ... [详细]
author-avatar
侯贵聪_680
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有