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组件的样式反而比功能复杂一些,但样式不是本组件库的主要任务,这里不再一一赘述。

相关推荐
一袋米扛几楼9810 分钟前
【网络安全】SIEM -Security Information and Event Management 工具是什么?
前端·安全·web安全
小陈工22 分钟前
2026年4月7日技术资讯洞察:下一代数据库融合、AI基础设施竞赛与异步编程实战
开发语言·前端·数据库·人工智能·python
Cobyte31 分钟前
3.响应式系统基础:从发布订阅模式的角度理解 Vue2 的数据响应式原理
前端·javascript·vue.js
竹林81834 分钟前
从零到一:在React前端中集成The Graph查询Uniswap V3池数据实战
前端·javascript
Mintopia42 分钟前
别再迷信"优化":大多数性能问题根本不在代码里
前端
倾颜42 分钟前
接入 MCP,不一定要先平台化:一次 AI Runtime 的实战取舍
前端·后端·mcp
军军君0144 分钟前
Three.js基础功能学习十八:智能黑板实现实例五
前端·javascript·vue.js·3d·typescript·前端框架·threejs
恋猫de小郭1 小时前
Android 上为什么主题字体对 Flutter 不生效,对 Compose 生效?Flutter 中文字体问题修复
android·前端·flutter
Moment1 小时前
AI全栈入门指南:一文搞清楚NestJs 中的 Controller 和路由
前端·javascript·后端
禅思院1 小时前
前端架构演进:基于AST的常量模块自动化迁移实践
前端·vue.js·前端框架