likes
comments
collection
share

在「同一路由」下,如何实现「组件切换」?单一路由下,常会出现不同组件间跳转的情况。对于这一常见的业务场景,基于便捷操作的

作者站长头像
站长
· 阅读数 20

需求来源

常见的 管理类项目 业务场景中,常需在单一路由下,在多个组件间切换 基于上述需求,首先想到的处理方式,可能是 定义多个组件结构,并通过 v-if 控制 显示与隐藏

<template>
  <div>
    <component-1 v-if="curComponent === 'c1'" />
    <component-2 v-else-if="curComponent === 'c2'" />
  </div>
</template>

上述做法,虽同样能实现业务需求,但冗杂的DOM结构代码,并不利于后期维护

那么,是否存在某种写法,可以简化上述的结构呢? 通过 Vue2官方文档,查询到其内部定义了一个 动态组件;该组件通过 is 参数,来决定哪个组件被渲染

<!-- 动态组件由 vm 实例的 `componentId` property 控制 -->  
<component :is="componentId"></component>  
  
<!-- 也能够渲染注册过的组件或 prop 传入的组件 -->  
<component :is="$options.components.child"></component>

考虑到 组件重用组件可扩展性,以 混入 思路 构建组件,是再好不过的选择 PS:虽实际构建的仍是组件,但最终是使用mixins方式引入。较常见的components方式引入,前者更有利于对全局DOM结构进行调整(例如额外包裹 keep-alive,参照文末《重构DOM结构》)

基础写法

<template>
  <component
    :is="curComponent"
    :enter-params="enterParams"
    @forward="forward"
  />
</template>

<script>
export default {
  data() {
    return {
      curComponent: 'List',
      enterParams: {},
    }
  },
  methods: {
    forward(comp, data) {
      this.curComponent = comp
      this.enterParams = data || {}
    },
  },
}
</script>

参数及方法解读

  1. curComponententerParams 分别存储 当前渲染组件名称当前组件所需参数集合
  2. forward 用于 组件切换。用法示例:this.$emit('forward', 'Detail', { id: 'xxx' })

功能改进

问题一:非直接子组件,不便使用 forward方法

方式一:为 孙组件实例 挂载事件

<!-- 子组件 -->
<grand-comp @forward="(curComp, enterParams) => { $emit('forward', curComp, enterParams) }">

方式二:使用 全局事件总线

<!-- 为本次定义的混入式组件,添加 额外操作 -->
beforeMount() {
  // 注册事件
  this.$eventbus.$on('forward', this.forward)
},
beforeDestroy() {
  // 销毁事件
  this.$eventbus.$off('forward', this.forward)
}
<!-- 孙组件,使用方式 -->
this.$eventbus.$emit('forward', 'HistoryDetail', { id: 'xxx' })

方式选择

处理二 本身利用了 全局事件 的优势,无需额外在 孙组件 绑定事件,便可实现通信;其较 处理一 有着明显的优势

问题二:返回操作不便捷

场景描述

现实业务场景中,组件的跳转方向,并不局限于

  • 台账页 -> 详情页、编辑页等
  • 详情页、编辑页等 -> 台账页 ……

