Files
OpenFrontIO/src/client/components/ToggleInputCard.ts
T
Mattia Migliorini b74e5c84e0 Fix height of ToggleInputCard in Lobby Options (#3201)
## Description:

Fixes height of ToggleInputCard elements in order to force consistency,
i.e. same height of elements in the same row.

### Before:

<img width="762" height="581" alt="Screenshot 2026-02-13 at 17 34 57"
src="https://github.com/user-attachments/assets/23a31fbc-880e-428c-a03d-0b49e4de00dc"
/>


### After - Single Player:

<img width="756" height="792" alt="image"
src="https://github.com/user-attachments/assets/62eaaa34-97f7-40ff-9291-c31a820b826a"
/>


### After - Private Lobby:

<img width="758" height="792" alt="image"
src="https://github.com/user-attachments/assets/37b4d996-5d81-4f76-b95f-3e64707f180e"
/>


### Behavior when toggling the highest element on the row:


https://github.com/user-attachments/assets/6ff26902-2f3c-474f-8bde-0eddcacf9570


## Please complete the following:

- [x] I have added screenshots for all UI updates
- [x] I process any text displayed to the user through translateText()
and I've added it to the en.json file
- [x] I have added relevant tests to the test directory
- [x] I confirm I have thoroughly tested these changes and take full
responsibility for any bugs introduced

## Please put your Discord username so you can be contacted if a bug or
regression is found:

deshack_82603
2026-02-13 20:02:25 -08:00

174 lines
5.9 KiB
TypeScript

import { LitElement, PropertyValues, html, nothing } from "lit";
import { customElement, property } from "lit/decorators.js";
import { translateText } from "../Utils";
const ACTIVE_CARD =
"bg-blue-500/20 border-blue-500/50 shadow-[0_0_15px_rgba(59,130,246,0.2)]";
const INACTIVE_CARD =
"bg-white/5 border-white/10 hover:bg-white/10 hover:border-white/20";
const INPUT_CLASS =
"w-full text-center rounded bg-black/60 text-white text-sm font-bold border border-white/20 focus:outline-none focus:border-blue-500 p-1 my-1";
const CARD_LABEL_CLASS =
"text-xs uppercase font-bold tracking-wider leading-tight break-words hyphens-auto";
function cardClass(active: boolean, extra = ""): string {
return `w-full h-full rounded-xl border cursor-pointer transition-all duration-200 active:scale-95 ${extra} ${active ? ACTIVE_CARD : INACTIVE_CARD}`;
}
@customElement("toggle-input-card")
export class ToggleInputCard extends LitElement {
@property({ attribute: false }) labelKey = "";
@property({ type: Boolean, attribute: false }) checked = false;
@property({ attribute: false }) inputId?: string;
@property({ attribute: false }) inputType = "number";
@property({ attribute: false }) inputMin?: number | string;
@property({ attribute: false }) inputMax?: number | string;
@property({ attribute: false }) inputStep?: number | string;
@property({ attribute: false }) inputValue?: number | string;
@property({ attribute: false }) inputAriaLabel?: string;
@property({ attribute: false }) inputPlaceholder?: string;
@property({ attribute: false }) defaultInputValue?: number | string;
@property({ attribute: false }) minValidOnEnable?: number;
@property({ attribute: false }) onToggle?: (
checked: boolean,
value: number | string | undefined,
) => void;
@property({ attribute: false }) onInput?: (e: Event) => void;
@property({ attribute: false }) onChange?: (e: Event) => void;
@property({ attribute: false }) onKeyDown?: (e: KeyboardEvent) => void;
createRenderRoot() {
return this;
}
protected updated(changedProperties: PropertyValues<this>) {
if (!changedProperties.has("checked")) return;
const previousChecked = changedProperties.get("checked");
if (previousChecked === false && this.checked) {
const input = this.querySelector("input");
if (input) {
input.focus();
input.select();
}
}
}
private toOptionalNumber(
value: number | string | undefined,
): number | undefined {
if (typeof value === "number") {
return Number.isFinite(value) ? value : undefined;
}
if (typeof value === "string") {
const trimmed = value.trim();
if (!trimmed) return undefined;
const numeric = Number(trimmed);
return Number.isFinite(numeric) ? numeric : undefined;
}
return undefined;
}
private resolveValueOnEnable(): number | string | undefined {
const currentValue = this.inputValue;
if (
currentValue === undefined ||
currentValue === null ||
currentValue === ""
) {
return this.defaultInputValue;
}
if (this.minValidOnEnable === undefined) {
return currentValue;
}
const numericValue = this.toOptionalNumber(currentValue);
if (numericValue === undefined || numericValue < this.minValidOnEnable) {
return this.defaultInputValue;
}
return numericValue;
}
private emitToggle() {
const nextChecked = !this.checked;
const nextValue = nextChecked ? this.resolveValueOnEnable() : undefined;
this.onToggle?.(nextChecked, nextValue);
}
private handleCardClick = () => {
this.emitToggle();
};
render() {
return html`
<div class="${cardClass(this.checked, "relative overflow-hidden")}">
<button
type="button"
aria-pressed=${this.checked}
@click=${this.handleCardClick}
class="w-full h-full p-3 flex flex-col items-center justify-between gap-2 focus:outline-none"
>
<div
class="w-5 h-5 rounded border flex items-center justify-center transition-colors mt-1 ${this
.checked
? "bg-blue-500 border-blue-500"
: "border-white/20 bg-white/5"}"
>
${this.checked
? html`<svg
xmlns="http://www.w3.org/2000/svg"
class="h-3 w-3 text-white"
viewBox="0 0 20 20"
fill="currentColor"
>
<path
fill-rule="evenodd"
d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z"
clip-rule="evenodd"
/>
</svg>`
: ""}
</div>
${this.checked
? html`<div class="h-[30px] my-1"></div>`
: html`<div class="h-[2px] w-4 rounded my-3 bg-white/10"></div>`}
<span
class="${CARD_LABEL_CLASS} text-center ${this.checked
? "text-white"
: "text-white/60"}"
>
${translateText(this.labelKey)}
</span>
</button>
${this.checked
? html`
<div
class="absolute left-3 right-3 top-1/2 -translate-y-1/2 z-10"
>
<input
type=${this.inputType}
id=${this.inputId ?? nothing}
min=${this.inputMin ?? nothing}
max=${this.inputMax ?? nothing}
step=${this.inputStep ?? nothing}
.value=${String(this.inputValue ?? "")}
class=${INPUT_CLASS}
aria-label=${this.inputAriaLabel ?? nothing}
placeholder=${this.inputPlaceholder ?? nothing}
@input=${this.onInput}
@change=${this.onChange}
@keydown=${this.onKeyDown}
/>
</div>
`
: nothing}
</div>
`;
}
}