import Select from "components/fields/Select.vue"; import { CoercableComponent, Component, GatherProps, getUniqueID, jsx, Replace, setDefault, StyleValue, Visibility } from "features/feature"; import MilestoneComponent from "features/milestones/Milestone.vue"; import { globalBus } from "game/events"; import "game/notifications"; import { makePersistent, Persistent, PersistentState } from "game/persistence"; import settings, { registerSettingField } from "game/settings"; import { camelToTitle } from "util/common"; import { Computable, GetComputableType, GetComputableTypeWithDefault, processComputable, ProcessedComputable } from "util/computed"; import { createLazyProxy } from "util/proxies"; import { coerceComponent, isCoercableComponent } from "util/vue"; import { computed, Ref, unref, watchEffect } from "vue"; import { useToast } from "vue-toastification"; const toast = useToast(); export const MilestoneType = Symbol("Milestone"); export enum MilestoneDisplay { All = "all", //Last = "last", Configurable = "configurable", Incomplete = "incomplete", None = "none" } export interface MilestoneOptions { visibility?: Computable; shouldEarn?: () => boolean; style?: Computable; classes?: Computable>; display?: Computable< | CoercableComponent | { requirement: CoercableComponent; effectDisplay?: CoercableComponent; optionsDisplay?: CoercableComponent; } >; onComplete?: VoidFunction; } export interface BaseMilestone extends Persistent { id: string; earned: Ref; complete: VoidFunction; type: typeof MilestoneType; [Component]: typeof MilestoneComponent; [GatherProps]: () => Record; } export type Milestone = Replace< T & BaseMilestone, { visibility: GetComputableTypeWithDefault; style: GetComputableType; classes: GetComputableType; display: GetComputableType; } >; export type GenericMilestone = Replace< Milestone, { visibility: ProcessedComputable; } >; export function createMilestone( optionsFunc: () => T & ThisType> ): Milestone { return createLazyProxy(() => { const milestone: T & Partial = optionsFunc(); makePersistent(milestone, false); milestone.id = getUniqueID("milestone-"); milestone.type = MilestoneType; milestone[Component] = MilestoneComponent; milestone.earned = milestone[PersistentState]; milestone.complete = function () { milestone[PersistentState].value = true; }; processComputable(milestone as T, "visibility"); setDefault(milestone, "visibility", Visibility.Visible); const visibility = milestone.visibility as ProcessedComputable; milestone.visibility = computed(() => { const display = unref((milestone as GenericMilestone).display); switch (settings.msDisplay) { default: case MilestoneDisplay.All: return unref(visibility); case MilestoneDisplay.Configurable: if ( unref(milestone.earned) && !( display != null && typeof display == "object" && "optionsDisplay" in (display as Record) ) ) { return Visibility.None; } return unref(visibility); case MilestoneDisplay.Incomplete: if (unref(milestone.earned)) { return Visibility.None; } return unref(visibility); case MilestoneDisplay.None: return Visibility.None; } }); processComputable(milestone as T, "style"); processComputable(milestone as T, "classes"); processComputable(milestone as T, "display"); milestone[GatherProps] = function (this: GenericMilestone) { const { visibility, display, style, classes, earned, id } = this; return { visibility, display, style: unref(style), classes, earned, id }; }; if (milestone.shouldEarn) { const genericMilestone = milestone as GenericMilestone; watchEffect(() => { if ( !genericMilestone.earned.value && unref(genericMilestone.visibility) === Visibility.Visible && genericMilestone.shouldEarn?.() ) { genericMilestone.earned.value = true; genericMilestone.onComplete?.(); if (genericMilestone.display) { const display = unref(genericMilestone.display); const Display = coerceComponent( isCoercableComponent(display) ? display : display.requirement ); toast( <>

Milestone earned!

{/* eslint-disable-next-line @typescript-eslint/ban-ts-comment */} {/* @ts-ignore */}
); } } }); } return milestone as unknown as Milestone; }); } declare module "game/settings" { interface Settings { msDisplay: MilestoneDisplay; } } globalBus.on("loadSettings", settings => { setDefault(settings, "msDisplay", MilestoneDisplay.All); }); const msDisplayOptions = Object.values(MilestoneDisplay).map(option => ({ label: camelToTitle(option), value: option })); registerSettingField( jsx(() => (