import { useQueryClient } from '@tanstack/react-query'; import React, { useState, useEffect, useRef } from 'react'; import { FormattedMessage } from 'react-intl'; import { Stack, HStack, Card, Avatar, Text, Icon } from 'soapbox/components/ui'; import IconButton from 'soapbox/components/ui/icon-button/icon-button'; import StatusCard from 'soapbox/features/status/components/card'; import { useAppSelector } from 'soapbox/hooks'; import { adKeys } from 'soapbox/queries/ads'; import type { Card as CardEntity } from 'soapbox/types/entities'; interface IAd { /** Embedded ad data in Card format (almost like OEmbed). */ card: CardEntity, /** Impression URL to fetch upon display. */ impression?: string, /** Time when the ad expires and should no longer be displayed. */ expires?: Date, } /** Displays an ad in sponsored post format. */ const Ad: React.FC = ({ card, impression, expires }) => { const queryClient = useQueryClient(); const instance = useAppSelector(state => state.instance); const timer = useRef(undefined); const infobox = useRef(null); const [showInfo, setShowInfo] = useState(false); /** Invalidate query cache for ads. */ const bustCache = (): void => { queryClient.invalidateQueries(adKeys.ads); }; /** Toggle the info box on click. */ const handleInfoButtonClick: React.MouseEventHandler = () => { setShowInfo(!showInfo); }; /** Hide the info box when clicked outside. */ const handleClickOutside = (event: MouseEvent) => { if (event.target && infobox.current && !infobox.current.contains(event.target as any)) { setShowInfo(false); } }; // Hide the info box when clicked outside. // https://stackoverflow.com/a/42234988 useEffect(() => { document.addEventListener('mousedown', handleClickOutside); return () => { document.removeEventListener('mousedown', handleClickOutside); }; }, [infobox]); // Fetch the impression URL (if any) upon displaying the ad. // It's common for ad providers to provide this. useEffect(() => { if (impression) { fetch(impression); } }, [impression]); // Wait until the ad expires, then invalidate cache. useEffect(() => { if (expires) { const delta = expires.getTime() - (new Date()).getTime(); timer.current = setTimeout(bustCache, delta); } return () => { if (timer.current) { clearTimeout(timer.current); } }; }, [expires]); return (
{instance.title} {}} horizontal /> {showInfo && (
)}
); }; export default Ad;