SearchResultContent.tsx 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. import React, {
  2. FC, useCallback, useEffect, useRef,
  3. } from 'react';
  4. import { getIdForRef } from '@growi/core';
  5. import { useTranslation } from 'next-i18next';
  6. import dynamic from 'next/dynamic';
  7. import { DropdownItem } from 'reactstrap';
  8. import { exportAsMarkdown } from '~/client/services/page-operation';
  9. import { toastSuccess } from '~/client/util/apiNotification';
  10. import { smoothScrollIntoView } from '~/client/util/smooth-scroll';
  11. import { IPageToDeleteWithMeta, IPageToRenameWithMeta } from '~/interfaces/page';
  12. import { IPageWithSearchMeta } from '~/interfaces/search';
  13. import { OnDuplicatedFunction, OnRenamedFunction, OnDeletedFunction } from '~/interfaces/ui';
  14. import { useCurrentUser } from '~/stores/context';
  15. import {
  16. usePageDuplicateModal, usePageRenameModal, usePageDeleteModal,
  17. } from '~/stores/modal';
  18. import { useDescendantsPageListForCurrentPathTermManager, usePageTreeTermManager } from '~/stores/page-listing';
  19. import { useSearchResultOptions } from '~/stores/renderer';
  20. import { useFullTextSearchTermManager } from '~/stores/search';
  21. import { AdditionalMenuItemsRendererProps, ForceHideMenuItems } from '../Common/Dropdown/PageItemControl';
  22. import { GrowiSubNavigationProps } from '../Navbar/GrowiSubNavigation';
  23. import { SubNavButtonsProps } from '../Navbar/SubNavButtons';
  24. import { RevisionLoaderProps } from '../Page/RevisionLoader';
  25. import { PageCommentProps } from '../PageComment';
  26. import { PageContentFooterProps } from '../PageContentFooter';
  27. const GrowiSubNavigation = dynamic<GrowiSubNavigationProps>(() => import('../Navbar/GrowiSubNavigation').then(mod => mod.GrowiSubNavigation), { ssr: false });
  28. const SubNavButtons = dynamic<SubNavButtonsProps>(() => import('../Navbar/SubNavButtons').then(mod => mod.SubNavButtons), { ssr: false });
  29. const RevisionLoader = dynamic<RevisionLoaderProps>(() => import('../Page/RevisionLoader').then(mod => mod.RevisionLoader), { ssr: false });
  30. const PageComment = dynamic<PageCommentProps>(() => import('../PageComment').then(mod => mod.PageComment), { ssr: false });
  31. const PageContentFooter = dynamic<PageContentFooterProps>(() => import('../PageContentFooter').then(mod => mod.PageContentFooter), { ssr: false });
  32. type AdditionalMenuItemsProps = AdditionalMenuItemsRendererProps & {
  33. pageId: string,
  34. revisionId: string,
  35. }
  36. const AdditionalMenuItems = (props: AdditionalMenuItemsProps): JSX.Element => {
  37. const { t } = useTranslation();
  38. const { pageId, revisionId } = props;
  39. return (
  40. // Export markdown
  41. <DropdownItem
  42. onClick={() => exportAsMarkdown(pageId, revisionId, 'md')}
  43. className="grw-page-control-dropdown-item"
  44. >
  45. <i className="icon-fw icon-cloud-download grw-page-control-dropdown-icon"></i>
  46. {t('export_bulk.export_page_markdown')}
  47. </DropdownItem>
  48. );
  49. };
  50. const SCROLL_OFFSET_TOP = 175; // approximate height of (navigation + subnavigation)
  51. const MUTATION_OBSERVER_CONFIG = { childList: true, subtree: true };
  52. type Props ={
  53. pageWithMeta : IPageWithSearchMeta,
  54. highlightKeywords?: string[],
  55. showPageControlDropdown?: boolean,
  56. forceHideMenuItems?: ForceHideMenuItems,
  57. }
  58. const scrollTo = (scrollElement:HTMLElement) => {
  59. // use querySelector to intentionally get the first element found
  60. const highlightedKeyword = scrollElement.querySelector('.highlighted-keyword') as HTMLElement | null;
  61. if (highlightedKeyword != null) {
  62. smoothScrollIntoView(highlightedKeyword, SCROLL_OFFSET_TOP, scrollElement);
  63. }
  64. };
  65. const generateObserverCallback = (doScroll: ()=>void) => {
  66. return (mutationRecords:MutationRecord[]) => {
  67. mutationRecords.forEach((record:MutationRecord) => {
  68. const target = record.target as HTMLElement;
  69. const targetId = target.id as string;
  70. if (targetId !== 'wiki') return;
  71. doScroll();
  72. });
  73. };
  74. };
  75. export const SearchResultContent: FC<Props> = (props: Props) => {
  76. const scrollElementRef = useRef(null);
  77. // for mutation
  78. const { advance: advancePt } = usePageTreeTermManager();
  79. const { advance: advanceFts } = useFullTextSearchTermManager();
  80. const { advance: advanceDpl } = useDescendantsPageListForCurrentPathTermManager();
  81. // *************************** Auto Scroll ***************************
  82. useEffect(() => {
  83. const scrollElement = scrollElementRef.current as HTMLElement | null;
  84. if (scrollElement == null) return;
  85. const observerCallback = generateObserverCallback(() => {
  86. scrollTo(scrollElement);
  87. });
  88. const observer = new MutationObserver(observerCallback);
  89. observer.observe(scrollElement, MUTATION_OBSERVER_CONFIG);
  90. return () => {
  91. observer.disconnect();
  92. };
  93. });
  94. // ******************************* end *******************************
  95. const {
  96. pageWithMeta,
  97. highlightKeywords,
  98. showPageControlDropdown,
  99. forceHideMenuItems,
  100. } = props;
  101. const { t } = useTranslation();
  102. const page = pageWithMeta?.data;
  103. const { open: openDuplicateModal } = usePageDuplicateModal();
  104. const { open: openRenameModal } = usePageRenameModal();
  105. const { open: openDeleteModal } = usePageDeleteModal();
  106. const { data: rendererOptions } = useSearchResultOptions(pageWithMeta.data.path, highlightKeywords);
  107. const { data: currentUser } = useCurrentUser();
  108. const duplicateItemClickedHandler = useCallback(async(pageToDuplicate) => {
  109. // eslint-disable-next-line @typescript-eslint/no-unused-vars
  110. const duplicatedHandler: OnDuplicatedFunction = (fromPath, toPath) => {
  111. toastSuccess(t('duplicated_pages', { fromPath }));
  112. advancePt();
  113. advanceFts();
  114. advanceDpl();
  115. };
  116. openDuplicateModal(pageToDuplicate, { onDuplicated: duplicatedHandler });
  117. }, [advanceDpl, advanceFts, advancePt, openDuplicateModal, t]);
  118. const renameItemClickedHandler = useCallback((pageToRename: IPageToRenameWithMeta) => {
  119. const renamedHandler: OnRenamedFunction = (path) => {
  120. toastSuccess(t('renamed_pages', { path }));
  121. advancePt();
  122. advanceFts();
  123. advanceDpl();
  124. };
  125. openRenameModal(pageToRename, { onRenamed: renamedHandler });
  126. }, [advanceDpl, advanceFts, advancePt, openRenameModal, t]);
  127. const onDeletedHandler: OnDeletedFunction = useCallback((pathOrPathsToDelete, isRecursively, isCompletely) => {
  128. if (typeof pathOrPathsToDelete !== 'string') {
  129. return;
  130. }
  131. const path = pathOrPathsToDelete;
  132. if (isCompletely) {
  133. toastSuccess(t('deleted_pages_completely', { path }));
  134. }
  135. else {
  136. toastSuccess(t('deleted_pages', { path }));
  137. }
  138. advancePt();
  139. advanceFts();
  140. advanceDpl();
  141. }, [advanceDpl, advanceFts, advancePt, t]);
  142. const deleteItemClickedHandler = useCallback((pageToDelete: IPageToDeleteWithMeta) => {
  143. openDeleteModal([pageToDelete], { onDeleted: onDeletedHandler });
  144. }, [onDeletedHandler, openDeleteModal]);
  145. const RightComponent = useCallback(() => {
  146. if (page == null) {
  147. return <></>;
  148. }
  149. const revisionId = getIdForRef(page.revision);
  150. return (
  151. <div className="d-flex flex-column align-items-end justify-content-center py-md-2">
  152. <SubNavButtons
  153. pageId={page._id}
  154. revisionId={revisionId}
  155. path={page.path}
  156. showPageControlDropdown={showPageControlDropdown}
  157. forceHideMenuItems={forceHideMenuItems}
  158. additionalMenuItemRenderer={props => <AdditionalMenuItems {...props} pageId={page._id} revisionId={revisionId} />}
  159. isCompactMode
  160. onClickDuplicateMenuItem={duplicateItemClickedHandler}
  161. onClickRenameMenuItem={renameItemClickedHandler}
  162. onClickDeleteMenuItem={deleteItemClickedHandler}
  163. />
  164. </div>
  165. );
  166. }, [page, showPageControlDropdown, forceHideMenuItems, duplicateItemClickedHandler, renameItemClickedHandler, deleteItemClickedHandler]);
  167. // return if page or growiRenderer is null
  168. if (page == null || rendererOptions == null) return <></>;
  169. return (
  170. <div key={page._id} data-testid="search-result-content" className="search-result-content grw-page-path-text-muted-container d-flex flex-column">
  171. <div className="grw-subnav-append-shadow-container">
  172. <GrowiSubNavigation
  173. pagePath={page.path}
  174. pageId={page._id}
  175. rightComponent={RightComponent}
  176. isCompactMode
  177. additionalClasses={['px-4']}
  178. />
  179. </div>
  180. <div className="search-result-content-body-container" ref={scrollElementRef}>
  181. <RevisionLoader
  182. rendererOptions={rendererOptions}
  183. pageId={page._id}
  184. pagePath={page.path}
  185. revisionId={page.revision}
  186. highlightKeywords={highlightKeywords}
  187. />
  188. <PageComment
  189. rendererOptions={rendererOptions}
  190. pageId={page._id}
  191. revision={page.revision}
  192. currentUser={currentUser}
  193. highlightKeywords={highlightKeywords}
  194. isReadOnly
  195. hideIfEmpty
  196. />
  197. <PageContentFooter
  198. page={page}
  199. />
  200. </div>
  201. </div>
  202. );
  203. };