import { useCallback, useEffect, useState } from 'react'; import type EventEmitter from 'events'; import { useRouter } from 'next/router'; import type { Element } from 'react-markdown/lib/rehype-filter'; import { useIsGuestUser, useIsReadOnlyUser, useIsSharedUser, useShareLinkId, } from '~/stores/context'; import loggerFactory from '~/utils/logger'; import { NextLink } from './NextLink'; import styles from './Header.module.scss'; const logger = loggerFactory('growi:components:Header'); declare global { // eslint-disable-next-line vars-on-top, no-var var globalEmitter: EventEmitter; } function setCaretLine(line?: number): void { if (line != null) { globalEmitter.emit('setCaretLine', line); } } type EditLinkProps = { line?: number, } /** * Inner FC to display edit link icon */ const EditLink = (props: EditLinkProps): JSX.Element => { const isDisabled = props.line == null; return ( setCaretLine(props.line)}> edit_square ); }; type HeaderProps = { children: React.ReactNode, node: Element, level: number, id?: string, } export const Header = (props: HeaderProps): JSX.Element => { const { node, id, children, level, } = props; const { data: isGuestUser } = useIsGuestUser(); const { data: isReadOnlyUser } = useIsReadOnlyUser(); const { data: isSharedUser } = useIsSharedUser(); const { data: shareLinkId } = useShareLinkId(); const router = useRouter(); const [isActive, setActive] = useState(false); const CustomTag = `h${level}` as keyof JSX.IntrinsicElements; const activateByHash = useCallback((url: string) => { try { const hash = (new URL(url, 'https://example.com')).hash.slice(1); setActive(decodeURIComponent(hash) === id); } catch (err) { logger.debug(err); setActive(false); } }, [id]); // init useEffect(() => { activateByHash(window.location.href); }, [activateByHash]); // update isActive when hash is changed by next router useEffect(() => { router.events.on('hashChangeComplete', activateByHash); return () => { router.events.off('hashChangeComplete', activateByHash); }; }, [activateByHash, router.events]); // update isActive when hash is changed useEffect(() => { const activeByHashWrapper = (e: HashChangeEvent) => { activateByHash(e.newURL); }; window.addEventListener('hashchange', activeByHashWrapper); return () => { window.removeEventListener('hashchange', activeByHashWrapper); }; }, [activateByHash, router.events]); const showEditButton = !isGuestUser && !isReadOnlyUser && !isSharedUser && shareLinkId == null; return ( {children} link {showEditButton && ( )} ); };