2021-12-13 04:01:30 +00:00
|
|
|
import * as electronRemote from '@electron/remote';
|
|
|
|
const electronConsole = electronRemote.getGlobal('console') as Console;
|
|
|
|
import Logger from '../../../../logger/logger';
|
|
|
|
const LOG = Logger.create(__filename, electronConsole);
|
|
|
|
|
2021-12-24 23:37:27 +00:00
|
|
|
import { Changes, GuildMetadata, Member, Message, Resource } from "../../data-types";
|
2021-12-13 04:01:30 +00:00
|
|
|
import CombinedGuild from "../../guild-combined";
|
|
|
|
|
2022-01-14 04:34:38 +00:00
|
|
|
import { Dispatch, SetStateAction, useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
2021-12-13 04:01:30 +00:00
|
|
|
import { AutoVerifierChangesType } from "../../auto-verifier";
|
|
|
|
import { Conflictable, Connectable } from "../../guild-types";
|
|
|
|
import { EventEmitter } from 'tsee';
|
2021-12-24 23:37:27 +00:00
|
|
|
import { IDQuery, PartialMessageListQuery } from '../../auto-verifier-with-args';
|
2021-12-17 01:25:55 +00:00
|
|
|
import { Token, Channel } from '../../data-types';
|
2021-12-31 03:28:30 +00:00
|
|
|
import { useIsMountedRef, useOneTimeAsyncAction } from './react-helper';
|
2021-12-24 23:37:27 +00:00
|
|
|
import Globals from '../../globals';
|
2021-12-25 18:00:20 +00:00
|
|
|
import ElementsUtil from './elements-util';
|
2021-12-13 04:01:30 +00:00
|
|
|
|
2021-12-13 05:25:23 +00:00
|
|
|
export type SingleSubscriptionEvents = {
|
2021-12-13 04:01:30 +00:00
|
|
|
'fetch': () => void;
|
2021-12-13 05:25:23 +00:00
|
|
|
'updated': () => void;
|
2021-12-13 04:01:30 +00:00
|
|
|
'conflict': () => void;
|
|
|
|
'fetch-error': () => void;
|
|
|
|
}
|
|
|
|
|
2021-12-13 05:25:23 +00:00
|
|
|
export type MultipleSubscriptionEvents<T> = {
|
|
|
|
'fetch': () => void;
|
|
|
|
'fetch-error': () => void;
|
|
|
|
'new': (newValues: T[]) => void;
|
|
|
|
'updated': (updatedValues: T[]) => void;
|
|
|
|
'removed': (removedValues: T[]) => void;
|
|
|
|
'conflict': (changes: Changes<T>) => void;
|
|
|
|
}
|
|
|
|
|
2021-12-13 04:01:30 +00:00
|
|
|
interface EffectParams<T> {
|
|
|
|
guild: CombinedGuild;
|
2022-01-14 05:17:15 +00:00
|
|
|
onFetch: (value: T | null, valueGuild: CombinedGuild) => void;
|
2021-12-13 04:01:30 +00:00
|
|
|
onFetchError: (e: unknown) => void;
|
2021-12-13 04:34:29 +00:00
|
|
|
bindEventsFunc: () => void;
|
|
|
|
unbindEventsFunc: () => void;
|
2021-12-13 04:01:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type Arguments<T> = T extends (...args: infer A) => unknown ? A : never;
|
|
|
|
|
2021-12-13 05:25:23 +00:00
|
|
|
interface SingleEventMappingParams<T, UE extends keyof Connectable, CE extends keyof Conflictable> {
|
|
|
|
updatedEventName: UE;
|
|
|
|
updatedEventArgsMap: (...args: Arguments<Connectable[UE]>) => T;
|
2021-12-13 04:01:30 +00:00
|
|
|
conflictEventName: CE;
|
2021-12-13 05:25:23 +00:00
|
|
|
conflictEventArgsMap: (...args: Arguments<Conflictable[CE]>) => T;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface MultipleEventMappingParams<
|
|
|
|
T,
|
|
|
|
NE extends keyof Connectable,
|
|
|
|
UE extends keyof Connectable,
|
|
|
|
RE extends keyof Connectable,
|
|
|
|
CE extends keyof Conflictable
|
|
|
|
> {
|
|
|
|
newEventName: NE;
|
|
|
|
newEventArgsMap: (...args: Arguments<Connectable[NE]>) => T[]; // list of new elements
|
|
|
|
updatedEventName: UE;
|
|
|
|
updatedEventArgsMap: (...args: Arguments<Connectable[UE]>) => T[]; // list of updated elements
|
|
|
|
removedEventName: RE;
|
|
|
|
removedEventArgsMap: (...args: Arguments<Connectable[RE]>) => T[]; // list of removed elements
|
|
|
|
|
|
|
|
conflictEventName: CE;
|
|
|
|
conflictEventArgsMap: (...args: Arguments<Conflictable[CE]>) => Changes<T>;
|
|
|
|
|
2021-12-24 23:37:27 +00:00
|
|
|
sortFunc: (a: T, b: T) => number; // Friendly reminder that v8 uses timsort so this is O(n) for pre-sorted stuff
|
2021-12-31 03:33:15 +00:00
|
|
|
}
|
2021-12-13 04:01:30 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
function useGuildSubscriptionEffect<T>(
|
|
|
|
subscriptionParams: EffectParams<T>,
|
|
|
|
fetchFunc: (() => Promise<T>) | (() => Promise<T | null>)
|
|
|
|
): [ fetchRetryCallable: () => Promise<void> ] {
|
|
|
|
const { guild, onFetch, onFetchError, bindEventsFunc, unbindEventsFunc } = subscriptionParams;
|
2021-12-24 23:37:27 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
const isMounted = useIsMountedRef();
|
2022-01-14 04:34:38 +00:00
|
|
|
const guildRef = useRef<CombinedGuild>(guild);
|
2022-01-14 05:17:15 +00:00
|
|
|
guildRef.current = guild;
|
2021-12-13 04:01:30 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
const fetchManagerFunc = useCallback(async () => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 04:34:38 +00:00
|
|
|
if (guildRef.current !== guild) return;
|
2021-12-31 03:33:15 +00:00
|
|
|
try {
|
|
|
|
const value = await fetchFunc();
|
2021-12-24 23:37:27 +00:00
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return; // Don't call onFetch if we changed guilds. TODO: Test this
|
|
|
|
onFetch(value, guild);
|
2021-12-31 03:33:15 +00:00
|
|
|
} catch (e: unknown) {
|
|
|
|
LOG.error('error fetching for subscription', e);
|
2021-12-24 23:37:27 +00:00
|
|
|
if (!isMounted.current) return;
|
2022-01-14 04:34:38 +00:00
|
|
|
if (guildRef.current !== guild) return;
|
2021-12-31 03:33:15 +00:00
|
|
|
onFetchError(e);
|
|
|
|
}
|
|
|
|
}, [ fetchFunc ]);
|
2021-12-13 05:25:23 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
useEffect(() => {
|
|
|
|
// Bind guild events to make sure we have the most up to date information
|
|
|
|
guild.on('connect', fetchManagerFunc);
|
2022-01-14 04:34:38 +00:00
|
|
|
guild.on('disconnect', fetchManagerFunc);
|
2021-12-31 03:33:15 +00:00
|
|
|
bindEventsFunc();
|
2021-12-24 23:37:27 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
// Fetch the data once
|
|
|
|
fetchManagerFunc();
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
// Unbind the events so that we don't have any memory leaks
|
|
|
|
guild.off('connect', fetchManagerFunc);
|
2022-01-14 04:34:38 +00:00
|
|
|
guild.off('disconnect', fetchManagerFunc);
|
2021-12-31 03:33:15 +00:00
|
|
|
unbindEventsFunc();
|
|
|
|
}
|
|
|
|
}, [ fetchManagerFunc ]);
|
2021-12-24 23:37:27 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
return [ fetchManagerFunc ];
|
|
|
|
}
|
|
|
|
|
|
|
|
function useSingleGuildSubscription<T, UE extends keyof Connectable, CE extends keyof Conflictable>(
|
|
|
|
guild: CombinedGuild,
|
|
|
|
eventMappingParams: SingleEventMappingParams<T, UE, CE>,
|
|
|
|
fetchFunc: (() => Promise<T>) | (() => Promise<T | null>)
|
2022-01-14 05:17:15 +00:00
|
|
|
): [value: T | null, valueGuild: CombinedGuild | null, fetchError: unknown | null, events: EventEmitter<SingleSubscriptionEvents>] {
|
2021-12-31 03:33:15 +00:00
|
|
|
const { updatedEventName, updatedEventArgsMap, conflictEventName, conflictEventArgsMap } = eventMappingParams;
|
|
|
|
|
|
|
|
const isMounted = useIsMountedRef();
|
2022-01-14 05:17:15 +00:00
|
|
|
const guildRef = useRef<CombinedGuild>(guild);
|
|
|
|
guildRef.current = guild;
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const [ fetchError, setFetchError ] = useState<unknown | null>(null);
|
|
|
|
const [ value, setValue ] = useState<T | null>(null);
|
2022-01-14 05:17:15 +00:00
|
|
|
const [ valueGuild, setValueGuild ] = useState<CombinedGuild | null>(null);
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const events = useMemo(() => new EventEmitter<SingleSubscriptionEvents>(), []);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onFetch = useCallback((fetchValue: T | null, fetchValueGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(fetchValue);
|
2022-01-14 05:17:15 +00:00
|
|
|
setValueGuild(fetchValueGuild);
|
2021-12-31 03:33:15 +00:00
|
|
|
setFetchError(null);
|
|
|
|
events.emit('fetch');
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
const onFetchError = useCallback((e: unknown) => {
|
|
|
|
setFetchError(e);
|
|
|
|
setValue(null);
|
2022-01-14 05:17:15 +00:00
|
|
|
setValueGuild(null);
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('fetch-error');
|
|
|
|
}, []);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onUpdated = useCallback((updateValue: T, updateValueGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(updateValue);
|
2022-01-14 05:17:15 +00:00
|
|
|
if (updateValueGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`update guild (${updateValueGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('updated');
|
|
|
|
}, []);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onConflict = useCallback((conflictValue: T, conflictValueGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(conflictValue);
|
2022-01-14 05:17:15 +00:00
|
|
|
if (conflictValueGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`conflict guild (${conflictValueGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('conflict');
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
// I think the typed EventEmitter class isn't ready for this level of insane type safety
|
|
|
|
// otherwise, I may have done this wrong. Forcing it to work with these calls
|
|
|
|
const boundUpdateFunc = useCallback((...args: Arguments<Connectable[UE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return;
|
2021-12-31 03:33:15 +00:00
|
|
|
const value = updatedEventArgsMap(...args);
|
2022-01-14 05:17:15 +00:00
|
|
|
onUpdated(value, guild);
|
|
|
|
}, [ guild ]) as (Connectable & Conflictable)[UE];
|
2021-12-31 03:33:15 +00:00
|
|
|
const boundConflictFunc = useCallback((...args: Arguments<Conflictable[CE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return;
|
2021-12-31 03:33:15 +00:00
|
|
|
const value = conflictEventArgsMap(...args);
|
2022-01-14 05:17:15 +00:00
|
|
|
onConflict(value, guild);
|
|
|
|
}, [ guild ]) as (Connectable & Conflictable)[CE];
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const bindEventsFunc = useCallback(() => {
|
|
|
|
guild.on(updatedEventName, boundUpdateFunc);
|
|
|
|
guild.on(conflictEventName, boundConflictFunc);
|
2022-01-14 05:17:15 +00:00
|
|
|
}, [ boundUpdateFunc, boundConflictFunc ]);
|
2021-12-31 03:33:15 +00:00
|
|
|
const unbindEventsFunc = useCallback(() => {
|
|
|
|
guild.off(updatedEventName, boundUpdateFunc);
|
|
|
|
guild.off(conflictEventName, boundConflictFunc);
|
2022-01-14 05:17:15 +00:00
|
|
|
}, [ boundUpdateFunc, boundConflictFunc ]);
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
useGuildSubscriptionEffect({
|
|
|
|
guild,
|
|
|
|
onFetch,
|
|
|
|
onFetchError,
|
|
|
|
bindEventsFunc,
|
|
|
|
unbindEventsFunc
|
|
|
|
}, fetchFunc);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
return [ value, valueGuild, fetchError, events ];
|
2021-12-31 03:33:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function useMultipleGuildSubscription<
|
|
|
|
T extends { id: string },
|
|
|
|
NE extends keyof Connectable,
|
|
|
|
UE extends keyof Connectable,
|
|
|
|
RE extends keyof Connectable,
|
|
|
|
CE extends keyof Conflictable
|
|
|
|
>(
|
|
|
|
guild: CombinedGuild,
|
|
|
|
eventMappingParams: MultipleEventMappingParams<T, NE, UE, RE, CE>,
|
|
|
|
fetchFunc: (() => Promise<T[]>) | (() => Promise<T[] | null>)
|
|
|
|
): [
|
|
|
|
fetchRetryCallable: () => Promise<void>,
|
|
|
|
value: T[] | null,
|
2022-01-14 05:17:15 +00:00
|
|
|
valueGuild: CombinedGuild | null,
|
2021-12-31 03:33:15 +00:00
|
|
|
fetchError: unknown | null,
|
|
|
|
events: EventEmitter<MultipleSubscriptionEvents<T>>
|
|
|
|
] {
|
|
|
|
const {
|
|
|
|
newEventName, newEventArgsMap,
|
|
|
|
updatedEventName, updatedEventArgsMap,
|
|
|
|
removedEventName, removedEventArgsMap,
|
|
|
|
conflictEventName, conflictEventArgsMap,
|
|
|
|
sortFunc
|
|
|
|
} = eventMappingParams;
|
|
|
|
|
|
|
|
const isMounted = useIsMountedRef();
|
2022-01-14 05:17:15 +00:00
|
|
|
const guildRef = useRef<CombinedGuild>(guild);
|
|
|
|
guildRef.current = guild;
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const [ fetchError, setFetchError ] = useState<unknown | null>(null);
|
|
|
|
const [ value, setValue ] = useState<T[] | null>(null);
|
2022-01-14 05:17:15 +00:00
|
|
|
const [ valueGuild, setValueGuild ] = useState<CombinedGuild | null>(null);
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const events = useMemo(() => new EventEmitter<MultipleSubscriptionEvents<T>>(), []);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onFetch = useCallback((fetchValue: T[] | null, fetchValueGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
if (fetchValue) fetchValue.sort(sortFunc);
|
|
|
|
setValue(fetchValue);
|
2022-01-14 05:17:15 +00:00
|
|
|
setValueGuild(fetchValueGuild);
|
2021-12-31 03:33:15 +00:00
|
|
|
setFetchError(null);
|
|
|
|
events.emit('fetch');
|
|
|
|
}, [ sortFunc ]);
|
|
|
|
|
|
|
|
const onFetchError = useCallback((e: unknown) => {
|
|
|
|
setFetchError(e);
|
|
|
|
setValue(null);
|
|
|
|
events.emit('fetch-error');
|
|
|
|
}, []);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onNew = useCallback((newElements: T[], newElementsGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(currentValue => {
|
2022-01-14 05:17:15 +00:00
|
|
|
if (currentValue === null) return Array.from(newElements).sort(sortFunc);
|
2021-12-31 03:33:15 +00:00
|
|
|
return currentValue.concat(newElements).sort(sortFunc);
|
2022-01-14 05:17:15 +00:00
|
|
|
});
|
|
|
|
if (newElementsGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`new elements guild (${newElementsGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('new', newElements);
|
|
|
|
}, [ sortFunc ]);
|
2022-01-14 05:17:15 +00:00
|
|
|
const onUpdated = useCallback((updatedElements: T[], updatedElementsGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(currentValue => {
|
|
|
|
if (currentValue === null) return null;
|
|
|
|
return currentValue.map(element => updatedElements.find(updatedElement => updatedElement.id === element.id) ?? element).sort(sortFunc);
|
|
|
|
});
|
2022-01-14 05:17:15 +00:00
|
|
|
if (updatedElementsGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`updated elements guild (${updatedElementsGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('updated', updatedElements);
|
|
|
|
}, [ sortFunc ]);
|
2022-01-14 05:17:15 +00:00
|
|
|
const onRemoved = useCallback((removedElements: T[], removedElementsGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(currentValue => {
|
|
|
|
if (currentValue === null) return null;
|
|
|
|
const deletedIds = new Set(removedElements.map(deletedElement => deletedElement.id));
|
|
|
|
return currentValue.filter(element => !deletedIds.has(element.id)).sort(sortFunc);
|
|
|
|
});
|
2022-01-14 05:17:15 +00:00
|
|
|
if (removedElementsGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`removed elements guild (${removedElementsGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('removed', removedElements);
|
|
|
|
}, [ sortFunc ]);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onConflict = useCallback((changes: Changes<T>, changesGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(currentValue => {
|
|
|
|
if (currentValue === null) return null;
|
|
|
|
const deletedIds = new Set(changes.deleted.map(deletedElement => deletedElement.id));
|
|
|
|
return currentValue
|
|
|
|
.concat(changes.added)
|
|
|
|
.map(element => changes.updated.find(change => change.newDataPoint.id === element.id)?.newDataPoint ?? element)
|
|
|
|
.filter(element => !deletedIds.has(element.id))
|
|
|
|
.sort(sortFunc);
|
|
|
|
});
|
2022-01-14 05:17:15 +00:00
|
|
|
if (changesGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`conflict changes guild (${changesGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
|
|
|
setValueGuild(changesGuild);
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('conflict', changes);
|
|
|
|
}, [ sortFunc ]);
|
|
|
|
|
|
|
|
// I think the typed EventEmitter class isn't ready for this level of insane type safety
|
|
|
|
// otherwise, I may have done this wrong. Forcing it to work with these calls
|
|
|
|
const boundNewFunc = useCallback((...args: Arguments<Connectable[NE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return; // prevent changes from a different guild
|
|
|
|
onNew(newEventArgsMap(...args), guild);
|
|
|
|
}, [ guild, onNew, newEventArgsMap ]) as (Connectable & Conflictable)[NE];
|
2021-12-31 03:33:15 +00:00
|
|
|
const boundUpdateFunc = useCallback((...args: Arguments<Connectable[UE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return;
|
|
|
|
onUpdated(updatedEventArgsMap(...args), guild);
|
|
|
|
}, [ guild, onUpdated, updatedEventArgsMap ]) as (Connectable & Conflictable)[UE];
|
2021-12-31 03:33:15 +00:00
|
|
|
const boundRemovedFunc = useCallback((...args: Arguments<Connectable[RE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return;
|
|
|
|
onRemoved(removedEventArgsMap(...args), guild);
|
|
|
|
}, [ guild, onRemoved, removedEventArgsMap ]) as (Connectable & Conflictable)[RE];
|
2021-12-31 03:33:15 +00:00
|
|
|
const boundConflictFunc = useCallback((...args: Arguments<Conflictable[CE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return;
|
|
|
|
onConflict(conflictEventArgsMap(...args), guild);
|
|
|
|
}, [ guild, onConflict, conflictEventArgsMap ]) as (Connectable & Conflictable)[CE];
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const bindEventsFunc = useCallback(() => {
|
|
|
|
guild.on(newEventName, boundNewFunc);
|
|
|
|
guild.on(updatedEventName, boundUpdateFunc);
|
|
|
|
guild.on(removedEventName, boundRemovedFunc);
|
|
|
|
guild.on(conflictEventName, boundConflictFunc);
|
|
|
|
}, [ boundNewFunc, boundUpdateFunc, boundRemovedFunc, boundConflictFunc ]);
|
|
|
|
const unbindEventsFunc = useCallback(() => {
|
|
|
|
guild.off(newEventName, boundNewFunc);
|
|
|
|
guild.off(updatedEventName, boundUpdateFunc);
|
|
|
|
guild.off(removedEventName, boundRemovedFunc);
|
|
|
|
guild.off(conflictEventName, boundConflictFunc);
|
|
|
|
}, [ boundNewFunc, boundUpdateFunc, boundRemovedFunc, boundConflictFunc ]);
|
|
|
|
|
|
|
|
const [ fetchRetryCallable ] = useGuildSubscriptionEffect({
|
|
|
|
guild,
|
|
|
|
onFetch,
|
|
|
|
onFetchError,
|
|
|
|
bindEventsFunc,
|
|
|
|
unbindEventsFunc
|
|
|
|
}, fetchFunc);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
return [ fetchRetryCallable, value, valueGuild, fetchError, events ];
|
2021-12-31 03:33:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function useMultipleGuildSubscriptionScrolling<
|
|
|
|
T extends { id: string },
|
|
|
|
NE extends keyof Connectable,
|
|
|
|
UE extends keyof Connectable,
|
|
|
|
RE extends keyof Connectable,
|
|
|
|
CE extends keyof Conflictable
|
|
|
|
>(
|
|
|
|
guild: CombinedGuild,
|
|
|
|
eventMappingParams: MultipleEventMappingParams<T, NE, UE, RE, CE>,
|
|
|
|
maxElements: number,
|
|
|
|
maxFetchElements: number,
|
|
|
|
fetchFunc: (() => Promise<T[]>) | (() => Promise<T[] | null>),
|
|
|
|
fetchAboveFunc: ((reference: T) => Promise<T[] | null>),
|
|
|
|
fetchBelowFunc: ((reference: T) => Promise<T[] | null>),
|
|
|
|
): [
|
|
|
|
fetchRetryCallable: () => Promise<void>,
|
|
|
|
fetchAboveCallable: () => Promise<{ hasMoreAbove: boolean, removedFromBottom: boolean }>,
|
|
|
|
fetchBelowCallable: () => Promise<{ hasMoreBelow: boolean, removedFromTop: boolean }>,
|
|
|
|
setScrollRatio: Dispatch<SetStateAction<number>>,
|
|
|
|
fetchResult: { hasMoreAbove: boolean, hasMoreBelow: boolean } | null,
|
|
|
|
value: T[] | null,
|
2022-01-14 05:17:15 +00:00
|
|
|
valueGuild: CombinedGuild | null,
|
2021-12-31 03:33:15 +00:00
|
|
|
fetchError: unknown | null,
|
|
|
|
fetchAboveError: unknown | null,
|
|
|
|
fetchBelowError: unknown | null,
|
|
|
|
events: EventEmitter<MultipleSubscriptionEvents<T>>
|
|
|
|
] {
|
|
|
|
const {
|
|
|
|
newEventName, newEventArgsMap,
|
|
|
|
updatedEventName, updatedEventArgsMap,
|
|
|
|
removedEventName, removedEventArgsMap,
|
|
|
|
conflictEventName, conflictEventArgsMap,
|
|
|
|
sortFunc
|
|
|
|
} = eventMappingParams;
|
|
|
|
|
|
|
|
const isMounted = useIsMountedRef();
|
2022-01-14 05:17:15 +00:00
|
|
|
const guildRef = useRef<CombinedGuild>(guild);
|
|
|
|
guildRef.current = guild;
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const [ value, setValue ] = useState<T[] | null>(null);
|
2022-01-14 05:17:15 +00:00
|
|
|
const [ valueGuild, setValueGuild ] = useState<CombinedGuild | null>(null);
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const [ fetchError, setFetchError ] = useState<unknown | null>(null);
|
|
|
|
const [ fetchAboveError, setFetchAboveError ] = useState<unknown | null>(null);
|
|
|
|
const [ fetchBelowError, setFetchBelowError ] = useState<unknown | null>(null);
|
|
|
|
|
|
|
|
const [ fetchResult, setFetchResult ] = useState<{ hasMoreAbove: boolean, hasMoreBelow: boolean } | null>(null);
|
|
|
|
|
|
|
|
// Percentage of scroll from top. i.e. 300px from top of 1000px scroll = 0.3 scroll ratio
|
|
|
|
const [ scrollRatio, setScrollRatio ] = useState<number>(0.5);
|
|
|
|
|
|
|
|
// Gets the number of elements to remove from the top or bottom. Tries to optimise so that if 10 elements
|
|
|
|
// are removed and we're 30% scrolled down, 3 get removed from the top and 7 are removed from the bottom.
|
|
|
|
// TBH, this function is pretty overkill but that's important
|
|
|
|
const getRemoveCounts = useCallback((toRemove: number): { fromTop: number, fromBottom: number } => {
|
|
|
|
// Make sure we round toward the side of the scrollbar that we are not closest to
|
|
|
|
// this is important for the most common case of 1 removed element.
|
|
|
|
const topRoundFunc = scrollRatio > 0.5 ? Math.ceil : Math.floor;
|
|
|
|
const bottomRoundFunc = scrollRatio > 0.5 ? Math.floor : Math.ceil;
|
|
|
|
return {
|
|
|
|
fromTop: topRoundFunc(toRemove * scrollRatio),
|
|
|
|
fromBottom: bottomRoundFunc(toRemove * scrollRatio)
|
2021-12-24 23:37:27 +00:00
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
}, [ scrollRatio ]);
|
|
|
|
|
|
|
|
function removeByCounts<T>(elements: T[], counts: { fromTop: number, fromBottom: number }): T[] {
|
|
|
|
const { fromTop, fromBottom } = counts;
|
|
|
|
return elements.slice(fromTop, elements.length - fromBottom);
|
|
|
|
}
|
2021-12-24 23:37:27 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
const events = useMemo(() => new EventEmitter<MultipleSubscriptionEvents<T>>(), []);
|
2021-12-24 23:37:27 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
const fetchAboveCallable = useCallback(async (): Promise<{ hasMoreAbove: boolean, removedFromBottom: boolean }> => {
|
|
|
|
if (!isMounted.current) return { hasMoreAbove: false, removedFromBottom: false };
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return { hasMoreAbove: false, removedFromBottom: false };
|
2021-12-31 03:33:15 +00:00
|
|
|
if (!value || value.length === 0) return { hasMoreAbove: false, removedFromBottom: false };
|
|
|
|
try {
|
|
|
|
const reference = value[0] as T;
|
|
|
|
const aboveElements = await fetchAboveFunc(reference);
|
2021-12-24 23:37:27 +00:00
|
|
|
if (!isMounted.current) return { hasMoreAbove: false, removedFromBottom: false };
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return { hasMoreAbove: false, removedFromBottom: false };
|
2021-12-31 03:33:15 +00:00
|
|
|
setFetchAboveError(null);
|
|
|
|
if (aboveElements) {
|
|
|
|
const hasMoreAbove = aboveElements.length >= maxFetchElements;
|
|
|
|
let removedFromBottom = false;
|
|
|
|
setValue(currentValue => {
|
|
|
|
let newValue = aboveElements.concat(currentValue ?? []).sort(sortFunc);
|
|
|
|
if (newValue.length > maxElements) {
|
|
|
|
newValue = newValue.slice(0, maxElements);
|
|
|
|
removedFromBottom = true;
|
|
|
|
}
|
|
|
|
return newValue;
|
|
|
|
});
|
|
|
|
return { hasMoreAbove, removedFromBottom };
|
|
|
|
} else {
|
|
|
|
return { hasMoreAbove: false, removedFromBottom: false };
|
2021-12-24 23:37:27 +00:00
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
} catch (e: unknown) {
|
|
|
|
LOG.error('error fetching above for subscription', e);
|
|
|
|
if (!isMounted.current) return { hasMoreAbove: false, removedFromBottom: false };
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return { hasMoreAbove: false, removedFromBottom: false };
|
2021-12-31 03:33:15 +00:00
|
|
|
setFetchAboveError(e);
|
|
|
|
return { hasMoreAbove: true, removedFromBottom: false };
|
|
|
|
}
|
2022-01-14 05:17:15 +00:00
|
|
|
}, [ guild, value, fetchAboveFunc, maxFetchElements ]);
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
const fetchBelowCallable = useCallback(async (): Promise<{ hasMoreBelow: boolean, removedFromTop: boolean }> => {
|
|
|
|
if (!isMounted.current) return { hasMoreBelow: false, removedFromTop: false };
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return { hasMoreBelow: false, removedFromTop: false };
|
2021-12-31 03:33:15 +00:00
|
|
|
if (!value || value.length === 0) return { hasMoreBelow: false, removedFromTop: false };
|
|
|
|
try {
|
|
|
|
const reference = value[value.length - 1] as T;
|
|
|
|
const belowElements = await fetchBelowFunc(reference);
|
2021-12-24 23:37:27 +00:00
|
|
|
if (!isMounted.current) return { hasMoreBelow: false, removedFromTop: false };
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return { hasMoreBelow: false, removedFromTop: false };
|
2021-12-31 03:33:15 +00:00
|
|
|
setFetchBelowError(null);
|
|
|
|
if (belowElements) {
|
|
|
|
const hasMoreBelow = belowElements.length >= maxFetchElements;
|
|
|
|
let removedFromTop = false;
|
|
|
|
setValue(currentValue => {
|
|
|
|
let newValue = (currentValue ?? []).concat(belowElements).sort(sortFunc);
|
|
|
|
if (newValue.length > maxElements) {
|
|
|
|
newValue = newValue.slice(Math.max(newValue.length - maxElements, 0));
|
|
|
|
removedFromTop = true;
|
|
|
|
}
|
|
|
|
return newValue;
|
|
|
|
});
|
|
|
|
return { hasMoreBelow, removedFromTop };
|
|
|
|
} else {
|
|
|
|
return { hasMoreBelow: false, removedFromTop: false };
|
2021-12-24 23:37:27 +00:00
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
} catch (e: unknown) {
|
|
|
|
LOG.error('error fetching below for subscription', e);
|
|
|
|
if (!isMounted.current) return { hasMoreBelow: false, removedFromTop: false };
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return { hasMoreBelow: false, removedFromTop: false };
|
2021-12-31 03:33:15 +00:00
|
|
|
setFetchBelowError(e);
|
|
|
|
return { hasMoreBelow: true, removedFromTop: false };
|
|
|
|
}
|
|
|
|
}, [ value, fetchBelowFunc, maxFetchElements ]);
|
2021-12-24 23:37:27 +00:00
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onFetch = useCallback((fetchValue: T[] | null, fetchValueGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
let hasMoreAbove = false;
|
|
|
|
if (fetchValue) {
|
|
|
|
if (fetchValue.length >= maxFetchElements) hasMoreAbove = true;
|
|
|
|
fetchValue = fetchValue.slice(Math.max(fetchValue.length - maxElements)).sort(sortFunc);
|
|
|
|
}
|
|
|
|
setFetchResult({ hasMoreAbove, hasMoreBelow: false });
|
|
|
|
setValue(fetchValue);
|
2022-01-14 05:17:15 +00:00
|
|
|
setValueGuild(fetchValueGuild);
|
2021-12-31 03:33:15 +00:00
|
|
|
setFetchError(null);
|
|
|
|
events.emit('fetch');
|
|
|
|
}, [ sortFunc, maxFetchElements, maxElements ]);
|
|
|
|
|
|
|
|
const onFetchError = useCallback((e: unknown) => {
|
|
|
|
setFetchError(e);
|
|
|
|
setValue(null);
|
2022-01-14 05:17:15 +00:00
|
|
|
setValueGuild(null);
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('fetch-error');
|
|
|
|
}, []);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onNew = useCallback((newElements: T[], newElementsGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(currentValue => {
|
|
|
|
if (currentValue === null) return null;
|
|
|
|
let newValue = currentValue.concat(newElements).sort(sortFunc);
|
|
|
|
if (newValue.length > maxElements) {
|
|
|
|
newValue = removeByCounts(newValue, getRemoveCounts(newValue.length - maxElements));
|
2021-12-24 23:37:27 +00:00
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
return newValue;
|
2022-01-14 05:17:15 +00:00
|
|
|
});
|
|
|
|
if (newElementsGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`new elements guild (${newElementsGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('new', newElements);
|
|
|
|
}, [ sortFunc, getRemoveCounts ]);
|
2022-01-14 05:17:15 +00:00
|
|
|
const onUpdated = useCallback((updatedElements: T[], updatedElementsGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(currentValue => {
|
|
|
|
if (currentValue === null) return null;
|
|
|
|
return currentValue.map(element => updatedElements.find(updatedElement => updatedElement.id === element.id) ?? element).sort(sortFunc);
|
|
|
|
});
|
2022-01-14 05:17:15 +00:00
|
|
|
if (updatedElementsGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`updated elements guild (${updatedElementsGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('updated', updatedElements);
|
|
|
|
}, [ sortFunc ]);
|
2022-01-14 05:17:15 +00:00
|
|
|
const onRemoved = useCallback((removedElements: T[], removedElementsGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(currentValue => {
|
|
|
|
if (currentValue === null) return null;
|
|
|
|
const deletedIds = new Set(removedElements.map(deletedElement => deletedElement.id));
|
|
|
|
return currentValue.filter(element => !deletedIds.has(element.id)).sort(sortFunc);
|
|
|
|
});
|
2022-01-14 05:17:15 +00:00
|
|
|
if (removedElementsGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`updated elements guild (${removedElementsGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('removed', removedElements);
|
|
|
|
}, [ sortFunc ]);
|
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
const onConflict = useCallback((changes: Changes<T>, changesGuild: CombinedGuild) => {
|
2021-12-31 03:33:15 +00:00
|
|
|
setValue(currentValue => {
|
|
|
|
if (currentValue === null) return null;
|
|
|
|
const deletedIds = new Set(changes.deleted.map(deletedElement => deletedElement.id));
|
|
|
|
let newValue = currentValue
|
|
|
|
.concat(changes.added)
|
|
|
|
.map(element => changes.updated.find(change => change.newDataPoint.id === element.id)?.newDataPoint ?? element)
|
|
|
|
.filter(element => !deletedIds.has(element.id))
|
|
|
|
.sort(sortFunc);
|
|
|
|
if (newValue.length > maxElements) {
|
|
|
|
newValue = removeByCounts(newValue, getRemoveCounts(newValue.length - maxElements));
|
|
|
|
}
|
|
|
|
return newValue;
|
|
|
|
});
|
2022-01-14 05:17:15 +00:00
|
|
|
if (changesGuild !== guildRef.current) {
|
|
|
|
LOG.warn(`conflict changes guild (${changesGuild.id}) != current guild (${guildRef.current})`);
|
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
events.emit('conflict', changes);
|
|
|
|
}, [ sortFunc, getRemoveCounts ]);
|
|
|
|
|
|
|
|
// I think the typed EventEmitter class isn't ready for this level of insane type safety
|
|
|
|
// otherwise, I may have done this wrong. Forcing it to work with these calls
|
|
|
|
const boundNewFunc = useCallback((...args: Arguments<Connectable[NE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return; // Cancel calls when the guild changes
|
|
|
|
onNew(newEventArgsMap(...args), guild);
|
|
|
|
}, [ guild, onNew, newEventArgsMap ]) as (Connectable & Conflictable)[NE];
|
2021-12-31 03:33:15 +00:00
|
|
|
const boundUpdateFunc = useCallback((...args: Arguments<Connectable[UE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return; // Cancel calls when the guild changes
|
|
|
|
onUpdated(updatedEventArgsMap(...args), guild);
|
2021-12-31 03:33:15 +00:00
|
|
|
}, [ onUpdated, updatedEventArgsMap ]) as (Connectable & Conflictable)[UE];
|
|
|
|
const boundRemovedFunc = useCallback((...args: Arguments<Connectable[RE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return; // Cancel calls when the guild changes
|
|
|
|
onRemoved(removedEventArgsMap(...args), guild);
|
2021-12-31 03:33:15 +00:00
|
|
|
}, [ onRemoved, removedEventArgsMap ]) as (Connectable & Conflictable)[RE];
|
|
|
|
const boundConflictFunc = useCallback((...args: Arguments<Conflictable[CE]>): void => {
|
|
|
|
if (!isMounted.current) return;
|
2022-01-14 05:17:15 +00:00
|
|
|
if (guildRef.current !== guild) return; // Cancel calls when the guild changes
|
|
|
|
onConflict(conflictEventArgsMap(...args), guild);
|
2021-12-31 03:33:15 +00:00
|
|
|
}, [ onConflict, conflictEventArgsMap ]) as (Connectable & Conflictable)[CE];
|
|
|
|
|
|
|
|
const bindEventsFunc = useCallback(() => {
|
|
|
|
guild.on(newEventName, boundNewFunc);
|
|
|
|
guild.on(updatedEventName, boundUpdateFunc);
|
|
|
|
guild.on(removedEventName, boundRemovedFunc);
|
|
|
|
guild.on(conflictEventName, boundConflictFunc);
|
|
|
|
}, [ boundNewFunc, boundUpdateFunc, boundRemovedFunc, boundConflictFunc ]);
|
|
|
|
const unbindEventsFunc = useCallback(() => {
|
|
|
|
guild.off(newEventName, boundNewFunc);
|
|
|
|
guild.off(updatedEventName, boundUpdateFunc);
|
|
|
|
guild.off(removedEventName, boundRemovedFunc);
|
|
|
|
guild.off(conflictEventName, boundConflictFunc);
|
|
|
|
}, [ boundNewFunc, boundUpdateFunc, boundRemovedFunc, boundConflictFunc ]);
|
|
|
|
|
|
|
|
const [ fetchRetryCallable ] = useGuildSubscriptionEffect({
|
|
|
|
guild,
|
|
|
|
onFetch,
|
|
|
|
onFetchError,
|
|
|
|
bindEventsFunc,
|
|
|
|
unbindEventsFunc
|
|
|
|
}, fetchFunc);
|
|
|
|
|
|
|
|
return [
|
|
|
|
fetchRetryCallable,
|
|
|
|
fetchAboveCallable,
|
|
|
|
fetchBelowCallable,
|
|
|
|
setScrollRatio,
|
|
|
|
fetchResult,
|
|
|
|
value,
|
2022-01-14 05:17:15 +00:00
|
|
|
valueGuild,
|
2021-12-31 03:33:15 +00:00
|
|
|
fetchError,
|
|
|
|
fetchAboveError,
|
|
|
|
fetchBelowError,
|
|
|
|
events
|
|
|
|
];
|
|
|
|
}
|
2021-12-13 04:01:30 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
export function useGuildMetadataSubscription(guild: CombinedGuild) {
|
|
|
|
const fetchMetadataFunc = useCallback(async () => {
|
|
|
|
//LOG.silly('fetching metadata for subscription');
|
|
|
|
return await guild.fetchMetadata();
|
|
|
|
}, [ guild ]);
|
|
|
|
return useSingleGuildSubscription<GuildMetadata, 'update-metadata', 'conflict-metadata'>(guild, {
|
|
|
|
updatedEventName: 'update-metadata',
|
|
|
|
updatedEventArgsMap: (guildMeta: GuildMetadata) => guildMeta,
|
|
|
|
conflictEventName: 'conflict-metadata',
|
|
|
|
conflictEventArgsMap: (changesType: AutoVerifierChangesType, oldGuildMeta: GuildMetadata, newGuildMeta: GuildMetadata) => newGuildMeta
|
|
|
|
}, fetchMetadataFunc);
|
|
|
|
}
|
2021-12-13 04:01:30 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
export function useResourceSubscription(guild: CombinedGuild, resourceId: string | null) {
|
|
|
|
const fetchResourceFunc = useCallback(async () => {
|
|
|
|
//LOG.silly('fetching resource for subscription (resourceId: ' + resourceId + ')');
|
|
|
|
if (resourceId === null) return null;
|
|
|
|
return await guild.fetchResource(resourceId);
|
|
|
|
}, [ guild, resourceId ]);
|
|
|
|
return useSingleGuildSubscription<Resource, 'update-resource', 'conflict-resource'>(guild, {
|
|
|
|
updatedEventName: 'update-resource',
|
|
|
|
updatedEventArgsMap: (resource: Resource) => resource,
|
|
|
|
conflictEventName: 'conflict-resource',
|
|
|
|
conflictEventArgsMap: (query: IDQuery, changesType: AutoVerifierChangesType, oldResource: Resource, newResource: Resource) => newResource
|
|
|
|
}, fetchResourceFunc);
|
|
|
|
}
|
2021-12-13 05:25:23 +00:00
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
export function useSoftImageSrcResourceSubscription(guild: CombinedGuild, resourceId: string | null): [ imgSrc: string, resource: Resource | null, resourceGuild: CombinedGuild | null, fetchError: unknown | null ] {
|
|
|
|
const [ resource, resourceGuild, fetchError ] = useResourceSubscription(guild, resourceId);
|
2021-12-25 18:00:20 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
const [ imgSrc ] = useOneTimeAsyncAction(
|
|
|
|
async () => {
|
|
|
|
if (fetchError) return './img/error.png';
|
|
|
|
if (!resource) return './img/loading.svg';
|
|
|
|
return await ElementsUtil.getImageSrcFromBufferFailSoftly(resource.data);
|
|
|
|
},
|
|
|
|
'./img/loading.svg',
|
|
|
|
[ resource, fetchError ]
|
|
|
|
);
|
2021-12-25 18:00:20 +00:00
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
return [ imgSrc, resource, resourceGuild, fetchError ];
|
2021-12-31 03:33:15 +00:00
|
|
|
}
|
2021-12-25 18:00:20 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
export function useChannelsSubscription(guild: CombinedGuild) {
|
|
|
|
const fetchChannelsFunc = useCallback(async () => {
|
|
|
|
return await guild.fetchChannels();
|
|
|
|
}, [ guild ]);
|
|
|
|
return useMultipleGuildSubscription<Channel, 'new-channels', 'update-channels', 'remove-channels', 'conflict-channels'>(guild, {
|
|
|
|
newEventName: 'new-channels',
|
|
|
|
newEventArgsMap: (channels: Channel[]) => channels,
|
|
|
|
updatedEventName: 'update-channels',
|
|
|
|
updatedEventArgsMap: (updatedChannels: Channel[]) => updatedChannels,
|
|
|
|
removedEventName: 'remove-channels',
|
|
|
|
removedEventArgsMap: (removedChannels: Channel[]) => removedChannels,
|
|
|
|
conflictEventName: 'conflict-channels',
|
|
|
|
conflictEventArgsMap: (changesType: AutoVerifierChangesType, changes: Changes<Channel>) => changes,
|
|
|
|
sortFunc: Channel.sortByIndex
|
|
|
|
}, fetchChannelsFunc);
|
|
|
|
}
|
2021-12-17 01:25:55 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
export function useMembersSubscription(guild: CombinedGuild) {
|
|
|
|
const fetchMembersFunc = useCallback(async () => {
|
|
|
|
return await guild.fetchMembers();
|
|
|
|
}, [ guild ]);
|
|
|
|
return useMultipleGuildSubscription<Member, 'new-members', 'update-members', 'remove-members', 'conflict-members'>(guild, {
|
|
|
|
newEventName: 'new-members',
|
|
|
|
newEventArgsMap: (members: Member[]) => members,
|
|
|
|
updatedEventName: 'update-members',
|
|
|
|
updatedEventArgsMap: (updatedMembers: Member[]) => updatedMembers,
|
|
|
|
removedEventName: 'remove-members',
|
|
|
|
removedEventArgsMap: (removedMembers: Member[]) => removedMembers,
|
|
|
|
conflictEventName: 'conflict-members',
|
|
|
|
conflictEventArgsMap: (changesType: AutoVerifierChangesType, changes: Changes<Member>) => changes,
|
|
|
|
sortFunc: Member.sortForList
|
|
|
|
}, fetchMembersFunc);
|
|
|
|
}
|
2021-12-21 05:24:05 +00:00
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
export function useSelfMemberSubscription(guild: CombinedGuild): [ selfMember: Member | null, selfMemberGuild: CombinedGuild | null ] {
|
|
|
|
const [ fetchRetryCallable, members, membersGuild, fetchError ] = useMembersSubscription(guild);
|
2021-12-31 03:33:15 +00:00
|
|
|
|
|
|
|
// TODO: Show an error if we can't fetch and allow retry
|
|
|
|
|
|
|
|
const selfMember = useMemo(() => {
|
|
|
|
if (members) {
|
|
|
|
const member = members.find(m => m.id === guild.memberId);
|
|
|
|
if (!member) {
|
|
|
|
LOG.warn('Unable to find self in members');
|
|
|
|
return null;
|
2021-12-25 18:00:20 +00:00
|
|
|
}
|
2021-12-31 03:33:15 +00:00
|
|
|
return member;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}, [ guild.memberId, members ]);
|
2021-12-25 18:00:20 +00:00
|
|
|
|
2022-01-14 05:17:15 +00:00
|
|
|
return [ selfMember, membersGuild ];
|
2021-12-31 03:33:15 +00:00
|
|
|
}
|
2021-12-25 18:00:20 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
export function useTokensSubscription(guild: CombinedGuild) {
|
|
|
|
const fetchTokensFunc = useCallback(async () => {
|
|
|
|
//LOG.silly('fetching tokens for subscription');
|
|
|
|
return await guild.fetchTokens();
|
|
|
|
}, [ guild ]);
|
|
|
|
return useMultipleGuildSubscription<Token, 'new-tokens', 'update-tokens', 'remove-tokens', 'conflict-tokens'>(guild, {
|
|
|
|
newEventName: 'new-tokens',
|
|
|
|
newEventArgsMap: (tokens: Token[]) => tokens,
|
|
|
|
updatedEventName: 'update-tokens',
|
|
|
|
updatedEventArgsMap: (updatedTokens: Token[]) => updatedTokens,
|
|
|
|
removedEventName: 'remove-tokens',
|
|
|
|
removedEventArgsMap: (removedTokens: Token[]) => removedTokens,
|
|
|
|
conflictEventName: 'conflict-tokens',
|
|
|
|
conflictEventArgsMap: (changesType: AutoVerifierChangesType, changes: Changes<Token>) => changes,
|
|
|
|
sortFunc: Token.sortRecentCreatedFirst
|
|
|
|
}, fetchTokensFunc);
|
|
|
|
}
|
2021-12-24 23:37:27 +00:00
|
|
|
|
2021-12-31 03:33:15 +00:00
|
|
|
export function useMessagesScrollingSubscription(guild: CombinedGuild, channel: Channel) {
|
|
|
|
const maxFetchElements = Globals.MESSAGES_PER_REQUEST;
|
|
|
|
const maxElements = Globals.MAX_CURRENT_MESSAGES;
|
|
|
|
const fetchMessagesFunc = useCallback(async () => {
|
|
|
|
return await guild.fetchMessagesRecent(channel.id, maxFetchElements);
|
|
|
|
}, [ guild, channel.id, maxFetchElements ]);
|
|
|
|
const fetchAboveFunc = useCallback(async (reference: Message) => {
|
|
|
|
return await guild.fetchMessagesBefore(channel.id, reference.id, maxFetchElements);
|
|
|
|
}, [ guild, channel.id, maxFetchElements ]);
|
|
|
|
const fetchBelowFunc = useCallback(async (reference: Message) => {
|
|
|
|
return await guild.fetchMessagesAfter(channel.id, reference.id, maxFetchElements);
|
|
|
|
}, [ guild, channel.id, maxFetchElements ]);
|
|
|
|
return useMultipleGuildSubscriptionScrolling(
|
|
|
|
guild, {
|
|
|
|
newEventName: 'new-messages',
|
|
|
|
newEventArgsMap: (messages: Message[]) => messages,
|
|
|
|
updatedEventName: 'update-messages',
|
|
|
|
updatedEventArgsMap: (updatedMessages) => updatedMessages,
|
|
|
|
removedEventName: 'remove-messages',
|
|
|
|
removedEventArgsMap: (removedMessages) => removedMessages,
|
|
|
|
conflictEventName: 'conflict-messages',
|
|
|
|
conflictEventArgsMap: (query: PartialMessageListQuery, changesType: AutoVerifierChangesType, changes: Changes<Message>) => changes,
|
|
|
|
sortFunc: Message.sortOrder
|
|
|
|
},
|
|
|
|
maxElements, maxFetchElements,
|
|
|
|
fetchMessagesFunc, fetchAboveFunc, fetchBelowFunc
|
|
|
|
)
|
2021-12-13 04:01:30 +00:00
|
|
|
}
|