mirror of
https://github.com/KuekHaoYang/KVideo.git
synced 2026-06-01 03:22:03 +08:00
- Added Danmaku settings including API URL, opacity, font size, and display area to PlayerSettings. - Improved button styles and transitions across various UI components for better user experience. - Updated BackToTop, Badge, Button, Card, ConfirmDialog, Input, ModalBackdrop, ModalHeader, SegmentedControl, and Switch components for consistency with Liquid Glass design. - Adjusted package versions in package.json and package-lock.json to reflect version 4.5.0.
81 lines
2.7 KiB
TypeScript
81 lines
2.7 KiB
TypeScript
'use client';
|
|
|
|
import { useRef, useEffect, useState } from 'react';
|
|
|
|
/**
|
|
* SegmentedControl - A switch-style tab component following Liquid Glass design
|
|
*/
|
|
|
|
interface SegmentedControlProps<T extends string> {
|
|
options: { label: string; value: T }[];
|
|
value: T;
|
|
onChange: (value: T) => void;
|
|
className?: string;
|
|
}
|
|
|
|
export function SegmentedControl<T extends string>({
|
|
options,
|
|
value,
|
|
onChange,
|
|
className = '',
|
|
}: SegmentedControlProps<T>) {
|
|
const containerRef = useRef<HTMLDivElement>(null);
|
|
const [indicatorStyle, setIndicatorStyle] = useState({ left: 0, width: 0 });
|
|
|
|
useEffect(() => {
|
|
const updateIndicator = () => {
|
|
if (!containerRef.current) return;
|
|
const activeElement = containerRef.current.querySelector(
|
|
`[data-value="${value}"]`
|
|
) as HTMLElement;
|
|
|
|
if (activeElement) {
|
|
setIndicatorStyle({
|
|
left: activeElement.offsetLeft,
|
|
width: activeElement.offsetWidth,
|
|
});
|
|
}
|
|
};
|
|
|
|
updateIndicator();
|
|
// Update on window resize as well
|
|
window.addEventListener('resize', updateIndicator);
|
|
return () => window.removeEventListener('resize', updateIndicator);
|
|
}, [value, options]);
|
|
|
|
return (
|
|
<div
|
|
ref={containerRef}
|
|
className={`
|
|
relative flex p-1 bg-[var(--glass-bg)] backdrop-blur-xl
|
|
border border-[var(--glass-border)] rounded-[var(--radius-2xl)]
|
|
shadow-[var(--shadow-sm)] ${className}
|
|
`}
|
|
>
|
|
{/* Sliding Indicator */}
|
|
<div
|
|
className="absolute top-1 bottom-1 bg-[var(--accent-color)] rounded-[calc(var(--radius-2xl)-4px)] shadow-[0_2px_8px_rgba(0,122,255,0.3)] transition-all duration-300 cubic-bezier(0.2, 0.8, 0.2, 1)"
|
|
style={{
|
|
left: `${indicatorStyle.left}px`,
|
|
width: `${indicatorStyle.width}px`,
|
|
}}
|
|
/>
|
|
|
|
{/* Segment Buttons */}
|
|
{options.map((option) => (
|
|
<button
|
|
key={option.value}
|
|
data-value={option.value}
|
|
onClick={() => onChange(option.value)}
|
|
className={`
|
|
relative z-10 flex-1 py-2 px-4 text-sm font-semibold transition-colors duration-200
|
|
${value === option.value ? 'text-white' : 'text-[var(--text-color-secondary)] hover:text-[var(--text-color)]'}
|
|
`}
|
|
>
|
|
{option.label}
|
|
</button>
|
|
))}
|
|
</div>
|
|
);
|
|
}
|