ui.tsx 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474
  1. import { RefObject } from 'react';
  2. import {
  3. isClient, isServer, pagePathUtils, Nullable,
  4. } from '@growi/core';
  5. import { withUtils, SWRResponseWithUtils } from '@growi/core/src/utils/with-utils';
  6. import { Breakpoint, addBreakpointListener } from '@growi/ui';
  7. import SimpleBar from 'simplebar-react';
  8. import {
  9. useSWRConfig, SWRResponse, Key, Fetcher,
  10. } from 'swr';
  11. import useSWRImmutable from 'swr/immutable';
  12. import { IFocusable } from '~/client/interfaces/focusable';
  13. import { useUserUISettings } from '~/client/services/user-ui-settings';
  14. import { apiv3Get, apiv3Put } from '~/client/util/apiv3-client';
  15. import { IPageGrantData } from '~/interfaces/page';
  16. import { ISidebarConfig } from '~/interfaces/sidebar-config';
  17. import { SidebarContentsType } from '~/interfaces/ui';
  18. import { UpdateDescCountData } from '~/interfaces/websocket';
  19. import loggerFactory from '~/utils/logger';
  20. import {
  21. useCurrentPageId, useCurrentPagePath, useIsEditable, useIsTrashPage, useIsUserPage, useIsGuestUser,
  22. useIsNotCreatable, useIsSharedUser, useIsForbidden, useIsIdenticalPath, useCurrentUser, useIsNotFound, useShareLinkId,
  23. } from './context';
  24. import { localStorageMiddleware } from './middlewares/sync-to-storage';
  25. import { useStaticSWR } from './use-static-swr';
  26. const { isTrashTopPage } = pagePathUtils;
  27. const logger = loggerFactory('growi:stores:ui');
  28. /** **********************************************************
  29. * Unions
  30. *********************************************************** */
  31. export const EditorMode = {
  32. View: 'view',
  33. Editor: 'editor',
  34. HackMD: 'hackmd',
  35. } as const;
  36. export type EditorMode = typeof EditorMode[keyof typeof EditorMode];
  37. /** **********************************************************
  38. * Storing RefObjects
  39. *********************************************************** */
  40. export const useSidebarScrollerRef = (initialData?: RefObject<SimpleBar>): SWRResponse<RefObject<SimpleBar>, Error> => {
  41. return useStaticSWR<RefObject<SimpleBar>, Error>('sidebarScrollerRef', initialData);
  42. };
  43. /** **********************************************************
  44. * SWR Hooks
  45. * for switching UI
  46. *********************************************************** */
  47. export const useIsMobile = (): SWRResponse<boolean, Error> => {
  48. const key = isClient() ? 'isMobile' : null;
  49. let configuration;
  50. if (isClient()) {
  51. const userAgent = window.navigator.userAgent.toLowerCase();
  52. configuration = {
  53. fallbackData: /iphone|ipad|android/.test(userAgent),
  54. };
  55. }
  56. return useStaticSWR<boolean, Error>(key, undefined, configuration);
  57. };
  58. const updateBodyClassesByEditorMode = (newEditorMode: EditorMode, isSidebar = false) => {
  59. const bodyElement = document.getElementsByTagName('body')[0];
  60. if (bodyElement == null) {
  61. logger.warn('The body tag was not successfully obtained');
  62. return;
  63. }
  64. switch (newEditorMode) {
  65. case EditorMode.View:
  66. bodyElement.classList.remove('on-edit', 'builtin-editor', 'hackmd', 'editing-sidebar');
  67. break;
  68. case EditorMode.Editor:
  69. bodyElement.classList.add('on-edit', 'builtin-editor');
  70. bodyElement.classList.remove('hackmd');
  71. // editing /Sidebar
  72. if (isSidebar) {
  73. bodyElement.classList.add('editing-sidebar');
  74. }
  75. break;
  76. case EditorMode.HackMD:
  77. bodyElement.classList.add('on-edit', 'hackmd');
  78. bodyElement.classList.remove('builtin-editor', 'editing-sidebar');
  79. break;
  80. }
  81. };
  82. const updateHashByEditorMode = (newEditorMode: EditorMode) => {
  83. const { pathname } = window.location;
  84. switch (newEditorMode) {
  85. case EditorMode.View:
  86. window.history.replaceState(null, '', pathname);
  87. break;
  88. case EditorMode.Editor:
  89. window.history.replaceState(null, '', `${pathname}#edit`);
  90. break;
  91. case EditorMode.HackMD:
  92. window.history.replaceState(null, '', `${pathname}#hackmd`);
  93. break;
  94. }
  95. };
  96. export const determineEditorModeByHash = (): EditorMode => {
  97. if (isServer()) {
  98. return EditorMode.View;
  99. }
  100. const { hash } = window.location;
  101. switch (hash) {
  102. case '#edit':
  103. return EditorMode.Editor;
  104. case '#hackmd':
  105. return EditorMode.HackMD;
  106. default:
  107. return EditorMode.View;
  108. }
  109. };
  110. let isEditorModeLoaded = false;
  111. export const useEditorMode = (): SWRResponse<EditorMode, Error> => {
  112. const { data: _isEditable } = useIsEditable();
  113. const editorModeByHash = determineEditorModeByHash();
  114. const isLoading = _isEditable === undefined;
  115. const isEditable = !isLoading && _isEditable;
  116. const initialData = isEditable ? editorModeByHash : EditorMode.View;
  117. const { data: currentPagePath } = useCurrentPagePath();
  118. const isSidebar = currentPagePath === '/Sidebar';
  119. const swrResponse = useSWRImmutable(
  120. isLoading ? null : ['editorMode', isEditable],
  121. null,
  122. { fallbackData: initialData },
  123. );
  124. // initial updating
  125. if (!isEditorModeLoaded && !isLoading && swrResponse.data != null) {
  126. if (isEditable) {
  127. updateBodyClassesByEditorMode(swrResponse.data, isSidebar);
  128. }
  129. isEditorModeLoaded = true;
  130. }
  131. return {
  132. ...swrResponse,
  133. // overwrite mutate
  134. mutate: (editorMode: EditorMode, shouldRevalidate?: boolean) => {
  135. if (!isEditable) {
  136. return Promise.resolve(EditorMode.View); // fixed if not editable
  137. }
  138. updateBodyClassesByEditorMode(editorMode, isSidebar);
  139. updateHashByEditorMode(editorMode);
  140. return swrResponse.mutate(editorMode, shouldRevalidate);
  141. },
  142. };
  143. };
  144. export const useIsDeviceSmallerThanMd = (): SWRResponse<boolean, Error> => {
  145. const key: Key = isClient() ? 'isDeviceSmallerThanMd' : null;
  146. const { cache, mutate } = useSWRConfig();
  147. if (isClient()) {
  148. const mdOrAvobeHandler = function(this: MediaQueryList): void {
  149. // sm -> md: matches will be true
  150. // md -> sm: matches will be false
  151. mutate(key, !this.matches);
  152. };
  153. const mql = addBreakpointListener(Breakpoint.MD, mdOrAvobeHandler);
  154. // initialize
  155. if (cache.get(key) == null) {
  156. document.addEventListener('DOMContentLoaded', () => {
  157. mutate(key, !mql.matches);
  158. });
  159. }
  160. }
  161. return useStaticSWR(key);
  162. };
  163. export const useIsDeviceSmallerThanLg = (): SWRResponse<boolean, Error> => {
  164. const key: Key = isClient() ? 'isDeviceSmallerThanLg' : null;
  165. const { cache, mutate } = useSWRConfig();
  166. if (isClient()) {
  167. const lgOrAvobeHandler = function(this: MediaQueryList): void {
  168. // md -> lg: matches will be true
  169. // lg -> md: matches will be false
  170. mutate(key, !this.matches);
  171. };
  172. const mql = addBreakpointListener(Breakpoint.LG, lgOrAvobeHandler);
  173. // initialize
  174. if (cache.get(key) == null) {
  175. document.addEventListener('DOMContentLoaded', () => {
  176. mutate(key, !mql.matches);
  177. });
  178. }
  179. }
  180. return useStaticSWR(key);
  181. };
  182. type PreferDrawerModeByUserUtils = {
  183. update: (preferDrawerMode: boolean) => void
  184. }
  185. export const usePreferDrawerModeByUser = (initialData?: boolean): SWRResponseWithUtils<PreferDrawerModeByUserUtils, boolean> => {
  186. const { data: isGuestUser } = useIsGuestUser();
  187. const { scheduleToPut } = useUserUISettings();
  188. const swrResponse: SWRResponse<boolean, Error> = useStaticSWR('preferDrawerModeByUser', initialData, { use: isGuestUser ? [localStorageMiddleware] : [] });
  189. const utils: PreferDrawerModeByUserUtils = {
  190. update: (preferDrawerMode: boolean) => {
  191. swrResponse.mutate(preferDrawerMode);
  192. if (!isGuestUser) {
  193. scheduleToPut({ preferDrawerModeByUser: preferDrawerMode });
  194. }
  195. },
  196. };
  197. return withUtils(swrResponse, utils);
  198. };
  199. export const usePreferDrawerModeOnEditByUser = (initialData?: boolean): SWRResponse<boolean, Error> => {
  200. return useStaticSWR('preferDrawerModeOnEditByUser', initialData, { fallbackData: true });
  201. };
  202. export const useSidebarCollapsed = (initialData?: boolean): SWRResponse<boolean, Error> => {
  203. return useStaticSWR('isSidebarCollapsed', initialData, { fallbackData: false });
  204. };
  205. export const useCurrentSidebarContents = (initialData?: SidebarContentsType): SWRResponse<SidebarContentsType, Error> => {
  206. return useStaticSWR('sidebarContents', initialData, { fallbackData: SidebarContentsType.TREE });
  207. };
  208. export const useCurrentProductNavWidth = (initialData?: number): SWRResponse<number, Error> => {
  209. return useStaticSWR('productNavWidth', initialData, { fallbackData: 320 });
  210. };
  211. export const useDrawerMode = (): SWRResponse<boolean, Error> => {
  212. const { data: preferDrawerModeByUser } = usePreferDrawerModeByUser();
  213. const { data: preferDrawerModeOnEditByUser } = usePreferDrawerModeOnEditByUser();
  214. const { data: editorMode } = useEditorMode();
  215. const { data: isDeviceSmallerThanMd } = useIsDeviceSmallerThanMd();
  216. const condition = editorMode != null || preferDrawerModeByUser != null || preferDrawerModeOnEditByUser != null || isDeviceSmallerThanMd != null;
  217. const calcDrawerMode: Fetcher<boolean> = (
  218. key: Key, editorMode: EditorMode, preferDrawerModeByUser: boolean, preferDrawerModeOnEditByUser: boolean, isDeviceSmallerThanMd: boolean,
  219. ): boolean => {
  220. // get preference on view or edit
  221. const preferDrawerMode = editorMode !== EditorMode.View ? preferDrawerModeOnEditByUser : preferDrawerModeByUser;
  222. return isDeviceSmallerThanMd || preferDrawerMode;
  223. };
  224. const isViewModeWithPreferDrawerMode = editorMode === EditorMode.View && preferDrawerModeByUser;
  225. const isEditModeWithPreferDrawerMode = editorMode === EditorMode.Editor && preferDrawerModeOnEditByUser;
  226. const useFallbackData = isViewModeWithPreferDrawerMode || isEditModeWithPreferDrawerMode;
  227. const fallbackOption = useFallbackData
  228. ? { fallbackData: true }
  229. : { fallback: calcDrawerMode };
  230. return useSWRImmutable(
  231. condition ? ['isDrawerMode', editorMode, preferDrawerModeByUser, preferDrawerModeOnEditByUser, isDeviceSmallerThanMd] : null,
  232. calcDrawerMode,
  233. fallbackOption,
  234. );
  235. };
  236. type SidebarConfigOption = {
  237. update: () => Promise<void>,
  238. isSidebarDrawerMode: boolean|undefined,
  239. isSidebarClosedAtDockMode: boolean|undefined,
  240. setIsSidebarDrawerMode: (isSidebarDrawerMode: boolean) => void,
  241. setIsSidebarClosedAtDockMode: (isSidebarClosedAtDockMode: boolean) => void
  242. }
  243. export const useSWRxSidebarConfig = (): SWRResponse<ISidebarConfig, Error> & SidebarConfigOption => {
  244. const swrResponse = useSWRImmutable<ISidebarConfig>(
  245. '/customize-setting/sidebar',
  246. endpoint => apiv3Get(endpoint).then(result => result.data),
  247. );
  248. return {
  249. ...swrResponse,
  250. update: async() => {
  251. const { data } = swrResponse;
  252. if (data == null) {
  253. return;
  254. }
  255. const { isSidebarDrawerMode, isSidebarClosedAtDockMode } = data;
  256. const updateData = {
  257. isSidebarDrawerMode,
  258. isSidebarClosedAtDockMode,
  259. };
  260. // invoke API
  261. await apiv3Put('/customize-setting/sidebar', updateData);
  262. },
  263. isSidebarDrawerMode: swrResponse.data?.isSidebarDrawerMode,
  264. isSidebarClosedAtDockMode: swrResponse.data?.isSidebarClosedAtDockMode,
  265. setIsSidebarDrawerMode: (isSidebarDrawerMode) => {
  266. const { data, mutate } = swrResponse;
  267. if (data == null) {
  268. return;
  269. }
  270. const updateData = {
  271. isSidebarDrawerMode,
  272. };
  273. // update isSidebarDrawerMode in cache, not revalidate
  274. mutate({ ...data, ...updateData }, false);
  275. },
  276. setIsSidebarClosedAtDockMode: (isSidebarClosedAtDockMode) => {
  277. const { data, mutate } = swrResponse;
  278. if (data == null) {
  279. return;
  280. }
  281. const updateData = {
  282. isSidebarClosedAtDockMode,
  283. };
  284. // update isSidebarClosedAtDockMode in cache, not revalidate
  285. mutate({ ...data, ...updateData }, false);
  286. },
  287. };
  288. };
  289. export const useDrawerOpened = (isOpened?: boolean): SWRResponse<boolean, Error> => {
  290. return useStaticSWR('isDrawerOpened', isOpened, { fallbackData: false });
  291. };
  292. export const useSidebarResizeDisabled = (isDisabled?: boolean): SWRResponse<boolean, Error> => {
  293. return useStaticSWR('isSidebarResizeDisabled', isDisabled, { fallbackData: false });
  294. };
  295. export const useSelectedGrant = (initialData?: Nullable<IPageGrantData>): SWRResponse<Nullable<IPageGrantData>, Error> => {
  296. return useStaticSWR<Nullable<IPageGrantData>, Error>('selectedGrant', initialData);
  297. };
  298. export const useGlobalSearchFormRef = (initialData?: RefObject<IFocusable>): SWRResponse<RefObject<IFocusable>, Error> => {
  299. return useStaticSWR('globalSearchTypeahead', initialData);
  300. };
  301. type PageTreeDescCountMapUtils = {
  302. update(newData?: UpdateDescCountData): Promise<UpdateDescCountData | undefined>
  303. getDescCount(pageId?: string): number | null | undefined
  304. }
  305. export const usePageTreeDescCountMap = (initialData?: UpdateDescCountData): SWRResponse<UpdateDescCountData, Error> & PageTreeDescCountMapUtils => {
  306. const key = 'pageTreeDescCountMap';
  307. const swrResponse = useStaticSWR<UpdateDescCountData, Error>(key, initialData, { fallbackData: new Map() });
  308. return {
  309. ...swrResponse,
  310. getDescCount: (pageId?: string) => (pageId != null ? swrResponse.data?.get(pageId) : null),
  311. update: (newData: UpdateDescCountData) => swrResponse.mutate(new Map([...(swrResponse.data || new Map()), ...newData])),
  312. };
  313. };
  314. /** **********************************************************
  315. * SWR Hooks
  316. * Determined value by context
  317. *********************************************************** */
  318. export const useIsAbleToShowTrashPageManagementButtons = (): SWRResponse<boolean, Error> => {
  319. const { data: currentUser } = useCurrentUser();
  320. const { data: isTrashPage } = useIsTrashPage();
  321. return useStaticSWR('isAbleToShowTrashPageManagementButtons', isTrashPage && currentUser != null);
  322. };
  323. export const useIsAbleToShowPageManagement = (): SWRResponse<boolean, Error> => {
  324. const key = 'isAbleToShowPageManagement';
  325. const { data: currentPageId } = useCurrentPageId();
  326. const { data: isTrashPage } = useIsTrashPage();
  327. const { data: isSharedUser } = useIsSharedUser();
  328. const { data: isNotFound } = useIsNotFound();
  329. const pageId = currentPageId;
  330. const includesUndefined = [pageId, isTrashPage, isSharedUser, isNotFound].some(v => v === undefined);
  331. const isPageExist = (pageId != null) && !isNotFound;
  332. return useSWRImmutable(
  333. includesUndefined ? null : [key, pageId],
  334. () => isPageExist && !isTrashPage && !isSharedUser,
  335. );
  336. };
  337. export const useIsAbleToShowTagLabel = (): SWRResponse<boolean, Error> => {
  338. const key = 'isAbleToShowTagLabel';
  339. const { data: pageId } = useCurrentPageId();
  340. const { data: isUserPage } = useIsUserPage();
  341. const { data: currentPagePath } = useCurrentPagePath();
  342. const { data: isIdenticalPath } = useIsIdenticalPath();
  343. const { data: isNotFound } = useIsNotFound();
  344. const { data: editorMode } = useEditorMode();
  345. const { data: shareLinkId } = useShareLinkId();
  346. const includesUndefined = [isUserPage, currentPagePath, isIdenticalPath, isNotFound, editorMode].some(v => v === undefined);
  347. const isViewMode = editorMode === EditorMode.View;
  348. return useSWRImmutable(
  349. includesUndefined ? null : [key, editorMode, pageId],
  350. // "/trash" page does not exist on page collection and unable to add tags
  351. // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
  352. () => !isUserPage && !isTrashTopPage(currentPagePath!) && shareLinkId == null && !isIdenticalPath && !(isViewMode && isNotFound),
  353. );
  354. };
  355. export const useIsAbleToShowPageEditorModeManager = (): SWRResponse<boolean, Error> => {
  356. const key = 'isAbleToShowPageEditorModeManager';
  357. const { data: isNotCreatable } = useIsNotCreatable();
  358. const { data: isForbidden } = useIsForbidden();
  359. const { data: isTrashPage } = useIsTrashPage();
  360. const { data: isSharedUser } = useIsSharedUser();
  361. const includesUndefined = [isNotCreatable, isForbidden, isTrashPage, isSharedUser].some(v => v === undefined);
  362. return useSWRImmutable(
  363. includesUndefined ? null : key,
  364. () => !isNotCreatable && !isForbidden && !isTrashPage && !isSharedUser,
  365. );
  366. };
  367. export const useIsAbleToShowPageAuthors = (): SWRResponse<boolean, Error> => {
  368. const key = 'isAbleToShowPageAuthors';
  369. const { data: pageId } = useCurrentPageId();
  370. const { data: isUserPage } = useIsUserPage();
  371. const { data: isNotFound } = useIsNotFound();
  372. const includesUndefined = [pageId, isUserPage, isNotFound].some(v => v === undefined);
  373. const isPageExist = (pageId != null) && !isNotFound;
  374. return useSWRImmutable(
  375. includesUndefined ? null : [key, pageId],
  376. () => isPageExist && !isUserPage,
  377. );
  378. };