use-create-page.tsx 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. import { useCallback, useState } from 'react';
  2. import { useRouter } from 'next/router';
  3. import { useTranslation } from 'react-i18next';
  4. import {
  5. exist,
  6. getIsNonUserRelatedGroupsGranted,
  7. } from '~/client/services/page-operation';
  8. import { toastWarning } from '~/client/util/toastr';
  9. import type { IApiv3PageCreateParams } from '~/interfaces/apiv3';
  10. import { useCurrentPagePath, useSetIsUntitledPage } from '~/states/page';
  11. import { EditorMode, useEditorMode } from '~/states/ui/editor';
  12. import { useGrantedGroupsInheritanceSelectModalActions } from '~/states/ui/modal/granted-groups-inheritance-select';
  13. import { createPage } from './create-page';
  14. /**
  15. * Invoked when creation and transition has finished
  16. */
  17. type OnCreated = () => void;
  18. /**
  19. * Invoked when either creation or transition has aborted
  20. */
  21. type OnAborted = () => void;
  22. /**
  23. * Always invoked after processing is terminated
  24. */
  25. type OnTerminated = () => void;
  26. export type CreatePageOpts = {
  27. skipPageExistenceCheck?: boolean;
  28. skipTransition?: boolean;
  29. onCreationStart?: OnCreated;
  30. onCreated?: OnCreated;
  31. onAborted?: OnAborted;
  32. onTerminated?: OnTerminated;
  33. };
  34. type CreatePage = (
  35. params: IApiv3PageCreateParams,
  36. opts?: CreatePageOpts,
  37. ) => Promise<void>;
  38. type UseCreatePage = () => {
  39. isCreating: boolean;
  40. create: CreatePage;
  41. };
  42. export const useCreatePage: UseCreatePage = () => {
  43. const router = useRouter();
  44. const { t } = useTranslation();
  45. const currentPagePath = useCurrentPagePath();
  46. const { setEditorMode } = useEditorMode();
  47. const setIsUntitledPage = useSetIsUntitledPage();
  48. const {
  49. open: openGrantedGroupsInheritanceSelectModal,
  50. close: closeGrantedGroupsInheritanceSelectModal,
  51. } = useGrantedGroupsInheritanceSelectModalActions();
  52. const [isCreating, setCreating] = useState(false);
  53. const create: CreatePage = useCallback(
  54. async (params, opts = {}) => {
  55. const { onCreationStart, onCreated, onAborted, onTerminated } = opts;
  56. const skipPageExistenceCheck = opts.skipPageExistenceCheck ?? false;
  57. const skipTransition = opts.skipTransition ?? false;
  58. // check the page existence
  59. if (!skipPageExistenceCheck && params.path != null) {
  60. const pagePath = params.path;
  61. try {
  62. const { isExist } = await exist(pagePath);
  63. if (isExist) {
  64. if (!skipTransition) {
  65. // routing
  66. if (pagePath !== currentPagePath) {
  67. await router.push(`${pagePath}#edit`);
  68. }
  69. setEditorMode(EditorMode.Editor);
  70. } else {
  71. toastWarning(
  72. t('duplicated_page_alert.same_page_name_exists', {
  73. pageName: pagePath,
  74. }),
  75. );
  76. }
  77. onAborted?.();
  78. return;
  79. }
  80. } catch (err) {
  81. throw err;
  82. } finally {
  83. onTerminated?.();
  84. }
  85. }
  86. const _create = async (onlyInheritUserRelatedGrantedGroups?: boolean) => {
  87. try {
  88. setCreating(true);
  89. onCreationStart?.();
  90. params.onlyInheritUserRelatedGrantedGroups =
  91. onlyInheritUserRelatedGrantedGroups;
  92. const response = await createPage(params);
  93. closeGrantedGroupsInheritanceSelectModal();
  94. if (!skipTransition) {
  95. await router.push(`/${response.page._id}#edit`);
  96. setEditorMode(EditorMode.Editor);
  97. }
  98. if (params.path == null) {
  99. setIsUntitledPage(true);
  100. }
  101. onCreated?.();
  102. } catch (err) {
  103. throw err;
  104. } finally {
  105. onTerminated?.();
  106. setCreating(false);
  107. }
  108. };
  109. // If parent page is granted to non-user-related groups, let the user select whether or not to inherit them.
  110. if (params.parentPath != null) {
  111. const { isNonUserRelatedGroupsGranted } =
  112. await getIsNonUserRelatedGroupsGranted(params.parentPath);
  113. if (isNonUserRelatedGroupsGranted) {
  114. // create and transit request will be made from modal
  115. openGrantedGroupsInheritanceSelectModal(_create);
  116. return;
  117. }
  118. }
  119. await _create();
  120. },
  121. [
  122. currentPagePath,
  123. setEditorMode,
  124. router,
  125. t,
  126. closeGrantedGroupsInheritanceSelectModal,
  127. setIsUntitledPage,
  128. openGrantedGroupsInheritanceSelectModal,
  129. ],
  130. );
  131. return {
  132. isCreating,
  133. create,
  134. };
  135. };