Profectus-Demo/src/features/milestones/milestone.tsx

208 lines
6.8 KiB
TypeScript
Raw Normal View History

import Select from "@/components/fields/Select.vue";
2022-01-13 22:25:47 -06:00
import {
CoercableComponent,
Component,
findFeatures,
GatherProps,
2022-01-13 22:25:47 -06:00
getUniqueID,
jsx,
2022-01-13 22:25:47 -06:00
Replace,
setDefault,
StyleValue,
Visibility
} from "@/features/feature";
import MilestoneComponent from "@/features/milestones/Milestone.vue";
2022-01-13 22:25:47 -06:00
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";
2022-01-13 22:25:47 -06:00
import {
Computable,
GetComputableType,
GetComputableTypeWithDefault,
processComputable,
ProcessedComputable
} from "@/util/computed";
import { createLazyProxy } from "@/util/proxies";
2022-01-13 22:25:47 -06:00
import { coerceComponent, isCoercableComponent } from "@/util/vue";
import { Unsubscribe } from "nanoevents";
import { computed, Ref, unref } from "vue";
import { useToast } from "vue-toastification";
export const MilestoneType = Symbol("Milestone");
export enum MilestoneDisplay {
All = "all",
//Last = "last",
Configurable = "configurable",
Incomplete = "incomplete",
None = "none"
}
export interface MilestoneOptions {
visibility?: Computable<Visibility>;
shouldEarn: Computable<boolean>;
style?: Computable<StyleValue>;
classes?: Computable<Record<string, boolean>>;
display?: Computable<
| CoercableComponent
| {
requirement: CoercableComponent;
effectDisplay?: CoercableComponent;
optionsDisplay?: CoercableComponent;
}
>;
onComplete?: VoidFunction;
}
interface BaseMilestone extends Persistent<boolean> {
id: string;
earned: Ref<boolean>;
type: typeof MilestoneType;
[Component]: typeof MilestoneComponent;
[GatherProps]: () => Record<string, unknown>;
2022-01-13 22:25:47 -06:00
}
export type Milestone<T extends MilestoneOptions> = Replace<
T & BaseMilestone,
{
visibility: GetComputableTypeWithDefault<T["visibility"], Visibility.Visible>;
shouldEarn: GetComputableType<T["shouldEarn"]>;
style: GetComputableType<T["style"]>;
classes: GetComputableType<T["classes"]>;
display: GetComputableType<T["display"]>;
}
>;
export type GenericMilestone = Replace<
Milestone<MilestoneOptions>,
{
visibility: ProcessedComputable<Visibility>;
}
>;
export function createMilestone<T extends MilestoneOptions>(
optionsFunc: () => T & ThisType<Milestone<T>>
2022-01-13 22:25:47 -06:00
): Milestone<T> {
return createLazyProxy(() => {
const milestone: T & Partial<BaseMilestone> = optionsFunc();
makePersistent<boolean>(milestone, false);
milestone.id = getUniqueID("milestone-");
milestone.type = MilestoneType;
milestone[Component] = MilestoneComponent;
2022-01-13 22:25:47 -06:00
milestone.earned = milestone[PersistentState];
processComputable(milestone as T, "visibility");
setDefault(milestone, "visibility", Visibility.Visible);
const visibility = milestone.visibility as ProcessedComputable<Visibility>;
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<string, unknown>)
)
) {
return Visibility.None;
}
return unref(visibility);
case MilestoneDisplay.Incomplete:
if (unref(milestone.earned)) {
return Visibility.None;
}
return unref(visibility);
case MilestoneDisplay.None:
2022-01-13 22:25:47 -06:00
return Visibility.None;
}
});
2022-01-13 22:25:47 -06:00
processComputable(milestone as T, "shouldEarn");
processComputable(milestone as T, "style");
processComputable(milestone as T, "classes");
processComputable(milestone as T, "display");
2022-01-13 22:25:47 -06:00
milestone[GatherProps] = function (this: GenericMilestone) {
const { visibility, display, style, classes, earned, id } = this;
return { visibility, display, style, classes, earned, id };
};
return milestone as unknown as Milestone<T>;
});
2022-01-13 22:25:47 -06:00
}
const toast = useToast();
2022-01-24 22:23:30 -06:00
const listeners: Record<string, Unsubscribe | undefined> = {};
2022-01-13 22:25:47 -06:00
globalBus.on("addLayer", layer => {
2022-01-24 22:23:30 -06:00
const milestones: GenericMilestone[] = (
findFeatures(layer, MilestoneType) as GenericMilestone[]
).filter(milestone => milestone.shouldEarn != null);
2022-01-13 22:25:47 -06:00
listeners[layer.id] = layer.on("postUpdate", () => {
milestones.forEach(milestone => {
if (
unref(milestone.visibility) === Visibility.Visible &&
!milestone.earned.value &&
unref(milestone.shouldEarn)
) {
2022-01-24 22:23:30 -06:00
milestone[PersistentState].value = true;
2022-01-13 22:25:47 -06:00
milestone.onComplete?.();
if (milestone.display) {
const display = unref(milestone.display);
2022-01-24 22:25:34 -06:00
const Display = coerceComponent(
isCoercableComponent(display) ? display : display.requirement
);
toast(
<>
2022-01-24 22:25:34 -06:00
<h3>Milestone earned!</h3>
2022-01-13 22:25:47 -06:00
<div>
2022-01-24 22:25:34 -06:00
{/* eslint-disable-next-line @typescript-eslint/ban-ts-comment */}
{/* @ts-ignore */}
<Display />
2022-01-13 22:25:47 -06:00
</div>
</>
2022-01-13 22:25:47 -06:00
);
}
}
});
});
});
globalBus.on("removeLayer", layer => {
// unsubscribe from postUpdate
listeners[layer.id]?.();
2022-01-24 22:23:30 -06:00
listeners[layer.id] = undefined;
2022-01-13 22:25:47 -06:00
});
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(() => (
<Select
title="Show Milestones"
options={msDisplayOptions}
onUpdate:modelValue={value => (settings.msDisplay = value as MilestoneDisplay)}
modelValue={settings.msDisplay}
/>
))
);