Element Plus 组件库实现:9. Switch组件

前言

Switch组件,顾名思义,就像现实生活中的开关一样,允许用户在两种状态之间切换。在数字界面中,这通常意味着一个按钮或滑块,用户可以通过点击或拖动来切换其状态。无论是在移动应用、网页还是桌面软件中,Switch组件都因其直观性和易用性而受到开发者和用户的青睐。本文将简单介绍Switch组件的实现。

组件分析

在原生HTML中,很难找到和Switch组件相似的标签,但是如果你去看过类似Element Plus等组件库的源码,你会发现Switch内部其实是由一个checkbox(复选框)实现的,checkbox允许用户选择多个选项,每个选项都是独立的,用户可以选择一个、多个或者都不选。而Switch组件设计的初衷也就是为了提供一个在两种状态之间简单切换的控件,这就与checkbox不谋而合了。

看到这里,你也许会问,为什么不适用radio(单选框)呢?相比于checkbox,radio用于在一组选择中选择一个,他要求的是用户从一组互斥的选项中做出选择,那么这样就与Switch组件的设计目的不相符了。对比之下,checkbox更适合作为Switch组件的内部实现。

需求分析

一个Switch组件需要的基本功能其实并不多:

  • 开/关两种状态
  • 不同状态对应的值以及文字描述
  • 样式实现

通过Switch组件的需求分析,我们不难想到,前两个需求是比较容易实现的,样式现在反而成了这个组件比较难的一块儿。

确定方案

  • 属性
ts 复制代码
export type SwitchValueType = boolean | number | string

export interface SwitchProps {
  // 实现v-model必备属性
  modelValue: SwitchValueType
  disabled?: boolean
  // 打开状态下的文字描述
  activeText?: string
  // 关闭状态下的文字描述
  inactiveText?: string
  // 打开状态下的值
  activeValue?: SwitchValueType
  // 关闭状态下的值
  inactiveValue?: SwitchValueType
  name?: string
  // input 的 id
  id?: string
  size?: 'small' | 'large'
}
  • 事件
ts 复制代码
export interface SwitchEmits {
   // 切换状态改变派发事件 
  (e: 'change', value: SwitchValueType): void
  // 结合modelValue实现v-model
  (e: 'update:modelValue', value: SwitchValueType): void
}
  • 组件
html 复制代码
<template>
    <div class="yv-switch">
        <input class="yv-switch__input" />
        <div class="yv-switch__core">
            <div class="yv-switch__core-inner">
                <span class="yv-switch__core-inner-text"></span>
            </div>
            <div class="yv-switch__core-action"></div>
        </div>
    </div>
</template>

代码实现

  • 开/关两种状态
ts 复制代码
const innerValue = ref(props.modelValue)
// 是否被选中
const checked = computed(() => innerValue.value === props.activeValue)
const switchValue = () => {
    if (props.disabled) return
    const newValue = checked.value ? props.inactiveValue : props.activeValue
    innerValue.value = newValue
    emits("update:modelValue", newValue)
    emits("change", newValue)
}
  • 不同状态对应的值以及文字描述
html 复制代码
<div class="yv-switch__core-inner">
    <span class="yv-switch__core-inner-text" v-if="activeText || inactiveText">
        {{ checked ? activeText : inactiveText }}
    </span>
</div>

不同状态对应不同的描述

  • 自定义开/关对应的值
ts 复制代码
const props = withDefaults(defineProps<SwitchProps>(), {
    activeValue: true,
    inactiveValue: false
})

默认为true和false,使用时可自定义,如: <Switch v-model="test" activeValue="right" inactiveValue="wrong"/>

总结

本文简单介绍了一个Switch组件的基本实现,剩下的就是样式实现,相比于之前的其他组件,Switch组件的样式反而比功能复杂一些,但样式不是本组件库的主要任务,这里不再一一赘述。

相关推荐
IT_陈寒8 小时前
Redis性能翻倍的5个冷门技巧,90%开发者都不知道第3个!
前端·人工智能·后端
T***u3338 小时前
前端框架在性能优化中的实践
javascript·vue.js·前端框架
wei_shuo8 小时前
100% AI 写的开源项目三周多已获得 800 star 了
开源·开发者·sealos
jingling5559 小时前
vue | 在 Vue 3 项目中集成高德地图(AMap)
前端·javascript·vue.js
油丶酸萝卜别吃9 小时前
Vue3 中如何在 setup 语法糖下,通过 Layer 弹窗组件弹出自定义 Vue 组件?
前端·vue.js·arcgis
快乐的学习15 小时前
开源相关术语及提交commit关键字总结
驱动开发·开源
J***Q29216 小时前
Vue数据可视化
前端·vue.js·信息可视化
ttod_qzstudio18 小时前
深入理解 Vue 3 的 h 函数:构建动态 UI 的利器
前端·vue.js
_大龄18 小时前
前端解析excel
前端·excel
一叶茶18 小时前
移动端平板打开的三种模式。
前端·javascript