context.tsx 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. import type { ColorScheme, IUserHasId } from '@growi/core';
  2. import useSWR, { SWRResponse } from 'swr';
  3. import useSWRImmutable from 'swr/immutable';
  4. import { SupportedActionType } from '~/interfaces/activity';
  5. import { EditorConfig } from '~/interfaces/editor-settings';
  6. import { RendererConfig } from '~/interfaces/services/renderer';
  7. import InterceptorManager from '~/services/interceptor-manager';
  8. import { TargetAndAncestors } from '../interfaces/page-listing-results';
  9. import { useContextSWR } from './use-context-swr';
  10. import { useStaticSWR } from './use-static-swr';
  11. type Nullable<T> = T | null;
  12. export const useInterceptorManager = (): SWRResponse<InterceptorManager, Error> => {
  13. return useContextSWR<InterceptorManager, Error>('interceptorManager', undefined, { fallbackData: new InterceptorManager() });
  14. };
  15. export const useCsrfToken = (initialData?: string): SWRResponse<string, Error> => {
  16. return useContextSWR<string, Error>('csrfToken', initialData);
  17. };
  18. export const useAppTitle = (initialData?: string): SWRResponse<string, Error> => {
  19. return useContextSWR('appTitle', initialData);
  20. };
  21. export const useSiteUrl = (initialData?: string): SWRResponse<string, Error> => {
  22. return useContextSWR<string, Error>('siteUrl', initialData);
  23. };
  24. export const useConfidential = (initialData?: string): SWRResponse<string, Error> => {
  25. return useContextSWR('confidential', initialData);
  26. };
  27. export const useCurrentUser = (initialData?: Nullable<IUserHasId>): SWRResponse<Nullable<IUserHasId>, Error> => {
  28. return useContextSWR('currentUser', initialData);
  29. };
  30. export const useCurrentPathname = (initialData?: string): SWRResponse<string, Error> => {
  31. return useContextSWR('currentPathname', initialData);
  32. };
  33. export const useIsIdenticalPath = (initialData?: boolean): SWRResponse<boolean, Error> => {
  34. return useContextSWR<boolean, Error>('isIdenticalPath', initialData, { fallbackData: false });
  35. };
  36. export const useIsForbidden = (initialData?: boolean): SWRResponse<boolean, Error> => {
  37. return useContextSWR<boolean, Error>('isForbidden', initialData, { fallbackData: false });
  38. };
  39. export const useIsNotCreatable = (initialData?: boolean): SWRResponse<boolean, Error> => {
  40. return useContextSWR<boolean, Error>('isNotCreatable', initialData, { fallbackData: false });
  41. };
  42. export const useIsSharedUser = (initialData?: boolean): SWRResponse<boolean, Error> => {
  43. return useContextSWR<boolean, Error>('isSharedUser', initialData);
  44. };
  45. export const useShareLinkId = (initialData?: string): SWRResponse<string, Error> => {
  46. return useContextSWR('shareLinkId', initialData);
  47. };
  48. export const useDisableLinkSharing = (initialData?: Nullable<boolean>): SWRResponse<Nullable<boolean>, Error> => {
  49. return useContextSWR<Nullable<boolean>, Error>('disableLinkSharing', initialData);
  50. };
  51. export const useRegistrationWhitelist = (initialData?: Nullable<string[]>): SWRResponse<Nullable<string[]>, Error> => {
  52. return useContextSWR<Nullable<string[]>, Error>('registrationWhitelist', initialData);
  53. };
  54. export const useHackmdUri = (initialData?: Nullable<string>): SWRResponse<Nullable<string>, Error> => {
  55. return useContextSWR<Nullable<string>, Error>('hackmdUri', initialData);
  56. };
  57. export const useIsSearchPage = (initialData?: Nullable<boolean>) : SWRResponse<Nullable<boolean>, Error> => {
  58. return useContextSWR<Nullable<any>, Error>('isSearchPage', initialData);
  59. };
  60. export const useTargetAndAncestors = (initialData?: TargetAndAncestors): SWRResponse<TargetAndAncestors, Error> => {
  61. return useContextSWR<TargetAndAncestors, Error>('targetAndAncestors', initialData);
  62. };
  63. export const useIsAclEnabled = (initialData?: boolean) : SWRResponse<boolean, Error> => {
  64. return useContextSWR<boolean, Error>('isAclEnabled', initialData);
  65. };
  66. export const useIsSearchServiceConfigured = (initialData?: boolean) : SWRResponse<boolean, Error> => {
  67. return useContextSWR<boolean, Error>('isSearchServiceConfigured', initialData);
  68. };
  69. export const useIsSearchServiceReachable = (initialData?: boolean) : SWRResponse<boolean, Error> => {
  70. return useContextSWR<boolean, Error>('isSearchServiceReachable', initialData);
  71. };
  72. export const useIsMailerSetup = (initialData?: boolean): SWRResponse<boolean, any> => {
  73. return useContextSWR('isMailerSetup', initialData);
  74. };
  75. export const useIsSearchScopeChildrenAsDefault = (initialData?: boolean) : SWRResponse<boolean, Error> => {
  76. return useContextSWR<boolean, Error>('isSearchScopeChildrenAsDefault', initialData, { fallbackData: false });
  77. };
  78. export const useIsEnabledMarp = (initialData?: boolean) : SWRResponse<boolean, Error> => {
  79. return useContextSWR<boolean, Error>('isEnabledMarp', initialData, { fallbackData: false });
  80. };
  81. export const useIsSlackConfigured = (initialData?: boolean) : SWRResponse<boolean, Error> => {
  82. return useContextSWR<boolean, Error>('isSlackConfigured', initialData);
  83. };
  84. export const useIsEnabledAttachTitleHeader = (initialData?: boolean) : SWRResponse<boolean, Error> => {
  85. return useContextSWR<boolean, Error>('isEnabledAttachTitleHeader', initialData);
  86. };
  87. export const useIsIndentSizeForced = (initialData?: boolean) : SWRResponse<boolean, Error> => {
  88. return useContextSWR<boolean, Error>('isIndentSizeForced', initialData, { fallbackData: false });
  89. };
  90. export const useDefaultIndentSize = (initialData?: number) : SWRResponse<number, Error> => {
  91. return useContextSWR<number, Error>('defaultIndentSize', initialData, { fallbackData: 4 });
  92. };
  93. export const useAuditLogEnabled = (initialData?: boolean): SWRResponse<boolean, Error> => {
  94. return useContextSWR<boolean, Error>('auditLogEnabled', initialData, { fallbackData: false });
  95. };
  96. // TODO: initialize in [[..path]].page.tsx?
  97. export const useActivityExpirationSeconds = (initialData?: number) : SWRResponse<number, Error> => {
  98. return useContextSWR<number, Error>('activityExpirationSeconds', initialData);
  99. };
  100. export const useAuditLogAvailableActions = (initialData?: Array<SupportedActionType>) : SWRResponse<Array<SupportedActionType>, Error> => {
  101. return useContextSWR<Array<SupportedActionType>, Error>('auditLogAvailableActions', initialData);
  102. };
  103. export const useGrowiVersion = (initialData?: string): SWRResponse<string, any> => {
  104. return useContextSWR('growiVersion', initialData);
  105. };
  106. export const useIsEnabledStaleNotification = (initialData?: boolean): SWRResponse<boolean, any> => {
  107. return useContextSWR('isEnabledStaleNotification', initialData);
  108. };
  109. export const useEditorConfig = (initialData?: EditorConfig): SWRResponse<EditorConfig, Error> => {
  110. return useContextSWR<EditorConfig, Error>('editorConfig', initialData);
  111. };
  112. export const useRendererConfig = (initialData?: RendererConfig): SWRResponse<RendererConfig, any> => {
  113. return useContextSWR('growiRendererConfig', initialData);
  114. };
  115. export const useIsAllReplyShown = (initialData?: boolean): SWRResponse<boolean, Error> => {
  116. return useContextSWR('isAllReplyShown', initialData);
  117. };
  118. export const useIsBlinkedHeaderAtBoot = (initialData?: boolean): SWRResponse<boolean, Error> => {
  119. return useContextSWR('isBlinkedAtBoot', initialData, { fallbackData: false });
  120. };
  121. export const useIsUploadableImage = (initialData?: boolean): SWRResponse<boolean, Error> => {
  122. return useContextSWR('isUploadableImage', initialData);
  123. };
  124. export const useIsUploadableFile = (initialData?: boolean): SWRResponse<boolean, Error> => {
  125. return useContextSWR('isUploadableFile', initialData);
  126. };
  127. export const useShowPageLimitationL = (initialData?: number): SWRResponse<number, Error> => {
  128. return useContextSWR('showPageLimitationL', initialData);
  129. };
  130. export const useShowPageLimitationXL = (initialData?: number): SWRResponse<number, Error> => {
  131. return useContextSWR('showPageLimitationXL', initialData);
  132. };
  133. export const useCustomizeTitle = (initialData?: string): SWRResponse<string, Error> => {
  134. return useContextSWR('CustomizeTitle', initialData);
  135. };
  136. export const useIsDefaultLogo = (initialData?: boolean): SWRResponse<boolean, Error> => {
  137. return useContextSWR('isDefaultLogo', initialData);
  138. };
  139. export const useIsCustomizedLogoUploaded = (initialData?: boolean): SWRResponse<boolean, Error> => {
  140. return useStaticSWR('isCustomizedLogoUploaded', initialData);
  141. };
  142. export const useForcedColorScheme = (initialData?: ColorScheme): SWRResponse<ColorScheme, Error> => {
  143. return useContextSWR('forcedColorScheme', initialData);
  144. };
  145. export const useGrowiCloudUri = (initialData?: string): SWRResponse<string, Error> => {
  146. return useStaticSWR('growiCloudUri', initialData);
  147. };
  148. export const useGrowiAppIdForGrowiCloud = (initialData?: number): SWRResponse<number, Error> => {
  149. return useStaticSWR('growiAppIdForGrowiCloud', initialData);
  150. };
  151. export const useIsContainerFluid = (initialData?: boolean): SWRResponse<boolean, Error> => {
  152. return useStaticSWR('isContainerFluid', initialData);
  153. };
  154. /** **********************************************************
  155. * Computed contexts
  156. *********************************************************** */
  157. export const useIsGuestUser = (): SWRResponse<boolean, Error> => {
  158. const { data: currentUser, isLoading } = useCurrentUser();
  159. return useSWRImmutable(
  160. isLoading ? null : ['isGuestUser', currentUser?._id],
  161. ([, currentUserId]) => currentUserId == null,
  162. { fallbackData: currentUser?._id == null },
  163. );
  164. };
  165. export const useIsReadOnlyUser = (): SWRResponse<boolean, Error> => {
  166. const { data: currentUser, isLoading: isCurrentUserLoading } = useCurrentUser();
  167. const { data: isGuestUser, isLoading: isGuestUserLoding } = useIsGuestUser();
  168. const isLoading = isCurrentUserLoading || isGuestUserLoding;
  169. const isReadOnlyUser = !isGuestUser && !!currentUser?.readOnly;
  170. return useSWRImmutable(
  171. isLoading ? null : ['isReadOnlyUser', isReadOnlyUser, currentUser?._id],
  172. () => isReadOnlyUser,
  173. { fallbackData: isReadOnlyUser },
  174. );
  175. };
  176. export const useIsAdmin = (): SWRResponse<boolean, Error> => {
  177. const { data: currentUser, isLoading } = useCurrentUser();
  178. return useSWR(
  179. isLoading ? null : ['isAdminUser', currentUser?._id, currentUser?.admin],
  180. ([, , isAdmin]) => isAdmin ?? false,
  181. {
  182. fallbackData: currentUser?.admin ?? false,
  183. keepPreviousData: true,
  184. // disable all revalidation but revalidateIfStale
  185. revalidateOnMount: false,
  186. revalidateOnFocus: false,
  187. revalidateOnReconnect: false,
  188. },
  189. );
  190. };
  191. export const useIsEditable = (): SWRResponse<boolean, Error> => {
  192. const { data: isGuestUser } = useIsGuestUser();
  193. const { data: isReadOnlyUser } = useIsReadOnlyUser();
  194. const { data: isForbidden } = useIsForbidden();
  195. const { data: isNotCreatable } = useIsNotCreatable();
  196. const { data: isIdenticalPath } = useIsIdenticalPath();
  197. return useSWRImmutable(
  198. ['isEditable', isGuestUser, isReadOnlyUser, isForbidden, isNotCreatable, isIdenticalPath],
  199. ([, isGuestUser, isReadOnlyUser, isForbidden, isNotCreatable, isIdenticalPath]) => {
  200. return (!isForbidden && !isIdenticalPath && !isNotCreatable && !isGuestUser && !isReadOnlyUser);
  201. },
  202. );
  203. };