Element Plus 组件库实现:2. Collapse(2)

前言

上篇文章已经介绍了Collapse组件的大概实现方案及思路,Element Plus 组件库实现:2. Collapse(1) - 掘金 (juejin.cn),本文将简单介绍实现具体实现。

Collapse组件实现

html 复制代码
// 先引入需要用到的接口或方法
<script setup lang="ts">
import { ref, provide, watch } from 'vue';
import { collapseContextKey } from './types';
import type { NameType, CollapseProps, CollaspeEmits } from './types';

defineOptions({
    name: 'YvCollapse'
})
const props = defineProps<CollapseProps>()
const emits = defineEmits<CollaspeEmits>()
</script>

对应上篇文章的设计思路:

  • 在Collapse组件中用一个数组来保存打开的item的name属性
ts 复制代码
const activeNames = ref<NameType[]>(props.modelValue)
// 因为初始值为props.modelValue,来自父组件,所以这里还需要来设置一个监听以保证数据响应式
watch(() => props.modelValue, () => {
    activeNames.value = props.modelValue
})
  • 点击item的时候,先在数组中实现了打开和关闭的状态改变
ts 复制代码
const handleItemClick = (item: NameType) => {
    // 找到当前点击列表的索引值
    const index = activeNames.value.indexOf(item)
    // 先来到手风琴效果分支
    if (props.accordion) {
        // 如果点击的是展开项,那就清空激活数组,否则替换,这样就保证了只有一项被激活
        activeNames.value = [activeNames.value[0] === item ? '' : item]
    } else {
        if (index > -1) {
            // 存在, 删除相应的一项
            activeNames.value.splice(index, 1)
        } else {
            // 不存在, 插入相应的一项
            activeNames.value.push(item)
        }
        // console.log(activeNames.value)
    }
    // 派发事件,让父组件知道数据被改变
    emits('update:modelValue', activeNames.value)
    emits('change', activeNames.value)
}
  • 这个数组将由Collapse组件传递给CollapseItem

这里由于使用的slot,所以不能再使用prop来向子组件传递属性和方法,要用到provideinject依赖注入的方法来传递:

ts 复制代码
// Symbol("collapseContextKey") 创建了一个唯一的符号,用于作为这个注入键
export const collapseContextKey: InjectionKey<CollapseContext> = Symbol("collapseContextKey");
ts 复制代码
// Collapse.vue
provide(collapseContextKey, {
    activeNames,
    // 将方法也传递到CollapseItem中,在点击Item列表时调用
    handleItemClick
})

CollapseItem组件实现

html 复制代码
// CollapseItem.vue
<script setup lang="ts">
import { inject, computed } from 'vue';
import { collapseContextKey } from './types';
import type { CollapseItemProps } from './types';

defineOptions({
    name: "YvCollapseItem"
})
const props = defineProps<CollapseItemProps>()
</script>


<template>
    <div class="yv-collapse-item" :class="{
        'is-disabled': props.disabled
    }">
        <div class="yv-collapse-item__header" :class="{
            'is-disabled': disabled,
            'is-active': isActive

        }" :id="`item-header-${props.name}`" @click="handleClick">
            <slot name="title">{{ title }}</slot>
        </div>
            <div class="yv-collapse-item__wrapper" v-show="isActive">
                // 这里绑定个id属性,使每个列表有唯一的id
                <div class="yv-collapse-item__content" :id="`item-content-${props.name}`">
                    <slot></slot>
                </div>
            </div>
    </div>
</template>

这个时候,子组件要做的事情显而易见了 => 根据父组件Collapse传递过来的状态数组activeNames,决定展示或隐藏相应的列表项,对应上篇文章的设计思路:

  • 然后在Item组件内部,也就是CollapseItem组件内部,判断name是否存在于数组中,使用一个计算属性结合v-show,真正实现打开和关闭
ts 复制代码
// 接受祖先组件传递过来的参数,默认值为undefined
const collapseContext = inject(collapseContextKey, undefined)
// 根据是否被添加到数据进行显示/隐藏处理
const isActive = computed(() => collapseContext?.activeNames.value.includes(props.name))
const handleClick = () => {
    // 禁用状态下当然就直接返回了
    if (props.disabled) { return }
    // 触发点击,并把当前列表独有的name属性传递给Collapse
    collapseContext?.handleItemClick(props.name)
}

这样,两个组件的就基本实现,关于样式和动画,不再赘述,也不是本文重点。

总结

本文通过结合上篇文章Element Plus 组件库实现:2. Collapse(1) - 掘金 (juejin.cn)的方案及思路完成了对Collapse组件和CollapseItem组件的具体实现,需要注意的是,这里的传递数据需要通过provideinject依赖注入的方法来传递。

相关推荐
带娃的IT创业者10 分钟前
开源之魂:Thunderbird 的生存困境与我们的数字主权
开源·邮件客户端·开源生态·非营利组织·thunderbird·数字主权
IvorySQL12 分钟前
开源共建分论坛圆桌讨论:如何真正融入 PostgreSQL 社区?
postgresql·开源·区块链
BY组态12 分钟前
Ricon组态系统技术深度解析:打造高性能Web可视化平台
前端·物联网·iot·web组态·组态
山屿落星辰26 分钟前
Flutter 高级特性实战:动画、自定义绘制、平台通道与 Web 优化
前端·flutter
@菜菜_达1 小时前
jquery.inputmask插件介绍
前端·javascript·jquery
QuZhengRong1 小时前
【Luck-Report】缓存
java·前端·后端·vue·excel
jiayong231 小时前
前端面试题库 - 浏览器与网络篇
前端·网络·面试
Csvn1 小时前
小程序开发:微信小程序与 uni-app 实战指南
前端
小歪不歪我是AI1 小时前
Pi 源码拆解:当一个极简主义的 agent harness 只有 4 个 tool
开源·agent
摸鱼小李上线了2 小时前
vue项目页面添加水印实现方法
前端·javascript·vue.js