Standardize Radio Button input (#1701)

This commit is contained in:
Matthew Diamant
2025-03-17 19:15:45 -07:00
committed by GitHub
parent ab96e041bb
commit d6c0cc9dc9
6 changed files with 101 additions and 93 deletions

View File

@@ -6,6 +6,7 @@ import { ModelRadioList } from './settings/models/ModelRadioList';
import { Document, ChevronUp, ChevronDown } from './icons';
import type { View } from '../App';
import { BottomMenuModeSelection } from './BottomMenuModeSelection';
import { RadioInput } from './ui/RadioInput';
export default function BottomMenu({
hasMessages,
@@ -96,34 +97,14 @@ export default function BottomMenu({
<div className="">
<ModelRadioList
className="divide-y divide-borderSubtle"
renderItem={({ model, isSelected, onSelect }) => (
<label key={model.alias ?? model.name} className="block cursor-pointer">
<div
className="flex items-center justify-between p-2 text-textStandard hover:bg-bgSubtle transition-colors"
onClick={onSelect}
>
<div>
<p className="text-sm ">{model.alias ?? model.name}</p>
<p className="text-xs text-textSubtle">{model.subtext ?? model.provider}</p>
</div>
<div className="relative">
<input
type="radio"
name="recentModels"
RenderItem={({ model, isSelected, onSelect }) => (
<RadioInput
label={model.alias ?? model.name}
description={model.subtext ?? model.provider}
value={model.name}
checked={isSelected}
onChange={onSelect}
className="peer sr-only"
isChecked={isSelected}
onClick={onSelect}
/>
<div
className="h-4 w-4 rounded-full border border-gray-400 dark:border-gray-500
peer-checked:border-[6px] peer-checked:border-black dark:peer-checked:border-white
peer-checked:bg-white dark:peer-checked:bg-black
transition-all duration-200 ease-in-out"
></div>
</div>
</div>
</label>
)}
/>
<div

View File

@@ -123,7 +123,7 @@ export const BottomMenuModeSelection = () => {
{/* Dropdown Menu */}
{isGooseModeMenuOpen && (
<div className="absolute bottom-[24px] right-0 w-[240px] bg-bgApp rounded-lg border border-borderSubtle">
<div>
<div className="divide-y divide-borderSubtle">
{filterGooseModes(gooseMode, all_goose_modes, previousApproveModel).map((mode) => (
<ModeSelectionItem
key={mode.key}

View File

@@ -1,6 +1,7 @@
import React, { useEffect, useState } from 'react';
import { Gear } from '../../icons';
import { ConfigureApproveMode } from './ConfigureApproveMode';
import { RadioInput } from '../../ui/RadioInput';
export interface GooseMode {
key: string;
@@ -66,6 +67,17 @@ export function filterGooseModes(
});
}
const ConfigureButton = ({ onClick }) => (
<button
onClick={(event) => {
event.stopPropagation();
onClick();
}}
>
<Gear className="w-5 h-5 text-textSubtle hover:text-textStandard" />
</button>
);
interface ModeSelectionItemProps {
currentMode: string;
mode: GooseMode;
@@ -88,61 +100,30 @@ export function ModeSelectionItem({
setChecked(currentMode === mode.key);
}, [currentMode, mode.key]);
const showConfigure =
!isApproveModeConfigure && (mode.key == 'approve' || mode.key == 'smart_approve');
return (
<div>
<div
className="flex items-center justify-between p-2 text-textStandard hover:bg-bgSubtle transition-colors"
<RadioInput
label={mode.label}
description={showDescription ? mode.description : null}
Accessory={() =>
showConfigure ? <ConfigureButton onClick={() => setIsDislogOpen(true)} /> : null
}
isChecked={checked}
onClick={() => handleModeChange(mode.key)}
>
<div>
<h3 className="text-sm font-semibold text-textStandard dark:text-gray-200">
{mode.label}
</h3>
{showDescription && (
<p className="text-xs text-textSubtle dark:text-gray-400 mt-[2px]">
{mode.description}
</p>
)}
</div>
<div className="relative flex items-center gap-3">
{!isApproveModeConfigure && (mode.key == 'approve' || mode.key == 'smart_approve') && (
<button
onClick={() => {
setIsDislogOpen(true);
}}
>
<Gear className="w-5 h-5 text-textSubtle hover:text-textStandard" />
</button>
)}
<input
type="radio"
name="modes"
value={mode.key}
checked={checked}
onChange={() => handleModeChange(mode.key)}
className="peer sr-only"
/>
<div
className="h-5 w-5 rounded-full border border-gray-400 dark:border-gray-500
peer-checked:border-[6px] peer-checked:border-black dark:peer-checked:border-white
peer-checked:bg-white dark:peer-checked:bg-black
transition-all duration-200 ease-in-out"
></div>
</div>
</div>
<div>
<div>
{isDislogOpen ? (
<ConfigureApproveMode
onClose={() => {
setIsDislogOpen(false);
}}
onClose={() => setIsDislogOpen(false)}
handleModeChange={handleModeChange}
currentMode={currentMode}
/>
) : null}
</div>
</div>
</div>
);
}

View File

@@ -5,11 +5,7 @@ import { useHandleModelSelection } from './utils';
import type { View } from '@/src/App';
interface ModelRadioListProps {
renderItem: (props: {
model: Model;
isSelected: boolean;
onSelect: () => void;
}) => React.ReactNode;
RenderItem: React.FC<{ model: Model; isSelected: boolean; onSelect: () => void }>;
className?: string;
}
@@ -29,7 +25,7 @@ export function SeeMoreModelsButtons({ setView }: { setView: (view: View) => voi
);
}
export function ModelRadioList({ renderItem, className = '' }: ModelRadioListProps) {
export function ModelRadioList({ RenderItem, className = '' }: ModelRadioListProps) {
const { recentModels } = useRecentModels();
const { currentModel } = useModel();
const handleModelSelection = useHandleModelSelection();
@@ -53,13 +49,14 @@ export function ModelRadioList({ renderItem, className = '' }: ModelRadioListPro
return (
<div className={className}>
{recentModels.map((model) =>
renderItem({
model,
isSelected: selectedModel === model.name,
onSelect: () => handleRadioChange(model),
})
)}
{recentModels.map((model) => (
<RenderItem
key={model.name}
model={model}
isSelected={selectedModel === model.name}
onSelect={() => handleRadioChange(model)}
/>
))}
</div>
);
}

View File

@@ -137,7 +137,7 @@ export function RecentModelsRadio({ setView }: { setView: (view: View) => void }
<div className="px-8">
<div className="space-y-2">
<ModelRadioList
renderItem={({ model, isSelected, onSelect }) => (
RenderItem={({ model, isSelected, onSelect }) => (
<label key={model.name} className="flex items-center py-2 cursor-pointer">
<div className="relative mr-4">
<input

View File

@@ -0,0 +1,49 @@
import React from 'react';
export interface RadioInputProps {
label: string;
description: string | null;
Accessory?: React.FC;
value: string;
isChecked: boolean;
onClick: () => void;
}
export const RadioInput = ({
label,
description,
Accessory,
value,
isChecked,
onClick,
}: RadioInputProps) => {
return (
<div className="block cursor-pointer" onClick={onClick}>
<div
className="flex items-center justify-between p-2 text-textStandard hover:bg-bgSubtle transition-colors"
onClick={onClick}
>
<div>
<p className="text-sm ">{label}</p>
<p className="text-xs text-textSubtle">{description}</p>
</div>
<div className="relative flex items-center gap-3">
{Accessory ? <Accessory /> : null}
<input
type="radio"
name="recentModels"
value={value}
checked={isChecked}
onChange={onClick}
className="peer sr-only"
/>
<div
className="h-4 w-4 rounded-full border border-gray-400 dark:border-gray-500
peer-checked:border-[6px] peer-checked:border-black dark:peer-checked:border-white
peer-checked:bg-white dark:peer-checked:bg-black
transition-all duration-200 ease-in-out"
></div>
</div>
</div>
</div>
);
};