feat(components): 添加 Switch 组件

- 实现了一个新的 Switch 组件,用于在开启和关闭状态之间切换
- 组件支持 name、checked、disabled 和 onChange 属性
- 使用 Solid.js 的 createEffect、createMemo 和 createSignal 等功能实现内部状态管理
- 通过 CSS 类和属性动态控制组件样式和行为
This commit is contained in:
Vixalie
2025-08-12 22:15:29 +08:00
parent 05aa4bc8e0
commit 2e253ff04c

53
src/components/Switch.tsx Normal file
View File

@@ -0,0 +1,53 @@
import { isNotNil } from 'es-toolkit';
import { Component, createEffect, createMemo, createSignal, mergeProps, Show } from 'solid-js';
interface SwitchProps {
name?: string;
checked?: boolean;
disabled?: boolean;
onChange?: (checked: boolean) => void;
}
const Switcher: Component<SwitchProps> = (props) => {
const mProps = mergeProps<SwitchProps[]>(
{
disabled: false,
},
props,
);
const originalChecked = createMemo(() => mProps.checked);
const [internalChecked, setInternalChecked] = createSignal<boolean | undefined>(undefined);
createEffect(() => {
if (isNotNil(originalChecked()) && originalChecked() !== internalChecked()) {
setInternalChecked(originalChecked());
}
});
const handleClick = () => {
if (mProps.disabled) {
return;
}
setInternalChecked((prev) => !prev);
mProps.onChange?.(internalChecked());
};
return (
<div
aria-disabled={mProps.disabled}
class="inline-flex items-center gap-[1em] cursor-pointer rounded-sm aria-disabled:cursor-not-allowed group"
classList={{ 'bg-primary/38 aria-disabled:bg-on-surface/8': internalChecked() }}>
<div
class="relative h-[1em] w-[1.75em] border border-solid border-primary group-aria-disabled:border-on-surface/8 rounded-sm before:absolute before:translate-x-[2px] before:translate-y-[2px] before:h-[calc(1em-6px)] before:aspect-square before:rounded-xs before:bg-primary before:transition before:duration-200 before:ease-in-out group-aria-disabled:before:bg-on-surface/18"
classList={{ 'before:translate-x-[calc(0.75em+2px)]': internalChecked() }}
onClick={handleClick}
/>
<Show when={isNotNil(mProps.name)}>
<input type="hidden" name={mProps.name} value={internalChecked() ? 'true' : 'false'} />
</Show>
</div>
);
};
export default Switcher;