2023-10-21 21:44:08 +02:00
|
|
|
import { useEffect } from "react";
|
|
|
|
import { useMeasure } from "react-use";
|
|
|
|
import { create } from "zustand";
|
|
|
|
import { immer } from "zustand/middleware/immer";
|
|
|
|
|
|
|
|
interface BannerInstance {
|
|
|
|
id: string;
|
|
|
|
height: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface BannerStore {
|
|
|
|
banners: BannerInstance[];
|
|
|
|
isOnline: boolean;
|
2024-01-03 23:54:08 +01:00
|
|
|
isTurnstile: boolean;
|
2023-10-21 21:44:08 +02:00
|
|
|
location: string | null;
|
|
|
|
updateHeight(id: string, height: number): void;
|
|
|
|
showBanner(id: string): void;
|
|
|
|
hideBanner(id: string): void;
|
|
|
|
setLocation(loc: string | null): void;
|
|
|
|
updateOnline(isOnline: boolean): void;
|
2024-01-03 23:54:08 +01:00
|
|
|
updateTurnstile(isTurnstile: boolean): void;
|
2023-10-21 21:44:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export const useBannerStore = create(
|
|
|
|
immer<BannerStore>((set) => ({
|
|
|
|
banners: [],
|
|
|
|
isOnline: true,
|
2024-01-03 23:54:08 +01:00
|
|
|
isTurnstile: false,
|
2023-10-21 21:44:08 +02:00
|
|
|
location: null,
|
|
|
|
updateOnline(isOnline) {
|
|
|
|
set((s) => {
|
|
|
|
s.isOnline = isOnline;
|
|
|
|
});
|
|
|
|
},
|
2024-01-03 23:54:08 +01:00
|
|
|
updateTurnstile(isTurnstile) {
|
|
|
|
set((s) => {
|
|
|
|
s.isTurnstile = isTurnstile;
|
|
|
|
});
|
|
|
|
},
|
2023-10-21 21:44:08 +02:00
|
|
|
setLocation(loc) {
|
|
|
|
set((s) => {
|
|
|
|
s.location = loc;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
showBanner(id) {
|
|
|
|
set((s) => {
|
|
|
|
if (s.banners.find((v) => v.id === id)) return;
|
|
|
|
s.banners.push({
|
|
|
|
id,
|
|
|
|
height: 0,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
hideBanner(id) {
|
|
|
|
set((s) => {
|
|
|
|
s.banners = s.banners.filter((v) => v.id !== id);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
updateHeight(id, height) {
|
|
|
|
set((s) => {
|
|
|
|
const found = s.banners.find((v) => v.id === id);
|
|
|
|
if (found) found.height = height;
|
|
|
|
});
|
|
|
|
},
|
2023-12-23 06:24:43 +01:00
|
|
|
})),
|
2023-10-21 21:44:08 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
export function useBannerSize(location?: string) {
|
|
|
|
const loc = location ?? null;
|
|
|
|
const banners = useBannerStore((s) => s.banners);
|
|
|
|
const currentLocation = useBannerStore((s) => s.location);
|
|
|
|
|
|
|
|
const size = banners.reduce((a, v) => a + v.height, 0);
|
|
|
|
if (loc !== currentLocation) return 0;
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function useRegisterBanner<T extends Element>(id: string) {
|
|
|
|
const [ref, { height }] = useMeasure<T>();
|
|
|
|
const updateHeight = useBannerStore((s) => s.updateHeight);
|
|
|
|
const showBanner = useBannerStore((s) => s.showBanner);
|
|
|
|
const hideBanner = useBannerStore((s) => s.hideBanner);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
showBanner(id);
|
|
|
|
return () => {
|
|
|
|
hideBanner(id);
|
|
|
|
};
|
|
|
|
}, [showBanner, hideBanner, id]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
updateHeight(id, height);
|
|
|
|
}, [height, id, updateHeight]);
|
|
|
|
|
|
|
|
return [ref];
|
|
|
|
}
|