PageEditorModeManager.tsx 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. import React, { type ReactNode, useCallback } from 'react';
  2. import { useTranslation } from 'next-i18next';
  3. import { useCreatePageAndTransit } from '~/client/services/create-page';
  4. import { toastError } from '~/client/util/toastr';
  5. import { useIsNotFound } from '~/stores/page';
  6. import { EditorMode, useEditorMode, useIsDeviceLargerThanMd } from '~/stores/ui';
  7. import styles from './PageEditorModeManager.module.scss';
  8. type PageEditorModeButtonProps = {
  9. currentEditorMode: EditorMode,
  10. editorMode: EditorMode,
  11. children?: ReactNode,
  12. isBtnDisabled?: boolean,
  13. onClick?: () => void,
  14. }
  15. const PageEditorModeButton = React.memo((props: PageEditorModeButtonProps) => {
  16. const {
  17. currentEditorMode, isBtnDisabled, editorMode, children, onClick,
  18. } = props;
  19. const classNames = ['btn py-1 px-2 d-flex align-items-center justify-content-center'];
  20. if (currentEditorMode === editorMode) {
  21. classNames.push('active');
  22. }
  23. if (isBtnDisabled) {
  24. classNames.push('disabled');
  25. }
  26. return (
  27. <button
  28. type="button"
  29. className={classNames.join(' ')}
  30. onClick={onClick}
  31. data-testid={`${editorMode}-button`}
  32. >
  33. {children}
  34. </button>
  35. );
  36. });
  37. type Props = {
  38. editorMode: EditorMode | undefined,
  39. isBtnDisabled: boolean,
  40. path?: string,
  41. }
  42. export const PageEditorModeManager = (props: Props): JSX.Element => {
  43. const {
  44. editorMode = EditorMode.View,
  45. isBtnDisabled,
  46. path,
  47. } = props;
  48. const { t } = useTranslation('commons');
  49. const { data: isNotFound } = useIsNotFound();
  50. const { mutate: mutateEditorMode } = useEditorMode();
  51. const { data: isDeviceLargerThanMd } = useIsDeviceLargerThanMd();
  52. const { isCreating, createAndTransit } = useCreatePageAndTransit();
  53. const editButtonClickedHandler = useCallback(async() => {
  54. if (isNotFound == null || isNotFound === false) {
  55. mutateEditorMode(EditorMode.Editor);
  56. return;
  57. }
  58. try {
  59. await createAndTransit(
  60. { path },
  61. { shouldCheckPageExists: true },
  62. );
  63. }
  64. catch (err) {
  65. toastError(t('toaster.create_failed', { target: path }));
  66. }
  67. }, [createAndTransit, isNotFound, mutateEditorMode, path, t]);
  68. const _isBtnDisabled = isCreating || isBtnDisabled;
  69. return (
  70. <>
  71. <div
  72. className={`btn-group grw-page-editor-mode-manager ${styles['grw-page-editor-mode-manager']}`}
  73. role="group"
  74. aria-label="page-editor-mode-manager"
  75. id="grw-page-editor-mode-manager"
  76. >
  77. {(isDeviceLargerThanMd || editorMode !== EditorMode.View) && (
  78. <PageEditorModeButton
  79. currentEditorMode={editorMode}
  80. editorMode={EditorMode.View}
  81. isBtnDisabled={_isBtnDisabled}
  82. onClick={() => mutateEditorMode(EditorMode.View)}
  83. >
  84. <span className="material-symbols-outlined fs-4">play_arrow</span>{t('View')}
  85. </PageEditorModeButton>
  86. )}
  87. {(isDeviceLargerThanMd || editorMode === EditorMode.View) && (
  88. <PageEditorModeButton
  89. currentEditorMode={editorMode}
  90. editorMode={EditorMode.Editor}
  91. isBtnDisabled={_isBtnDisabled}
  92. onClick={editButtonClickedHandler}
  93. >
  94. <span className="material-symbols-outlined me-1 fs-5">edit_square</span>{t('Edit')}
  95. </PageEditorModeButton>
  96. )}
  97. </div>
  98. </>
  99. );
  100. };