[wip] extractProperty proper typing with union types

This commit is contained in:
2025-09-06 17:30:47 -04:00
parent bedafb0b7c
commit b854fec9e5
3 changed files with 17 additions and 32 deletions

View File

@@ -41,33 +41,23 @@ export default (props: { tableKey: string }) => {
ws.on("open", () => res(ws));
ws.on("error", () => res(ws));
});
const sendWs = (msg: TWsIn) => wsPromise.then((ws) => ws.send(msg));
const wsEvents = fromPromise(wsPromise).flatMap((ws) =>
fromWebsocket<TWsOut>(ws)
);
onCleanup(() => wsPromise.then((ws) => ws.close()));
const presenceEvents = wsEvents.filter(
(evt) => evt.playersPresent !== undefined
);
const gameConfig = extractProperty(wsEvents, "gameConfig").thru(
createObservable
);
const view = extractProperty(wsEvents, "view").thru(createObservable);
const gameEvents = wsEvents.filter((evt) => evt.view !== undefined);
const resultEvents = wsEvents.filter((evt) => evt.results !== undefined);
const players = createObservableWithInit<string[]>(
presenceEvents.map((evt) => evt.playersPresent!),
[]
const players = extractProperty(wsEvents, "playersPresent").thru(
createObservable
);
const playerNames = createObservableStore(
wsEvents
.filter((evt) => evt.playerNames != null)
.map(({ playerNames }) => playerNames!)
.toProperty(),
extractProperty(wsEvents, "playerNames"),
{}
);
@@ -83,9 +73,6 @@ export default (props: { tableKey: string }) => {
createEffect(() => sendWs({ ready: ready() }));
createEffect(() => sendWs({ name: name() }));
const viewProp = gameEvents.map((evt) => evt.view).toProperty();
const view = createObservable(viewProp);
return (
<TableContext.Provider

View File

@@ -48,10 +48,8 @@ export const createObservableStore = <T extends object = {}>(
return store;
};
export const extractProperty = <
P extends string,
T extends { [key in P]?: any }
>(
type UnionKeys<T> = T extends any ? keyof T : never;
export const extractProperty = <T extends object, P extends UnionKeys<T>>(
obs: Observable<T, any>,
property: P
): Property<T[P], any> =>

View File

@@ -13,16 +13,16 @@ import { combine, constant, merge, Observable, pool, Property } from "kefir";
import Bus, { type Bus as TBus } from "kefir-bus";
import { log } from "./logging";
export const WsOut = t.Object({
playersPresent: t.Optional(t.Array(t.String())),
playerNames: t.Optional(t.Record(t.String(), t.String())),
playersReady: t.Optional(t.Nullable(t.Record(t.String(), t.Boolean()))),
gameConfig: t.Optional(
t.Object({ game: t.String(), players: t.Array(t.String()) })
),
view: t.Optional(t.Any()),
results: t.Optional(t.Any()),
});
export const WsOut = t.Union([
t.Object({ playersPresent: t.Array(t.String()) }),
t.Object({ playerNames: t.Record(t.String(), t.String()) }),
t.Object({ playersReady: t.Nullable(t.Record(t.String(), t.Boolean())) }),
t.Object({
gameConfig: t.Object({ game: t.String(), players: t.Array(t.String()) }),
}),
t.Object({ view: t.Any() }),
t.Object({ results: t.Any() }),
]);
export type TWsOut = typeof WsOut.static;
export const WsIn = t.Union([
t.Object({ name: t.String() }),