obs-ss-plugin-webui/components/Dropdown.tsx
Decobus 3c58ccc5af
Some checks failed
Lint and Build / build (pull_request) Failing after 24s
Add stream deletion functionality and improve UI
- Add delete button to each stream with confirmation modal
- Implement DELETE endpoint that removes sources from OBS before database deletion
- Fix dropdown positioning issue when scrolling by removing scroll offsets
- Change add stream form to use Twitch username instead of full URL
- Automatically calculate Twitch URL from username (https://twitch.tv/{username})
- Add username validation (4-25 chars, alphanumeric and underscores only)
- Improve "View Stream" link visibility with button styling
- Ensure streams list refreshes immediately after deletion

🤖 Generated with [Claude Code](https://claude.ai/code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-07-20 13:16:06 -04:00

150 lines
No EOL
4.2 KiB
TypeScript

'use client';
import { useRef, useEffect, useState } from 'react';
import { createPortal } from 'react-dom';
type DropdownProps = {
options: Array<{ id: number; name: string }>;
activeId: number | null;
onSelect: (id: number) => void;
label: string;
isOpen?: boolean;
onToggle?: (isOpen: boolean) => void;
};
export default function Dropdown({
options,
activeId,
onSelect,
label,
isOpen: controlledIsOpen,
onToggle,
}: DropdownProps) {
const dropdownRef = useRef<HTMLDivElement>(null);
const buttonRef = useRef<HTMLButtonElement>(null);
const [isOpen, setIsOpen] = useState(controlledIsOpen ?? false);
const [dropdownPosition, setDropdownPosition] = useState({ top: 0, left: 0, width: 0 });
const [mounted, setMounted] = useState(false);
useEffect(() => {
setMounted(true);
}, []);
useEffect(() => {
const handleClickOutside = (event: MouseEvent) => {
if (!dropdownRef.current || !buttonRef.current || !(event.target instanceof Node)) return;
if (!dropdownRef.current.contains(event.target) && !buttonRef.current.contains(event.target)) {
if (onToggle) onToggle(false);
else setIsOpen(false);
}
};
if (controlledIsOpen || isOpen) {
document.addEventListener('mousedown', handleClickOutside);
}
return () => {
document.removeEventListener('mousedown', handleClickOutside);
};
}, [controlledIsOpen, isOpen, onToggle]);
useEffect(() => {
const updatePosition = () => {
if ((controlledIsOpen ?? isOpen) && buttonRef.current && mounted) {
const rect = buttonRef.current.getBoundingClientRect();
setDropdownPosition({
top: rect.bottom + 4,
left: rect.left,
width: rect.width
});
}
};
updatePosition();
if ((controlledIsOpen ?? isOpen) && mounted) {
window.addEventListener('scroll', updatePosition, true);
window.addEventListener('resize', updatePosition);
return () => {
window.removeEventListener('scroll', updatePosition, true);
window.removeEventListener('resize', updatePosition);
};
}
}, [controlledIsOpen, isOpen, mounted]);
const activeOption = options.find((option) => option.id === activeId) || null;
const handleSelect = (option: { id: number }) => {
onSelect(option.id);
if (onToggle) onToggle(false);
else setIsOpen(false);
};
const toggleDropdown = () => {
if (onToggle) onToggle(!isOpen);
else setIsOpen((prev) => !prev);
};
const dropdownMenu = (controlledIsOpen ?? isOpen) && mounted ? (
<div
ref={dropdownRef}
className="dropdown-menu"
style={{
position: 'fixed',
top: dropdownPosition.top,
left: dropdownPosition.left,
width: dropdownPosition.width,
zIndex: 999999
}}
>
{options.length === 0 ? (
<div className="dropdown-item text-center">
No teams available
</div>
) : (
options.map((option) => (
<div
key={option.id}
onClick={() => handleSelect(option)}
className={`dropdown-item ${activeOption?.id === option.id ? 'active' : ''}`}
>
{option.name}
</div>
))
)}
</div>
) : null;
return (
<>
<div className="relative w-full">
<button
ref={buttonRef}
type="button"
onClick={toggleDropdown}
className="dropdown-button"
>
<span>
{activeOption ? activeOption.name : label}
</span>
<svg
className={`icon-sm transition-transform duration-200 ${(controlledIsOpen ?? isOpen) ? 'rotate-180' : ''}`}
fill="currentColor"
viewBox="0 0 20 20"
>
<path
fillRule="evenodd"
d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a 1 1 0 01-1.414 0l-4-4a 1 1 0 010-1.414z"
clipRule="evenodd"
/>
</svg>
</button>
</div>
{mounted && typeof document !== 'undefined' && dropdownMenu ?
createPortal(dropdownMenu, document.body) : null
}
</>
);
}