PageCreateButton.tsx 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. import React, { useCallback, useState } from 'react';
  2. import { pagePathUtils } from '@growi/core/dist/utils';
  3. import { format } from 'date-fns';
  4. import { useRouter } from 'next/router';
  5. import { createPage, exist } from '~/client/services/page-operation';
  6. import { toastError } from '~/client/util/toastr';
  7. import { useCurrentUser } from '~/stores/context';
  8. import { useSWRxCurrentPage } from '~/stores/page';
  9. import loggerFactory from '~/utils/logger';
  10. import { DropendMenu } from './DropendMenu';
  11. import { CreateButton } from './CreateButton';
  12. import { DropendToggle } from './DropendToggle';
  13. const logger = loggerFactory('growi:cli:PageCreateButton');
  14. export const PageCreateButton = React.memo((): JSX.Element => {
  15. const router = useRouter();
  16. const { data: currentPage, isLoading } = useSWRxCurrentPage();
  17. const { data: currentUser } = useCurrentUser();
  18. const [isHovered, setIsHovered] = useState(false);
  19. const [isCreating, setIsCreating] = useState(false);
  20. const now = format(new Date(), 'yyyy/MM/dd');
  21. const userHomepagePath = pagePathUtils.userHomepagePath(currentUser);
  22. const todaysPath = `${userHomepagePath}/memo/${now}`;
  23. const onMouseEnterHandler = () => {
  24. setIsHovered(true);
  25. };
  26. const onMouseLeaveHandler = () => {
  27. setIsHovered(false);
  28. };
  29. const onClickCreateNewPageButtonHandler = useCallback(async() => {
  30. if (isLoading) return;
  31. try {
  32. setIsCreating(true);
  33. const parentPath = currentPage == null
  34. ? '/'
  35. : currentPage.path;
  36. const params = {
  37. isSlackEnabled: false,
  38. slackChannels: '',
  39. grant: currentPage?.grant || 1,
  40. pageTags: [],
  41. grantUserGroupId: currentPage?.grantedGroup?._id,
  42. shouldGeneratePath: true,
  43. };
  44. const response = await createPage(parentPath, '', params);
  45. router.push(`${response.page.id}#edit`);
  46. }
  47. catch (err) {
  48. logger.warn(err);
  49. toastError(err);
  50. }
  51. finally {
  52. setIsCreating(false);
  53. }
  54. }, [currentPage, isLoading, router]);
  55. const onClickCreateTodaysButtonHandler = useCallback(async() => {
  56. if (currentUser == null) {
  57. return;
  58. }
  59. try {
  60. setIsCreating(true);
  61. // TODO: get grant, grantUserGroupId data from parent page
  62. // https://redmine.weseek.co.jp/issues/133892
  63. const params = {
  64. isSlackEnabled: false,
  65. slackChannels: '',
  66. grant: 1,
  67. pageTags: [],
  68. };
  69. const res = await exist(JSON.stringify([todaysPath]));
  70. if (!res.pages[todaysPath]) {
  71. await createPage(todaysPath, '', params);
  72. }
  73. router.push(`${todaysPath}#edit`);
  74. }
  75. catch (err) {
  76. logger.warn(err);
  77. toastError(err);
  78. }
  79. finally {
  80. setIsCreating(false);
  81. }
  82. }, [currentUser, router, todaysPath]);
  83. const onClickTemplateForChildrenButtonHandler = useCallback(async() => {
  84. if (isLoading) return;
  85. try {
  86. setIsCreating(true);
  87. const path = currentPage == null || currentPage.path === '/'
  88. ? '/_template'
  89. : `${currentPage.path}/_template`;
  90. const params = {
  91. isSlackEnabled: false,
  92. slackChannels: '',
  93. grant: currentPage?.grant || 1,
  94. pageTags: [],
  95. grantUserGroupId: currentPage?.grantedGroup?._id,
  96. };
  97. const res = await exist(JSON.stringify([path]));
  98. if (!res.pages[path]) {
  99. await createPage(path, '', params);
  100. }
  101. router.push(`${path}#edit`);
  102. }
  103. catch (err) {
  104. logger.warn(err);
  105. toastError(err);
  106. }
  107. finally {
  108. setIsCreating(false);
  109. }
  110. }, [currentPage, isLoading, router]);
  111. const onClickTemplateForDescendantsButtonHandler = useCallback(async() => {
  112. if (isLoading) return;
  113. try {
  114. setIsCreating(true);
  115. const path = currentPage == null || currentPage.path === '/'
  116. ? '/__template'
  117. : `${currentPage.path}/__template`;
  118. const params = {
  119. isSlackEnabled: false,
  120. slackChannels: '',
  121. grant: currentPage?.grant || 1,
  122. pageTags: [],
  123. grantUserGroupId: currentPage?.grantedGroup?._id,
  124. };
  125. const res = await exist(JSON.stringify([path]));
  126. if (!res.pages[path]) {
  127. await createPage(path, '', params);
  128. }
  129. router.push(`${path}#edit`);
  130. }
  131. catch (err) {
  132. logger.warn(err);
  133. toastError(err);
  134. }
  135. finally {
  136. setIsCreating(false);
  137. }
  138. }, [currentPage, isLoading, router]);
  139. // TODO: update button design
  140. // https://redmine.weseek.co.jp/issues/132683
  141. return (
  142. <div
  143. className="d-flex flex-row"
  144. onMouseEnter={onMouseEnterHandler}
  145. onMouseLeave={onMouseLeaveHandler}
  146. >
  147. <div className="btn-group flex-grow-1">
  148. <CreateButton
  149. className="z-2"
  150. onClick={onClickCreateNewPageButtonHandler}
  151. disabled={isCreating}
  152. />
  153. </div>
  154. { isHovered && (
  155. <div className="btn-group dropend position-absolute">
  156. <DropendToggle
  157. className="dropdown-toggle dropdown-toggle-split"
  158. data-bs-toggle="dropdown"
  159. aria-expanded="false"
  160. />
  161. <DropendMenu
  162. todaysPath={todaysPath}
  163. onClickCreateNewPageButtonHandler={onClickCreateNewPageButtonHandler}
  164. onClickCreateTodaysButtonHandler={onClickCreateTodaysButtonHandler}
  165. onClickTemplateForChildrenButtonHandler={onClickTemplateForChildrenButtonHandler}
  166. onClickTemplateForDescendantsButtonHandler={onClickTemplateForDescendantsButtonHandler}
  167. />
  168. </div>
  169. )}
  170. </div>
  171. );
  172. });