Compare commits

...

4 Commits

9 changed files with 63 additions and 4 deletions

View File

@ -97,3 +97,21 @@ To regenerate the frontend client during development:
```sh
npm run openapi-generate
```
## Screenshots
### Home
![Homepage](./docs/homepage.png)
### Schedule
![Schedule](./docs/schedule-page.png)
### Team
![Team](./docs/teampage.png)
### Team Settings
![Team Settings](./docs/settings-page.png)

View File

@ -78,6 +78,7 @@ nav {
margin: 0;
align-items: center;
justify-content: space-between;
border-bottom: 1px solid var(--surface-0);
}
nav .nav-links {

View File

@ -8,6 +8,39 @@ export const useClientStore = defineStore("client", () => {
const calls = new Map<string, Promise<any>>();
type CachedValue<T> = {
value: T,
timestamp: number,
};
const cached = new Map<string, any>();
function callCached<T>(
key: string,
apiCall: () => CancelablePromise<T>,
maxLifetime: number = 2000,
thenOnce?: (result: T) => T,
catchOnce?: (error: any) => any,
finallyOnce?: () => void,
): Promise<T> {
if (cached.has(key)) {
const cachedValue = cached.get(key) as CachedValue<T>;
if (Date.now() - cachedValue.timestamp < maxLifetime) {
return Promise.resolve(cachedValue.value);
}
}
// cache miss
let promise = call(key, apiCall, thenOnce, catchOnce, finallyOnce);
promise.then((result) => {
cached.set(key, {
value: result,
timestamp: Date.now(),
});
});
return promise;
}
function call<T>(
key: string,
apiCall: () => CancelablePromise<T>,
@ -17,11 +50,15 @@ export const useClientStore = defineStore("client", () => {
): Promise<T> {
console.log("Fetching call " + key);
if (!calls.has(key)) {
console.log("Making new call " + key);
const promise = apiCall();
calls.set(key, promise);
// remove from calls once completed
promise.finally(() => calls.delete(key));
promise.finally(() => {
console.log("Call " + key + " completed");
calls.delete(key);
});
// only execute this "then" once if the call was just freshly made
if (thenOnce) {
@ -37,6 +74,8 @@ export const useClientStore = defineStore("client", () => {
}
return promise;
} else {
console.log("Returning concurrent call " + key);
}
return calls.get(key) as Promise<T>;
}
@ -44,6 +83,7 @@ export const useClientStore = defineStore("client", () => {
return {
client,
call,
callCached,
calls,
}
});

View File

@ -33,7 +33,7 @@ export const useTeamsStore = defineStore("teams", () => {
}
async function fetchTeam(id: number) {
const response = await clientStore.call(
const response = await clientStore.callCached(
fetchTeam.name,
() => client.default.getTeam(id)
);
@ -42,7 +42,7 @@ export const useTeamsStore = defineStore("teams", () => {
}
async function fetchTeamMembers(id: number) {
const response = await clientStore.call(
const response = await clientStore.callCached(
fetchTeamMembers.name,
() => client.default.getTeamMembers(id)
);

View File

@ -117,7 +117,7 @@ class Event(app_db.BaseModel):
f"Maximum roles filled: {matchings}" + ringers_needed_msg,
"",
"[Confirm attendance here]" +
f"(https://{domain}/team/id/{self.team.id}/events/{self.id})",
f"(https://{domain}/team/id/{self.team.id})",
])
def get_or_create_webhook(self):

BIN
docs/homepage.png 100644

Binary file not shown.

After

Width:  |  Height:  |  Size: 97 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 107 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 115 KiB

BIN
docs/teampage.png 100644

Binary file not shown.

After

Width:  |  Height:  |  Size: 221 KiB