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