某些时候下,还存在

  • 详情页 -> 历史详情页(下称前者
  • 历史详情页 -> 详情页(下称后者) ……

得益于 详情数据 往往会存储 历史详情ID集合 数据,前者的跳转较为方便

然而,受限于 历史详情数据 往往不包含 详情ID;为达跳转目的,往往需要在 历史详情页面 保存 详情ID等数据;这也导致了 不必要数据的混入,不符合 高内聚低耦合 要求

实现思路

网页路由 得到思路,可以借助 数据缓存,保存 历史跳转信息

代码新增内容

<template>
  <component
    xxx
    @back="back"
    @go="go"
  />
</template>

<script>
  export default {
    data() {
      return {
        xxx
        history: [],
        curIndex: -1,
      }
    },
    watch: {
      curIndex(index) {
        const { component, params, extraParams } = this.history[index]
        Object.assign(this, {
          curComponent: component,
          enterParams: { ...params, ...extraParams },
        })
      },
    },
    beforeMount() {
      // 为跳转历史补充首跳信息
      this.curComponent &&
        this.history.push({
          component: this.curComponent,
          params: this.enterParams,
          extraParams: {},
        })
      this.curIndex = this.history.length - 1

      xxx
    },
    beforeDestroy() {
      xxx
    },
    methods: {
      forward(comp, data, preExtraData) {
        // 跳转前,补充额外数据(例如:补充tab名称,以供返回时跳转指定tab)
        Object.assign(this.history[this.curIndex].extraParams, preExtraData)

        // 存储跳转信息(默认覆盖curIndex之后的数据)
        this.history.splice(++this.curIndex, Infinity, {
          component: comp,
          params: data ?? {},
          extraParams: {},
        })
      },
      back(data) {
        this.curIndex = Math.max(0, this.curIndex - 1)
        // 补充数据
        Object.assign(this.history[this.curIndex].extraParams, data)
      },
      go(data) {
        this.curIndex = Math.min(this.history.length - 1, this.curIndex + 1)
        // 补充数据
        Object.assign(this.history[this.curIndex].extraParams, data)
      },
    },
  }
</script>

新增内容解析

参数与方法解读
  • history 用于缓存 已跳转组件相关信息(包含 componentparamsextraParams
  • curIndex 表示当前组件相关信息 于history所处索引值
  • back方法 go方法分别用于 返回前一组件前往后一组件(前后关系由相对history索引而来);二者思路均来自于 Vue-Router 同名方法,为其简化版本;二者均仅有一个参数 data,用于 为待返回/前往组件 补充参数

使用示例

基础混入
<script>
  // 引入混入
  import CompBasicMixin from '@/mixin/compBasicMixin'
  // 引入组件
  import List from './list'
  import EditContainer from './components/editContainer'
  import DetailContainer from './components/detailContainer'
  
  export default {
    components: { List, EditContainer, DetailContainer },
    mixins: [CompBasicMixin],
  }
</script>
forward方法
// 基础使用
this.$emit('forward', 'EditContainer', { id: 'xxxxxxx' })
this.$eventbus.$emit('forward', 'DetailContainer', { id: 'xxxxxxx' })

// 特殊使用
// - 当页面存在tabs时,为使得返回时能跳转先前tab,常常将tabName存储到待跳转组件中,以供返回使用
// - 该操作方法,所存在的问题便是:使得无关数据杂糅在组件当中,加剧了组间间的耦合
// - 因而,在forward方法中,额外增加了preExtraData参数
this.$emit('forward', 'xxx', { xxx }, { tabName: 'xxx' })
this.$eventbus.$emit('forward', 'xxx', { xxx }, { tabName: 'xxx' })
back方法

使用场景: 某些页面中,存在 详情页面 跳转 历史详情页面 等情况;此时,返回详情页面 功能存在细节考量 若使用forward方法,则需 额外缓存详情页面 组件名称ID等相关参数 而使用 back方法,则无需关心相关参数(特殊使用除外)

// 基础使用(可用于部分取代 $emit('forward', 'List'))
this.$emit('back')
this.$eventbus('back')

// 特殊使用(绑定参数)
this.$emit('back', { tabName: 'xxx' })	// 仅作示例,不建议以此绑定tabName,建议使用forward方法的额外参数绑定
this.$eventbus.$emit('back', { tabName: 'xxx' })
go方法

back方法

重构DOM结构
<template>
  <!-- 增加缓存机制 -->
  <keep-alive :include="['xxx']">
    <component
      :is="curComponent"
      :enter-params="enterParams"
      @forward="forward"
      @back="back"
      @go="go"
    />
  </keep-alive>
</template>
转载自:https://juejin.cn/post/7389446404190388243
评论
请登录