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 { NextLink } from '~/components/ReactMarkdownComponents/NextLink'; import { useIsGuestUser, useIsReadOnlyUser, useIsSharedUser, useShareLinkId, } from '~/stores-universal/context'; import { useCurrentPageYjsData } from '~/stores/yjs'; import loggerFactory from '~/utils/logger'; import styles from './Header.module.scss'; const logger = loggerFactory('growi:components:Header'); const moduleClass = styles['revision-head'] ?? ''; declare global { // eslint-disable-next-line vars-on-top, no-var var globalEmitter: EventEmitter; } function setCaretLine(line?: number): void { if (line != null) { globalEmitter.emit('reservedNextCaretLine', 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 { data: currentPageYjsData } = useCurrentPageYjsData(); 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]); // TODO: currentPageYjsData?.hasYdocsNewerThanLatestRevision === false make to hide the edit button when a Yjs draft exists // This is because the current conditional logic cannot handle cases where the draft is an empty string. // It will be possible to address this TODO ySyncAnnotation become available for import. // Ref: https://github.com/yjs/y-codemirror.next/pull/30 const showEditButton = !isGuestUser && !isReadOnlyUser && !isSharedUser && shareLinkId == null && currentPageYjsData?.hasYdocsNewerThanLatestRevision === false; return ( <> # {children} { showEditButton && ( ) } ); };