自定义指令-优化v-if和v-show上的使用

v-show-if

背景

像我们在开发vue过程,有时候会遇到这样的问题,在初始化渲染时候,并不希望某个组件渲染,当需要时候再渲染,但不需要又不希望DOM把被删除,导致下次需要时候重新渲染DOM节点。即是希望拥有v-if初始化下不渲染的能力,也要v-show在渲染后不被销毁的能力

使用场景

像很多时候,在使用组件库的开发上,使用tabs,我们就很容易写出如下的代码:

xml 复制代码
<template>
  <a-tabs v-model:activeKey="activeKey">
    <a-tab-pane key="1" tab="Tab 1">Content of Tab Pane 1</a-tab-pane>
    <a-tab-pane key="2" tab="Tab 2" force-render>Content of Tab Pane 2</a-tab-pane>
    <a-tab-pane key="3" tab="Tab 3">Content of Tab Pane 3</a-tab-pane>
  </a-tabs>
</template>
<script lang="ts" setup>
import { ref } from 'vue';
const activeKey = ref('1');
</script>

在tabs下的,每一个pane都是已经被渲染出来的,但有时候我们并不是很希望渲染这部分的,这是我们会采用 v-if 去防止第一个渲染,这时候就会得到如下的代码:

xml 复制代码
<template>
  <a-tabs v-model:activeKey="activeKey">
    <a-tab-pane v-if="activeKey === '1'" key="1" tab="Tab 1">Content of Tab Pane 1</a-tab-pane>
    <a-tab-pane v-if="activeKey === '2'" key="2" tab="Tab 2" force-render>Content of Tab Pane 2</a-tab-pane>
    <a-tab-pane v-if="activeKey === '3'" key="3" tab="Tab 3">Content of Tab Pane 3</a-tab-pane>
  </a-tabs>
</template>
<script lang="ts" setup>
import { ref } from 'vue';
const activeKey = ref('1');
</script>

但在这种情况下,我们就会发现,在v-if切换时候,DOM节点被删除了,会造成一定的性能。因此,我们希望有个指令能实现首次不渲染,需要时候渲染,并在下次为falsy时候不清除DOM节点

xml 复制代码
<template>
  <a-tabs v-model:activeKey="activeKey">
    <a-tab-pane v-show-if="activeKey === '1'" key="1" tab="Tab 1">Content of Tab Pane 1</a-tab-pane>
    <a-tab-pane v-show-if="activeKey === '2'" key="2" tab="Tab 2" force-render>Content of Tab Pane 2</a-tab-pane>
    <a-tab-pane v-show-if="activeKey === '3'" key="3" tab="Tab 3">Content of Tab Pane 3</a-tab-pane>
  </a-tabs>
</template>
<script lang="ts" setup>
import { ref } from 'vue';
const activeKey = ref('1');
</script>

设计

1、挂载时候判断是否挂载注释节点,还是渲染节点

2、在下一次渲染时候触发修改

相关代码实现

ini 复制代码
export default {
  mounted(el, binding, vnode) {
    // 初始值为空情况
    if (!binding.value) {
      // 创建一个v-show-if的注释节点
      el.__showIfComment = document.createComment(' v-show-if ');
      // 初始为 false,用注释节点替代
      vnode.el = el.__showIfComment;
      el.__showIfInitialHidden = true;
      replaceNode(el, el.__showIfComment);
    }
    // 初始值非空情况
    if (binding.value) {
      el.style.display = '';
    }
  },
  updated(el, binding, vnode) {
    if (binding.value === binding.oldValue) return;
​
    if (binding.value) {
      // 从 false 变为 true
      if (el.__showIfInitialHidden) {
        vnode.el = el;
        replaceNode(el.__showIfComment, el);
        el.__showIfInitialHidden = false;
      }
      el.style.display = '';
    } else {
      // 从 true 变为 false
      el.style.display = 'none';
    }
  },
  beforeUnmount(el) {
    // 清理注释节点
    if (el.__showIfComment && el.__showIfComment.parentNode) {
      el.__showIfComment.parentNode.removeChild(el.__showIfComment);
    }
  },
};
​
function replaceNode(oldNode, newNode) {
  if (oldNode.parentNode) {
    oldNode.parentNode.replaceChild(newNode, oldNode);
  }
}

总结

在项目开发过程,难免会遇到上诉描述的问题。本指令v-show-if因此被设计出来,但在使用上还是会存在一定的局限性的,还是需要和业务系统绑定。

相关推荐
3824278276 分钟前
Edge开发者工具:保留日志与禁用缓存详解
java·前端·javascript·python·selenium
web小白成长日记17 分钟前
什么是margin重叠,如何解决
前端·css·html·css3
java porter25 分钟前
系统架构设计之单例模式(下)
开发语言·javascript·单例模式
凌乱风雨121129 分钟前
从源码角度解析C++20新特性如何简化线程超时取消
前端·算法·c++20
两个西柚呀31 分钟前
每日前端面试题-css塌陷
前端·css
IT_陈寒34 分钟前
Vite 5大实战优化技巧:让你的开发效率提升200%|2025前端工程化指南
前端·人工智能·后端
C_心欲无痕37 分钟前
react - createPortal魔法传送门
javascript·vue.js·react.js
前端小L39 分钟前
双指针专题(五):灵活的起跳——「无重复字符的最长子串」
javascript·算法·双指针与滑动窗口
未来之窗软件服务43 分钟前
幽冥大陆(八十八 ) 操作系统应用封装技术C#自解压 —东方仙盟练气期
java·前端·c#·软件打包·仙盟创梦ide·东方仙盟·阿雪技术观
靓仔建1 小时前
在Electron用npm install 失败。
javascript·electron·npm