import type { TemplateResult } from "lit"; import { html } from "lit"; import { customElement, state } from "lit/decorators.js"; import { UserMeResponse } from "../core/ApiSchemas"; import { Cosmetics, Skin } from "../core/CosmeticSchemas"; import { PATTERN_KEY, USER_SETTINGS_CHANGED_EVENT, UserSettings, } from "../core/game/UserSettings"; import { PlayerPattern } from "../core/Schemas"; import { BaseModal } from "./components/BaseModal"; import "./components/CosmeticButton"; import "./components/NotLoggedInWarning"; import { modalHeader } from "./components/ui/ModalHeader"; import { fetchCosmetics, getPlayerCosmetics, resolveCosmetics, ResolvedCosmetic, resolvedToPlayerPattern, } from "./Cosmetics"; import { translateText } from "./Utils"; @customElement("territory-patterns-modal") export class TerritoryPatternsModal extends BaseModal { protected routerName = "territory-patterns"; public previewButton: HTMLElement | null = null; @state() private selectedPattern: PlayerPattern | null; @state() private selectedColor: string | null = null; @state() private selectedSkinName: string | null = null; @state() private search = ""; private cosmetics: Cosmetics | null = null; private userSettings: UserSettings = new UserSettings(); private userMeResponse: UserMeResponse | false = false; private _onPatternSelected = async () => { await this.updateFromSettings(); this.refresh(); }; connectedCallback() { super.connectedCallback(); document.addEventListener( "userMeResponse", (event: CustomEvent) => { this.onUserMe(event.detail); }, ); window.addEventListener( `${USER_SETTINGS_CHANGED_EVENT}:${PATTERN_KEY}`, this._onPatternSelected, ); } disconnectedCallback() { super.disconnectedCallback(); window.removeEventListener( `${USER_SETTINGS_CHANGED_EVENT}:${PATTERN_KEY}`, this._onPatternSelected, ); } private async updateFromSettings() { const cosmetics = await getPlayerCosmetics(); this.selectedPattern = cosmetics.pattern ?? null; this.selectedColor = cosmetics.color?.color ?? null; this.selectedSkinName = cosmetics.skin?.name ?? null; } async onUserMe(userMeResponse: UserMeResponse | false) { this.userMeResponse = userMeResponse; this.cosmetics = await fetchCosmetics(); await this.updateFromSettings(); this.refresh(); } private includedInSearch(name: string): boolean { const displayName = name.replace(/_/g, " "); return displayName.toLowerCase().includes(this.search.toLowerCase()); } private handleSearch(event: Event) { this.search = (event.target as HTMLInputElement).value; } /** Combined patterns + skins grid. To the user they're the same: "skins". */ private renderSkinGrid(): TemplateResult { const items = resolveCosmetics( this.cosmetics, this.userMeResponse, null, ).filter( (r) => (r.type === "pattern" || r.type === "skin") && r.relationship === "owned" && (r.cosmetic === null ? !this.search : this.includedInSearch(r.cosmetic.name)), ); return html`
${items.map((r) => { const isSelected = r.type === "pattern" ? (r.cosmetic === null && this.selectedPattern === null) || (r.cosmetic !== null && this.selectedPattern?.name === r.cosmetic.name && (this.selectedPattern?.colorPalette?.name ?? null) === (r.colorPalette?.name ?? null)) : (() => { const skinName = (r.cosmetic as Skin | null)?.name ?? null; return ( (skinName === null && this.selectedSkinName === null) || (skinName !== null && this.selectedSkinName === skinName) ); })(); return html` this.selectCosmetic(rc)} > `; })}
`; } protected renderHeaderSlot() { return html`
${modalHeader({ title: translateText("territory_patterns.title"), onBack: () => this.close(), ariaLabel: translateText("common.back"), rightContent: html``, })}
`; } protected renderBody() { return html`
{ this.close(); window.showPage?.("page-item-store"); }} >
${this.renderSkinGrid()}
`; } protected async onOpen(): Promise { await this.refresh(); } protected onClose(): void { this.search = ""; } private selectCosmetic(resolved: ResolvedCosmetic) { if (resolved.type === "pattern") { this.selectPattern(resolvedToPlayerPattern(resolved)); } else if (resolved.type === "skin") { this.selectSkin((resolved.cosmetic as Skin | null)?.name ?? null); } } private selectSkin(skinName: string | null) { this.userSettings.setSelectedPatternName( skinName === null ? undefined : `skin:${skinName}`, ); this.selectedSkinName = skinName; this.selectedPattern = null; this.refresh(); this.close(); } private selectPattern(pattern: PlayerPattern | null) { this.selectedColor = null; if (pattern === null) { this.userSettings.setSelectedPatternName(undefined); } else { const name = pattern.colorPalette?.name === undefined ? pattern.name : `${pattern.name}:${pattern.colorPalette.name}`; this.userSettings.setSelectedPatternName(`pattern:${name}`); } this.selectedPattern = pattern; this.selectedSkinName = null; this.refresh(); this.showSkinSelectedPopup(); this.close(); } private showSkinSelectedPopup() { let skinName = translateText("territory_patterns.pattern.default"); if (this.selectedPattern && this.selectedPattern.name) { skinName = this.selectedPattern.name .split("_") .map((w) => w.charAt(0).toUpperCase() + w.slice(1)) .join(" "); if ( this.selectedPattern.colorPalette && this.selectedPattern.colorPalette.name ) { skinName += ` (${this.selectedPattern.colorPalette.name})`; } } window.dispatchEvent( new CustomEvent("show-message", { detail: { message: `${skinName} ${translateText("territory_patterns.selected")}`, duration: 2000, }, }), ); } public async refresh() { this.requestUpdate(); } }