mirror of
https://github.com/openfrontio/OpenFrontIO.git
synced 2026-07-02 14:18:21 +00:00
Slider UI change (#2148)
## Description: 1. adds feature so when making single player server, bots amount can be determined by slider and manual 1-1. [video](https://github.com/user-attachments/assets/23186592-2330-478b-8f73-313ca073803b) ## 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: jack_45183 --------- Co-authored-by: evanpelle <evanpelle@gmail.com>
This commit is contained in:
@@ -0,0 +1,137 @@
|
||||
import { LitElement, css, html } from "lit";
|
||||
import { property, query, state } from "lit/decorators.js";
|
||||
import { translateText } from "../Utils";
|
||||
|
||||
export class FluentSlider extends LitElement {
|
||||
static styles = css`
|
||||
:host {
|
||||
display: block;
|
||||
width: 100%;
|
||||
font-family: inherit;
|
||||
}
|
||||
|
||||
.slider-container {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
gap: 4px;
|
||||
width: 100%;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.option-card-title {
|
||||
font-size: 14px; /* match other cards */
|
||||
color: #aaa; /* light gray text */
|
||||
text-align: center;
|
||||
margin: 0 0 4px 0;
|
||||
font-weight: normal;
|
||||
}
|
||||
|
||||
input[type="range"] {
|
||||
width: 100%;
|
||||
max-width: 100%;
|
||||
background-color: transparent;
|
||||
}
|
||||
|
||||
input[type="number"] {
|
||||
width: 60px;
|
||||
background-color: #2d3748;
|
||||
color: #aaa; /* match label color */
|
||||
border: 1px solid #4a5568;
|
||||
text-align: center;
|
||||
border-radius: 4px;
|
||||
font-weight: normal;
|
||||
font-family: inherit;
|
||||
}
|
||||
|
||||
span.editable {
|
||||
cursor: pointer;
|
||||
min-width: 60px;
|
||||
display: inline-block;
|
||||
text-align: center;
|
||||
color: #aaa; /* match label color */
|
||||
font-weight: normal;
|
||||
user-select: none;
|
||||
}
|
||||
`;
|
||||
|
||||
@property({ type: Number }) value = 0;
|
||||
@property({ type: Number }) min = 0;
|
||||
@property({ type: Number }) max = 400;
|
||||
@property({ type: Number }) step = 1;
|
||||
|
||||
@state() private isEditing = false;
|
||||
|
||||
@query("input[type='number']") private numberInput!: HTMLInputElement;
|
||||
|
||||
private handleSliderChange(e: Event) {
|
||||
const target = e.target as HTMLInputElement;
|
||||
this.value = Number(target.value);
|
||||
}
|
||||
|
||||
private handleNumberChange(e: Event) {
|
||||
const target = e.target as HTMLInputElement;
|
||||
let val = Number(target.value);
|
||||
if (val < this.min) val = this.min;
|
||||
if (val > this.max) val = this.max;
|
||||
this.value = val;
|
||||
}
|
||||
|
||||
private handleNumberKeyDown(e: KeyboardEvent) {
|
||||
if (e.key === "Enter") this.isEditing = false;
|
||||
}
|
||||
|
||||
private enableEditing() {
|
||||
this.isEditing = true;
|
||||
this.updateComplete.then(() => this.numberInput?.focus());
|
||||
}
|
||||
|
||||
render() {
|
||||
return html`
|
||||
<div class="slider-container">
|
||||
<input
|
||||
type="range"
|
||||
.min=${this.min}
|
||||
.max=${this.max}
|
||||
.step=${this.step}
|
||||
.value=${String(this.value)}
|
||||
@input=${this.handleSliderChange}
|
||||
/>
|
||||
|
||||
<div class="option-card-title">
|
||||
<span>${translateText("slider.label")}</span>
|
||||
|
||||
${this.isEditing
|
||||
? html`<input
|
||||
type="number"
|
||||
.min=${this.min}
|
||||
.max=${this.max}
|
||||
.value=${this.value}
|
||||
@input=${this.handleNumberChange}
|
||||
@blur=${() => (this.isEditing = false)}
|
||||
@keydown=${this.handleNumberKeyDown}
|
||||
/>`
|
||||
: html`<span
|
||||
class="editable"
|
||||
role="button"
|
||||
tabindex="0"
|
||||
@click=${this.enableEditing}
|
||||
@keydown=${(e: KeyboardEvent) => {
|
||||
if (e.key === "Enter" || e.key === " ") {
|
||||
this.enableEditing();
|
||||
e.preventDefault();
|
||||
}
|
||||
}}
|
||||
>
|
||||
${this.value === 0
|
||||
? translateText("slider.disabled")
|
||||
: this.value}
|
||||
</span>`}
|
||||
</div>
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
}
|
||||
|
||||
customElements.define("fluent-slider", FluentSlider);
|
||||
Reference in New Issue
Block a user