Replace native select elements with custom themed SelectDropdown

- New SelectDropdown component with themed styling matching item/talent pickers
- Replaced all 6 native <select> elements (InfoPanel, CampaignView, GearList)
- Consistent appearance across Linux, Mac, and all browsers
- Alegreya font, gold hover highlights, proper theme colors

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
Aaron Wood 2026-04-10 00:32:12 -04:00
parent fbdd9c49ee
commit 9b12482921
5 changed files with 165 additions and 64 deletions

View file

@ -2,6 +2,7 @@ import { useState } from "react";
import type { Gear, GameItem } from "../types"; import type { Gear, GameItem } from "../types";
import ItemPicker from "./ItemPicker"; import ItemPicker from "./ItemPicker";
import CurrencyRow from "./CurrencyRow"; import CurrencyRow from "./CurrencyRow";
import SelectDropdown from "./SelectDropdown";
import styles from "./GearList.module.css"; import styles from "./GearList.module.css";
interface GearListProps { interface GearListProps {
@ -148,16 +149,11 @@ export default function GearList({
onChange={(e) => setCustomName(e.target.value)} onChange={(e) => setCustomName(e.target.value)}
autoFocus autoFocus
/> />
<select <SelectDropdown
className={styles.customSelect} value={customType.charAt(0).toUpperCase() + customType.slice(1)}
value={customType} options={["Weapon", "Armor", "Gear", "Spell"]}
onChange={(e) => setCustomType(e.target.value)} onChange={(v) => setCustomType(v.toLowerCase())}
> />
<option value="weapon">Weapon</option>
<option value="armor">Armor</option>
<option value="gear">Gear</option>
<option value="spell">Spell</option>
</select>
<button className={styles.addBtn} type="submit"> <button className={styles.addBtn} type="submit">
Add Add
</button> </button>

View file

@ -1,6 +1,7 @@
import { useRef, useEffect } from "react"; import { useRef, useEffect } from "react";
import type { Character } from "../types"; import type { Character } from "../types";
import TalentList from "./TalentList"; import TalentList from "./TalentList";
import SelectDropdown from "./SelectDropdown";
import styles from "./InfoPanel.module.css"; import styles from "./InfoPanel.module.css";
const CLASSES = ["Fighter", "Priest", "Thief", "Wizard"]; const CLASSES = ["Fighter", "Priest", "Thief", "Wizard"];
@ -100,31 +101,19 @@ export default function InfoPanel({
<div className={styles.editRow}> <div className={styles.editRow}>
<div className={styles.field}> <div className={styles.field}>
<label className={styles.fieldLabel}>Class</label> <label className={styles.fieldLabel}>Class</label>
<select <SelectDropdown
className={styles.editSelect}
value={character.class} value={character.class}
onChange={(e) => handleField("class", e.target.value)} options={CLASSES}
> onChange={(v) => handleField("class", v)}
{CLASSES.map((c) => ( />
<option key={c} value={c}>
{c}
</option>
))}
</select>
</div> </div>
<div className={styles.field}> <div className={styles.field}>
<label className={styles.fieldLabel}>Ancestry</label> <label className={styles.fieldLabel}>Ancestry</label>
<select <SelectDropdown
className={styles.editSelect}
value={character.ancestry} value={character.ancestry}
onChange={(e) => handleField("ancestry", e.target.value)} options={ANCESTRIES}
> onChange={(v) => handleField("ancestry", v)}
{ANCESTRIES.map((a) => ( />
<option key={a} value={a}>
{a}
</option>
))}
</select>
</div> </div>
</div> </div>
<div className={styles.editRow}> <div className={styles.editRow}>
@ -140,17 +129,11 @@ export default function InfoPanel({
</div> </div>
<div className={styles.field}> <div className={styles.field}>
<label className={styles.fieldLabel}>Alignment</label> <label className={styles.fieldLabel}>Alignment</label>
<select <SelectDropdown
className={styles.editSelect}
value={character.alignment} value={character.alignment}
onChange={(e) => handleField("alignment", e.target.value)} options={ALIGNMENTS}
> onChange={(v) => handleField("alignment", v)}
{ALIGNMENTS.map((a) => ( />
<option key={a} value={a}>
{a}
</option>
))}
</select>
</div> </div>
</div> </div>
<div className={styles.field}> <div className={styles.field}>

View file

@ -0,0 +1,74 @@
.container {
position: relative;
width: 100%;
}
.trigger {
display: flex;
justify-content: space-between;
align-items: center;
width: 100%;
padding: 0.3rem 0.5rem;
background: var(--bg-inset);
border: 1px solid rgba(var(--gold-rgb), 0.15);
border-radius: 4px;
color: var(--text-primary);
font-size: 0.85rem;
font-family: "Alegreya", Georgia, serif;
cursor: pointer;
text-align: left;
}
.trigger:hover {
border-color: rgba(var(--gold-rgb), 0.35);
}
.arrow {
font-size: 0.65rem;
color: var(--text-tertiary);
margin-left: 0.5rem;
}
.dropdown {
position: absolute;
top: 100%;
left: 0;
right: 0;
margin-top: 0.2rem;
background-color: var(--bg-modal);
background-image: var(--texture-surface);
background-size: 256px 256px;
background-repeat: repeat;
border: 1px solid rgba(var(--gold-rgb), 0.3);
border-radius: 4px;
z-index: 200;
max-height: 180px;
overflow-y: auto;
box-shadow:
0 4px 16px rgba(var(--shadow-rgb), 0.5),
inset 0 1px 0 rgba(var(--gold-rgb), 0.06);
scrollbar-width: thin;
scrollbar-color: rgba(var(--gold-rgb), 0.15) transparent;
}
.option {
display: block;
width: 100%;
padding: 0.35rem 0.6rem;
background: none;
border: none;
color: var(--text-primary);
font-size: 0.85rem;
font-family: "Alegreya", Georgia, serif;
cursor: pointer;
text-align: left;
}
.option:hover {
background: rgba(var(--gold-rgb), 0.12);
}
.option.active {
color: var(--gold);
font-weight: 600;
}

View file

@ -0,0 +1,63 @@
import { useState, useRef, useEffect } from "react";
import styles from "./SelectDropdown.module.css";
interface SelectDropdownProps {
value: string;
options: string[];
onChange: (value: string) => void;
className?: string;
}
export default function SelectDropdown({
value,
options,
onChange,
className,
}: SelectDropdownProps) {
const [open, setOpen] = useState(false);
const containerRef = useRef<HTMLDivElement>(null);
useEffect(() => {
if (!open) return;
function handleClickOutside(e: MouseEvent) {
if (
containerRef.current &&
!containerRef.current.contains(e.target as Node)
) {
setOpen(false);
}
}
document.addEventListener("mousedown", handleClickOutside);
return () => document.removeEventListener("mousedown", handleClickOutside);
}, [open]);
return (
<div className={styles.container} ref={containerRef}>
<button
type="button"
className={`${styles.trigger} ${className || ""}`}
onClick={() => setOpen(!open)}
>
<span>{value}</span>
<span className={styles.arrow}>{open ? "▴" : "▾"}</span>
</button>
{open && (
<div className={styles.dropdown}>
{options.map((opt) => (
<button
key={opt}
type="button"
className={`${styles.option} ${opt === value ? styles.active : ""}`}
onClick={() => {
onChange(opt);
setOpen(false);
}}
>
{opt}
</button>
))}
</div>
)}
</div>
);
}

View file

@ -19,6 +19,7 @@ import CharacterCard from "../components/CharacterCard";
import CharacterDetail from "../components/CharacterDetail"; import CharacterDetail from "../components/CharacterDetail";
import RollLog from "../components/RollLog"; import RollLog from "../components/RollLog";
import DiceTray from "../components/DiceTray"; import DiceTray from "../components/DiceTray";
import SelectDropdown from "../components/SelectDropdown";
import styles from "./CampaignView.module.css"; import styles from "./CampaignView.module.css";
const CLASSES = ["Fighter", "Priest", "Thief", "Wizard"]; const CLASSES = ["Fighter", "Priest", "Thief", "Wizard"];
@ -400,35 +401,19 @@ export default function CampaignView() {
</div> </div>
<div className={styles.formField}> <div className={styles.formField}>
<label className={styles.formLabel}>Class</label> <label className={styles.formLabel}>Class</label>
<select <SelectDropdown
className={styles.formSelect}
value={newChar.class} value={newChar.class}
onChange={(e) => options={CLASSES}
setNewChar({ ...newChar, class: e.target.value }) onChange={(v) => setNewChar({ ...newChar, class: v })}
} />
>
{CLASSES.map((c) => (
<option key={c} value={c}>
{c}
</option>
))}
</select>
</div> </div>
<div className={styles.formField}> <div className={styles.formField}>
<label className={styles.formLabel}>Ancestry</label> <label className={styles.formLabel}>Ancestry</label>
<select <SelectDropdown
className={styles.formSelect}
value={newChar.ancestry} value={newChar.ancestry}
onChange={(e) => options={ANCESTRIES}
setNewChar({ ...newChar, ancestry: e.target.value }) onChange={(v) => setNewChar({ ...newChar, ancestry: v })}
} />
>
{ANCESTRIES.map((a) => (
<option key={a} value={a}>
{a}
</option>
))}
</select>
</div> </div>
<div className={styles.formField}> <div className={styles.formField}>
<label className={styles.formLabel}>Max HP</label> <label className={styles.formLabel}>Max HP</label>