innerHTML + bypassSecurityTrustHtml 导致节点重复渲染(innerHTML 中有多个函数时的执行顺序问题)

目录

1.问题

2.思考

2.1 何谓 DomSanitizer?何谓 bypassSecurityTrustHtml()?

2.2 回到问题本身进行分析

3.解决


1.问题

[innerHTML] 和 DomSanitizer 的 bypassSecurityTrustHtml() 搭配使用,导致节点无限重复渲染

// 产生 html代码
render: (row) => {
    return `<div> ${row.rankNum) </div>`;
)

// 获取安全 Html代码
getSafeHtml(html): SafeHtml {
    return this.sanitizer.bypassSecurityTrustHtml(html);
}

// html 里使用 innerHTML
<div class="header-td"
    [innerHTML]="getSafeHtml(render(row))">
</div>

2.思考

2.1 何谓 DomSanitizer?何谓 bypassSecurityTrustHtml()?

DomSanitizer:通过清理在不同 DOM上下文 中安全使用的值,来防止跨站点脚本安全漏洞(XSS)

bypassSecurityTrustHtml():是 DomSanitizer 下面的一个方法,通过接受 html代码svg图片,将他们转换为安全的值,防止出现报错

使用 innerHTML 的时候,可能会出现报错:WARNING: sanitizing HTML stripped some content (see http://g.co/ng/security#xss).

参考博客:Angular4 绑定html内容 警告处理_xiaotuni的专栏,每天进步一点点 -CSDN博客icon-default.png?t=M1L8https://blog.csdn.net/xiaotuni/article/details/77171240

2.2 回到问题本身进行分析

可能性一:

bypassSecurityTrustHtml() 接收的参数应该是 string

此处的 bypassSecurityTrustHtml() 接受了 render(row),render() 方法返回的是 string

重点来了:render() 这个方法 可能 在 bypassSecurityTrustHtml() 之后执行

就是说 render() 还没有返回数据,bypassSecurityTrustHtml() 直接接收了的 render() 这么个空方法

导致 render() 最终在 bypassSecurityTrustHtml() 内部运行,导致重复渲染

可能性二:

innerHTML 接收到的是 getSafeHtml(render(row)) 函数,而不是一段转换完成的 html代码,导致重复渲染

总结:

[innerHTML]、getSafeHtml()、render() 执行顺序不能确定,导致了重复渲染

3.解决

利用 管道 进行 html 代码安全处理,严格控制各个函数的执行顺序

// 产生 html代码
render: (row) => {
    return `<div> ${row.rankNum) </div>`;
)

// 通过定义管道,实现获取安全的 html代码 这个功能
transform(html: string): any {
    return this.sanitizer.bypassSecurityTrustHtml(html);
}

// html 里使用 innerHTML
<div class="header-td"
    [innerHTML]="render(row) | safeHtml">
</div>

执行顺序:

  • 先由 render(row) 生成 html代码
  • 再进入管道处理
  • 最后交由 [innerHTML] 插入节点

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是使用zTree实现一级节点后面增加一个按钮靠右显示的完整代码: HTML代码: ```html <ul id="tree" class="ztree"></ul> ``` CSS样式: ```css .ztree li button { float: right; margin-right: 10px; } ``` JavaScript代码: ```javascript var zNodes = [ { id:1, pId:0, name:"节点1", open:true }, { id:11, pId:1, name:"节点1-1" }, { id:12, pId:1, name:"节点1-2" }, { id:13, pId:1, name:"节点1-3" }, { id:2, pId:0, name:"节点2", open:true }, { id:21, pId:2, name:"节点2-1" }, { id:22, pId:2, name:"节点2-2" }, { id:23, pId:2, name:"节点2-3" }, { id:3, pId:0, name:"节点3", open:true }, { id:31, pId:3, name:"节点3-1" }, { id:32, pId:3, name:"节点3-2" }, { id:33, pId:3, name:"节点3-3" } ]; var setting = { view: { addDiyDom: function(treeId, treeNode) { if (treeNode.level === 1) { var button = document.createElement('button'); button.innerHTML = '操作'; button.onclick = function() { alert('点击了操作按钮'); }; var aObj = $("#" + treeNode.tId + "_a"); aObj.append(button); } } }, data: { simpleData: { enable: true } } }; $(document).ready(function(){ $.fn.zTree.init($("#tree"), setting, zNodes); }); ``` 在上面的代码,我们通过zTree的`addDiyDom`回调函数,在渲染节点,判断当前节点的层级是否为1(即一级节点),如果是,则创建一个按钮元素,并绑定点击事件,最后将按钮元素添加到节点的`<a>`标签。CSS样式,我们将按钮元素的`float`属性设置为`right`,使其靠右显示。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Lyrelion

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值