decap-cms-core 3.10.1 → 3.12.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/decap-cms-core.js +18 -18
- package/dist/decap-cms-core.js.map +1 -1
- package/dist/esm/actions/deploys.js +3 -2
- package/dist/esm/backend.js +13 -5
- package/dist/esm/bootstrap.js +2 -2
- package/dist/esm/components/App/StatusBar.js +1 -1
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +1 -0
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +6 -4
- package/dist/esm/components/Editor/EditorToolbar.js +52 -34
- package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +7 -8
- package/dist/esm/components/UI/ErrorBoundary.js +10 -10
- package/dist/esm/constants/configSchema.js +9 -0
- package/dist/esm/formats/yaml.js +11 -2
- package/dist/esm/lib/formatters.js +14 -3
- package/dist/esm/lib/i18n.js +8 -3
- package/dist/esm/reducers/deploys.js +8 -3
- package/index.d.ts +6 -0
- package/package.json +2 -3
- package/src/actions/deploys.ts +4 -3
- package/src/backend.ts +12 -1
- package/src/components/Editor/EditorPreviewPane/EditorPreviewContent.js +1 -0
- package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +3 -0
- package/src/components/Editor/EditorToolbar.js +40 -3
- package/src/components/Editor/__tests__/EditorToolbar.spec.js +46 -0
- package/src/components/MediaLibrary/MediaLibraryButtons.js +2 -3
- package/src/components/UI/ErrorBoundary.js +5 -5
- package/src/constants/configSchema.js +6 -0
- package/src/formats/__tests__/frontmatter.spec.js +21 -0
- package/src/formats/__tests__/yaml.spec.js +27 -0
- package/src/formats/yaml.ts +16 -1
- package/src/lib/__tests__/formatters.spec.js +21 -0
- package/src/lib/formatters.ts +15 -3
- package/src/lib/i18n.ts +14 -6
- package/src/reducers/__tests__/deploys.spec.ts +111 -0
- package/src/reducers/deploys.ts +5 -3
- package/src/types/redux.ts +6 -0
|
@@ -16,7 +16,7 @@ const styles = {
|
|
|
16
16
|
styles: "overflow:hidden;white-space:nowrap;text-overflow:ellipsis;label:noOverflow;"
|
|
17
17
|
} : {
|
|
18
18
|
name: "1ctb18k-noOverflow",
|
|
19
|
-
styles: "overflow:hidden;white-space:nowrap;text-overflow:ellipsis;label:noOverflow;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAuBiB","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
19
|
+
styles: "overflow:hidden;white-space:nowrap;text-overflow:ellipsis;label:noOverflow;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAuBiB","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
20
20
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
21
21
|
},
|
|
22
22
|
buttonMargin: process.env.NODE_ENV === "production" ? {
|
|
@@ -24,11 +24,11 @@ const styles = {
|
|
|
24
24
|
styles: "margin:0 10px;label:buttonMargin;"
|
|
25
25
|
} : {
|
|
26
26
|
name: "bx8mhy-buttonMargin",
|
|
27
|
-
styles: "margin:0 10px;label:buttonMargin;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA4BmB","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
27
|
+
styles: "margin:0 10px;label:buttonMargin;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA4BmB","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
28
28
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
29
29
|
},
|
|
30
|
-
toolbarSection: /*#__PURE__*/css("height:100%;display:flex;align-items:center;border:0 solid ", colors.textFieldBorder, ";;label:toolbarSection;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA+BqB","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */")),
|
|
31
|
-
publishedButton: /*#__PURE__*/css("background-color:", colorsRaw.tealLight, ";color:", colorsRaw.tealDark, ";;label:publishedButton;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAqCsB","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"))
|
|
30
|
+
toolbarSection: /*#__PURE__*/css("height:100%;display:flex;align-items:center;border:0 solid ", colors.textFieldBorder, ";;label:toolbarSection;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA+BqB","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */")),
|
|
31
|
+
publishedButton: /*#__PURE__*/css("background-color:", colorsRaw.tealLight, ";color:", colorsRaw.tealDark, ";;label:publishedButton;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAqCsB","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"))
|
|
32
32
|
};
|
|
33
33
|
const TooltipText = /*#__PURE__*/_styled("div", {
|
|
34
34
|
target: "e1d2l9mo26",
|
|
@@ -38,13 +38,13 @@ const TooltipText = /*#__PURE__*/_styled("div", {
|
|
|
38
38
|
styles: "visibility:hidden;width:321px;background-color:#555;color:#fff;text-align:unset;border-radius:6px;padding:5px;position:absolute;z-index:1;top:145%;left:50%;margin-left:-320px;opacity:0;transition:opacity 0.3s"
|
|
39
39
|
} : {
|
|
40
40
|
name: "1ykbu60",
|
|
41
|
-
styles: "visibility:hidden;width:321px;background-color:#555;color:#fff;text-align:unset;border-radius:6px;padding:5px;position:absolute;z-index:1;top:145%;left:50%;margin-left:-320px;opacity:0;transition:opacity 0.3s/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA2C8B","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
41
|
+
styles: "visibility:hidden;width:321px;background-color:#555;color:#fff;text-align:unset;border-radius:6px;padding:5px;position:absolute;z-index:1;top:145%;left:50%;margin-left:-320px;opacity:0;transition:opacity 0.3s/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA2C8B","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
42
42
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
43
43
|
});
|
|
44
44
|
const Tooltip = /*#__PURE__*/_styled("div", {
|
|
45
45
|
target: "e1d2l9mo25",
|
|
46
46
|
label: "Tooltip"
|
|
47
|
-
})("position:relative;display:inline-block;&:hover+", TooltipText, "{visibility:visible;opacity:0.9;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAgE0B","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
47
|
+
})("position:relative;display:inline-block;&:hover+", TooltipText, "{visibility:visible;opacity:0.9;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAgE0B","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
48
48
|
const TooltipContainer = /*#__PURE__*/_styled("div", {
|
|
49
49
|
target: "e1d2l9mo24",
|
|
50
50
|
label: "TooltipContainer"
|
|
@@ -53,21 +53,21 @@ const TooltipContainer = /*#__PURE__*/_styled("div", {
|
|
|
53
53
|
styles: "position:relative"
|
|
54
54
|
} : {
|
|
55
55
|
name: "bjn8wh",
|
|
56
|
-
styles: "position:relative/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAyEmC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
56
|
+
styles: "position:relative/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAyEmC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
57
57
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
58
58
|
});
|
|
59
59
|
const DropdownButton = /*#__PURE__*/_styled(StyledDropdownButton, {
|
|
60
60
|
target: "e1d2l9mo23",
|
|
61
61
|
label: "DropdownButton"
|
|
62
|
-
})(styles.noOverflow, "@media (max-width: 1200px){padding-left:10px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA6EmD","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
62
|
+
})(styles.noOverflow, "@media (max-width: 1200px){padding-left:10px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA6EmD","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
63
63
|
const ToolbarContainer = /*#__PURE__*/_styled("div", {
|
|
64
64
|
target: "e1d2l9mo22",
|
|
65
65
|
label: "ToolbarContainer"
|
|
66
|
-
})("box-shadow:0 2px 6px 0 rgba(68, 74, 87, 0.05),0 1px 3px 0 rgba(68, 74, 87, 0.1),0 2px 54px rgba(0, 0, 0, 0.1);position:absolute;top:0;left:0;width:100%;min-width:800px;z-index:", zIndex.zIndex300, ";background-color:#fff;height:66px;display:flex;justify-content:space-between;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAoFmC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
66
|
+
})("box-shadow:0 2px 6px 0 rgba(68, 74, 87, 0.05),0 1px 3px 0 rgba(68, 74, 87, 0.1),0 2px 54px rgba(0, 0, 0, 0.1);position:absolute;top:0;left:0;width:100%;min-width:800px;z-index:", zIndex.zIndex300, ";background-color:#fff;height:66px;display:flex;justify-content:space-between;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAoFmC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
67
67
|
const ToolbarSectionMain = /*#__PURE__*/_styled("div", {
|
|
68
68
|
target: "e1d2l9mo21",
|
|
69
69
|
label: "ToolbarSectionMain"
|
|
70
|
-
})(styles.toolbarSection, ";flex:10;display:flex;justify-content:space-between;padding:0 10px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAmGqC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
70
|
+
})(styles.toolbarSection, ";flex:10;display:flex;justify-content:space-between;padding:0 10px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAmGqC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
71
71
|
const ToolbarSubSectionFirst = /*#__PURE__*/_styled("div", {
|
|
72
72
|
target: "e1d2l9mo20",
|
|
73
73
|
label: "ToolbarSubSectionFirst"
|
|
@@ -76,7 +76,7 @@ const ToolbarSubSectionFirst = /*#__PURE__*/_styled("div", {
|
|
|
76
76
|
styles: "display:flex;align-items:center"
|
|
77
77
|
} : {
|
|
78
78
|
name: "s5xdrg",
|
|
79
|
-
styles: "display:flex;align-items:center/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA2GyC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
79
|
+
styles: "display:flex;align-items:center/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA2GyC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
80
80
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
81
81
|
});
|
|
82
82
|
const ToolbarSubSectionLast = /*#__PURE__*/_styled(ToolbarSubSectionFirst, {
|
|
@@ -87,29 +87,29 @@ const ToolbarSubSectionLast = /*#__PURE__*/_styled(ToolbarSubSectionFirst, {
|
|
|
87
87
|
styles: "justify-content:flex-end"
|
|
88
88
|
} : {
|
|
89
89
|
name: "1f60if8",
|
|
90
|
-
styles: "justify-content:flex-end/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAgH4D","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
90
|
+
styles: "justify-content:flex-end/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAgH4D","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
91
91
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
92
92
|
});
|
|
93
93
|
const ToolbarSectionBackLink = /*#__PURE__*/_styled(Link, {
|
|
94
94
|
target: "e1d2l9mo18",
|
|
95
95
|
label: "ToolbarSectionBackLink"
|
|
96
|
-
})(styles.toolbarSection, ";border-right-width:1px;font-weight:normal;padding:0 20px;&:hover,&:focus{background-color:#f1f2f4;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAoH2C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
96
|
+
})(styles.toolbarSection, ";border-right-width:1px;font-weight:normal;padding:0 20px;&:hover,&:focus{background-color:#f1f2f4;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAoH2C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
97
97
|
const ToolbarSectionMeta = /*#__PURE__*/_styled("div", {
|
|
98
98
|
target: "e1d2l9mo17",
|
|
99
99
|
label: "ToolbarSectionMeta"
|
|
100
|
-
})(styles.toolbarSection, ";border-left-width:1px;padding:0 7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAgIqC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
100
|
+
})(styles.toolbarSection, ";border-left-width:1px;padding:0 7px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAgIqC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
101
101
|
const ToolbarDropdown = /*#__PURE__*/_styled(Dropdown, {
|
|
102
102
|
target: "e1d2l9mo16",
|
|
103
103
|
label: "ToolbarDropdown"
|
|
104
|
-
})(styles.buttonMargin, ";", Icon, "{color:", colorsRaw.teal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAsIwC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
104
|
+
})(styles.buttonMargin, ";", Icon, "{color:", colorsRaw.teal, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAsIwC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
105
105
|
const BackArrow = /*#__PURE__*/_styled("div", {
|
|
106
106
|
target: "e1d2l9mo15",
|
|
107
107
|
label: "BackArrow"
|
|
108
|
-
})("color:", colors.textLead, ";font-size:21px;font-weight:600;margin-right:16px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA8I4B","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
108
|
+
})("color:", colors.textLead, ";font-size:21px;font-weight:600;margin-right:16px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA8I4B","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
109
109
|
const BackCollection = /*#__PURE__*/_styled("div", {
|
|
110
110
|
target: "e1d2l9mo14",
|
|
111
111
|
label: "BackCollection"
|
|
112
|
-
})("color:", colors.textLead, ";font-size:14px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAqJiC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
112
|
+
})("color:", colors.textLead, ";font-size:14px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAqJiC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
113
113
|
const BackStatus = /*#__PURE__*/_styled("div", {
|
|
114
114
|
target: "e1d2l9mo13",
|
|
115
115
|
label: "BackStatus"
|
|
@@ -118,53 +118,53 @@ const BackStatus = /*#__PURE__*/_styled("div", {
|
|
|
118
118
|
styles: "margin-top:6px"
|
|
119
119
|
} : {
|
|
120
120
|
name: "j9a02z",
|
|
121
|
-
styles: "margin-top:6px/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA0J6B","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
121
|
+
styles: "margin-top:6px/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA0J6B","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
122
122
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
123
123
|
});
|
|
124
124
|
const BackStatusUnchanged = /*#__PURE__*/_styled(BackStatus, {
|
|
125
125
|
target: "e1d2l9mo12",
|
|
126
126
|
label: "BackStatusUnchanged"
|
|
127
|
-
})(components.textBadgeSuccess, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA8J8C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
127
|
+
})(components.textBadgeSuccess, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA8J8C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
128
128
|
const BackStatusChanged = /*#__PURE__*/_styled(BackStatus, {
|
|
129
129
|
target: "e1d2l9mo11",
|
|
130
130
|
label: "BackStatusChanged"
|
|
131
|
-
})(components.textBadgeDanger, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAkK4C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
131
|
+
})(components.textBadgeDanger, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAkK4C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
132
132
|
const ToolbarButton = /*#__PURE__*/_styled("button", {
|
|
133
133
|
target: "e1d2l9mo10",
|
|
134
134
|
label: "ToolbarButton"
|
|
135
|
-
})(buttons.button, ";", buttons.default, ";", styles.buttonMargin, ";", styles.noOverflow, ";display:block;@media (max-width: 1200px){padding:0 10px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAsKmC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
135
|
+
})(buttons.button, ";", buttons.default, ";", styles.buttonMargin, ";", styles.noOverflow, ";display:block;@media (max-width: 1200px){padding:0 10px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAsKmC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
136
136
|
const DeleteButton = /*#__PURE__*/_styled(ToolbarButton, {
|
|
137
137
|
target: "e1d2l9mo9",
|
|
138
138
|
label: "DeleteButton"
|
|
139
|
-
})(buttons.lightRed, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAkL0C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
139
|
+
})(buttons.lightRed, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAkL0C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
140
140
|
const SaveButton = /*#__PURE__*/_styled(ToolbarButton, {
|
|
141
141
|
target: "e1d2l9mo8",
|
|
142
142
|
label: "SaveButton"
|
|
143
|
-
})(buttons.lightBlue, ";&[disabled]{", buttons.disabled, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAsLwC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
143
|
+
})(buttons.lightBlue, ";&[disabled]{", buttons.disabled, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAsLwC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
144
144
|
const PublishedToolbarButton = /*#__PURE__*/_styled(DropdownButton, {
|
|
145
145
|
target: "e1d2l9mo7",
|
|
146
146
|
label: "PublishedToolbarButton"
|
|
147
|
-
})(styles.publishedButton, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA6LqD","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
147
|
+
})(styles.publishedButton, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA6LqD","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
148
148
|
const PublishedButton = /*#__PURE__*/_styled(ToolbarButton, {
|
|
149
149
|
target: "e1d2l9mo6",
|
|
150
150
|
label: "PublishedButton"
|
|
151
|
-
})(styles.publishedButton, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAiM6C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
151
|
+
})(styles.publishedButton, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAiM6C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
152
152
|
const PublishButton = /*#__PURE__*/_styled(DropdownButton, {
|
|
153
153
|
target: "e1d2l9mo5",
|
|
154
154
|
label: "PublishButton"
|
|
155
|
-
})("background-color:", colorsRaw.teal, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAqM4C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
155
|
+
})("background-color:", colorsRaw.teal, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAqM4C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
156
156
|
const StatusButton = /*#__PURE__*/_styled(DropdownButton, {
|
|
157
157
|
target: "e1d2l9mo4",
|
|
158
158
|
label: "StatusButton"
|
|
159
|
-
})("background-color:", colorsRaw.tealLight, ";color:", colorsRaw.teal, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAyM2C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
159
|
+
})("background-color:", colorsRaw.tealLight, ";color:", colorsRaw.teal, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAyM2C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
160
160
|
const PreviewButtonContainer = /*#__PURE__*/_styled("div", {
|
|
161
161
|
target: "e1d2l9mo3",
|
|
162
162
|
label: "PreviewButtonContainer"
|
|
163
|
-
})("margin-right:12px;color:", colorsRaw.blue, ";display:flex;align-items:center;a,", Icon, "{color:", colorsRaw.blue, ";}", Icon, "{position:relative;top:1px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA8MyC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
163
|
+
})("margin-right:12px;color:", colorsRaw.blue, ";display:flex;align-items:center;a,", Icon, "{color:", colorsRaw.blue, ";}", Icon, "{position:relative;top:1px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA8MyC","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
164
164
|
const RefreshPreviewButton = /*#__PURE__*/_styled("button", {
|
|
165
165
|
target: "e1d2l9mo2",
|
|
166
166
|
label: "RefreshPreviewButton"
|
|
167
|
-
})("background:none;border:0;cursor:pointer;color:", colorsRaw.blue, ";span{margin-right:6px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA+N0C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
167
|
+
})("background:none;border:0;cursor:pointer;color:", colorsRaw.blue, ";span{margin-right:6px;}&:disabled{cursor:default;opacity:0.6;}", Icon, "{", props => props.$spinning && `animation: spin 1s linear infinite;`, ";}@keyframes spin{from{transform:rotate(0deg);}to{transform:rotate(360deg);}}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA+N0C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
168
168
|
const PreviewLink = RefreshPreviewButton.withComponent('a', {
|
|
169
169
|
target: "e1d2l9mo27",
|
|
170
170
|
label: "PreviewLink"
|
|
@@ -177,13 +177,13 @@ const PublishDropDownItem = /*#__PURE__*/_styled(DropdownItem, {
|
|
|
177
177
|
styles: "min-width:initial"
|
|
178
178
|
} : {
|
|
179
179
|
name: "svxjwj",
|
|
180
|
-
styles: "min-width:initial/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA4OgD","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
180
|
+
styles: "min-width:initial/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AA8PgD","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */",
|
|
181
181
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
182
182
|
});
|
|
183
183
|
const StatusDropdownItem = /*#__PURE__*/_styled(DropdownItem, {
|
|
184
184
|
target: "e1d2l9mo0",
|
|
185
185
|
label: "StatusDropdownItem"
|
|
186
|
-
})(Icon, "{color:", colors.infoText, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAgP+C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      loadDeployPreview({ maxAttempts: 3 });\n    }\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton onClick={loadDeployPreview}>\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
186
|
+
})(Icon, "{color:", colors.infoText, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/Editor/EditorToolbar.js"],"names":[],"mappings":"AAkQ+C","file":"../../../../src/components/Editor/EditorToolbar.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { css } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { translate } from 'react-polyglot';\nimport { Link } from 'react-router-dom';\nimport {\n  Icon,\n  Dropdown,\n  DropdownItem,\n  StyledDropdownButton,\n  colorsRaw,\n  colors,\n  components,\n  buttons,\n  zIndex,\n} from 'decap-cms-ui-default';\n\nimport { status } from '../../constants/publishModes';\nimport { SettingsDropdown } from '../UI';\n\nconst styles = {\n  noOverflow: css`\n    overflow: hidden;\n    white-space: nowrap;\n    text-overflow: ellipsis;\n  `,\n  buttonMargin: css`\n    margin: 0 10px;\n  `,\n  toolbarSection: css`\n    height: 100%;\n    display: flex;\n    align-items: center;\n    border: 0 solid ${colors.textFieldBorder};\n  `,\n  publishedButton: css`\n    background-color: ${colorsRaw.tealLight};\n    color: ${colorsRaw.tealDark};\n  `,\n};\n\nconst TooltipText = styled.div`\n  visibility: hidden;\n  width: 321px;\n  background-color: #555;\n  color: #fff;\n  text-align: unset;\n  border-radius: 6px;\n  padding: 5px;\n\n  /* Position the tooltip text */\n  position: absolute;\n  z-index: 1;\n  top: 145%;\n  left: 50%;\n  margin-left: -320px;\n\n  /* Fade in tooltip */\n  opacity: 0;\n  transition: opacity 0.3s;\n`;\n\nconst Tooltip = styled.div`\n  position: relative;\n  display: inline-block;\n  &:hover + ${TooltipText} {\n    visibility: visible;\n    opacity: 0.9;\n  }\n`;\n\nconst TooltipContainer = styled.div`\n  position: relative;\n`;\n\nconst DropdownButton = styled(StyledDropdownButton)`\n  ${styles.noOverflow}\n  @media (max-width: 1200px) {\n    padding-left: 10px;\n  }\n`;\n\nconst ToolbarContainer = styled.div`\n  box-shadow: 0 2px 6px 0 rgba(68, 74, 87, 0.05), 0 1px 3px 0 rgba(68, 74, 87, 0.1),\n    0 2px 54px rgba(0, 0, 0, 0.1);\n  position: absolute;\n  top: 0;\n  left: 0;\n  width: 100%;\n  min-width: 800px;\n  z-index: ${zIndex.zIndex300};\n  background-color: #fff;\n  height: 66px;\n  display: flex;\n  justify-content: space-between;\n`;\n\nconst ToolbarSectionMain = styled.div`\n  ${styles.toolbarSection};\n  flex: 10;\n  display: flex;\n  justify-content: space-between;\n  padding: 0 10px;\n`;\n\nconst ToolbarSubSectionFirst = styled.div`\n  display: flex;\n  align-items: center;\n`;\n\nconst ToolbarSubSectionLast = styled(ToolbarSubSectionFirst)`\n  justify-content: flex-end;\n`;\n\nconst ToolbarSectionBackLink = styled(Link)`\n  ${styles.toolbarSection};\n  border-right-width: 1px;\n  font-weight: normal;\n  padding: 0 20px;\n\n  &:hover,\n  &:focus {\n    background-color: #f1f2f4;\n  }\n`;\n\nconst ToolbarSectionMeta = styled.div`\n  ${styles.toolbarSection};\n  border-left-width: 1px;\n  padding: 0 7px;\n`;\n\nconst ToolbarDropdown = styled(Dropdown)`\n  ${styles.buttonMargin};\n\n  ${Icon} {\n    color: ${colorsRaw.teal};\n  }\n`;\n\nconst BackArrow = styled.div`\n  color: ${colors.textLead};\n  font-size: 21px;\n  font-weight: 600;\n  margin-right: 16px;\n`;\n\nconst BackCollection = styled.div`\n  color: ${colors.textLead};\n  font-size: 14px;\n`;\n\nconst BackStatus = styled.div`\n  margin-top: 6px;\n`;\n\nconst BackStatusUnchanged = styled(BackStatus)`\n  ${components.textBadgeSuccess};\n`;\n\nconst BackStatusChanged = styled(BackStatus)`\n  ${components.textBadgeDanger};\n`;\n\nconst ToolbarButton = styled.button`\n  ${buttons.button};\n  ${buttons.default};\n  ${styles.buttonMargin};\n  ${styles.noOverflow};\n  display: block;\n\n  @media (max-width: 1200px) {\n    padding: 0 10px;\n  }\n`;\n\nconst DeleteButton = styled(ToolbarButton)`\n  ${buttons.lightRed};\n`;\n\nconst SaveButton = styled(ToolbarButton)`\n  ${buttons.lightBlue};\n  &[disabled] {\n    ${buttons.disabled};\n  }\n`;\n\nconst PublishedToolbarButton = styled(DropdownButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishedButton = styled(ToolbarButton)`\n  ${styles.publishedButton}\n`;\n\nconst PublishButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.teal};\n`;\n\nconst StatusButton = styled(DropdownButton)`\n  background-color: ${colorsRaw.tealLight};\n  color: ${colorsRaw.teal};\n`;\n\nconst PreviewButtonContainer = styled.div`\n  margin-right: 12px;\n  color: ${colorsRaw.blue};\n  display: flex;\n  align-items: center;\n\n  a,\n  ${Icon} {\n    color: ${colorsRaw.blue};\n  }\n\n  ${Icon} {\n    position: relative;\n    top: 1px;\n  }\n`;\n\nconst RefreshPreviewButton = styled.button`\n  background: none;\n  border: 0;\n  cursor: pointer;\n  color: ${colorsRaw.blue};\n\n  span {\n    margin-right: 6px;\n  }\n\n  &:disabled {\n    cursor: default;\n    opacity: 0.6;\n  }\n\n  ${Icon} {\n    ${props => props.$spinning && `animation: spin 1s linear infinite;`}\n  }\n\n  @keyframes spin {\n    from {\n      transform: rotate(0deg);\n    }\n    to {\n      transform: rotate(360deg);\n    }\n  }\n`;\n\nconst PreviewLink = RefreshPreviewButton.withComponent('a');\n\nconst PublishDropDownItem = styled(DropdownItem)`\n  min-width: initial;\n`;\n\nconst StatusDropdownItem = styled(DropdownItem)`\n  ${Icon} {\n    color: ${colors.infoText};\n  }\n`;\n\nexport class EditorToolbar extends React.Component {\n  static propTypes = {\n    isPersisting: PropTypes.bool,\n    isPublishing: PropTypes.bool,\n    isUpdatingStatus: PropTypes.bool,\n    isDeleting: PropTypes.bool,\n    onPersist: PropTypes.func.isRequired,\n    onPersistAndNew: PropTypes.func.isRequired,\n    onPersistAndDuplicate: PropTypes.func.isRequired,\n    showDelete: PropTypes.bool.isRequired,\n    onDelete: PropTypes.func.isRequired,\n    onDeleteUnpublishedChanges: PropTypes.func.isRequired,\n    onChangeStatus: PropTypes.func.isRequired,\n    onPublish: PropTypes.func.isRequired,\n    unPublish: PropTypes.func.isRequired,\n    onDuplicate: PropTypes.func.isRequired,\n    onPublishAndNew: PropTypes.func.isRequired,\n    onPublishAndDuplicate: PropTypes.func.isRequired,\n    user: PropTypes.object,\n    hasChanged: PropTypes.bool,\n    displayUrl: PropTypes.string,\n    collection: ImmutablePropTypes.map.isRequired,\n    hasWorkflow: PropTypes.bool,\n    useOpenAuthoring: PropTypes.bool,\n    hasUnpublishedChanges: PropTypes.bool,\n    isNewEntry: PropTypes.bool,\n    isModification: PropTypes.bool,\n    currentStatus: PropTypes.string,\n    onLogoutClick: PropTypes.func.isRequired,\n    deployPreview: PropTypes.object,\n    loadDeployPreview: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    editorBackLink: PropTypes.string.isRequired,\n  };\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorToolbar.propTypes, this.props, 'prop', 'EditorToolbar');\n\n    const { isNewEntry, loadDeployPreview } = this.props;\n    if (!isNewEntry) {\n      // 24 attempts × 5s interval = ~2 min polling window.\n      // With editorial workflow, saving remounts the component (navigates to\n      // the unpublished entry view), so componentDidMount is the primary\n      // polling trigger — not componentDidUpdate.\n      this._pollController = new AbortController();\n      loadDeployPreview({ maxAttempts: 24, signal: this._pollController.signal });\n    }\n  }\n\n  componentDidUpdate(prevProps) {\n    const { isNewEntry, isPersisting, loadDeployPreview } = this.props;\n    if (!isNewEntry && prevProps.isPersisting && !isPersisting) {\n      // Abort any in-flight poll before starting a new one.\n      this._pollController?.abort();\n      this._pollController = new AbortController();\n      // Fires on subsequent saves when the component survives (no remount).\n      // In editorial workflow the first save remounts, so this mainly\n      // covers the second-save-and-beyond case.\n      loadDeployPreview({ maxAttempts: 3, signal: this._pollController.signal });\n    }\n  }\n\n  componentWillUnmount() {\n    this._pollController?.abort();\n  }\n\n  renderSimpleControls = () => {\n    const { collection, hasChanged, isNewEntry, showDelete, onDelete, t } = this.props;\n    const canCreate = collection.get('create');\n\n    return (\n      <>\n        {!isNewEntry && !hasChanged\n          ? this.renderExistingEntrySimplePublishControls({ canCreate })\n          : this.renderNewEntrySimplePublishControls({ canCreate })}\n        <div>\n          {showDelete ? (\n            <DeleteButton onClick={onDelete}>{t('editor.editorToolbar.deleteEntry')}</DeleteButton>\n          ) : null}\n        </div>\n      </>\n    );\n  };\n\n  renderDeployPreviewControls = label => {\n    const { deployPreview = {}, loadDeployPreview, t } = this.props;\n    const { url, status, isFetching } = deployPreview;\n\n    if (!status) {\n      return;\n    }\n\n    const deployPreviewReady = status === 'SUCCESS' && !isFetching;\n    return (\n      <PreviewButtonContainer>\n        {deployPreviewReady ? (\n          <PreviewLink rel=\"noopener noreferrer\" target=\"_blank\" href={url}>\n            <span>{label}</span>\n            <Icon type=\"new-tab\" size=\"xsmall\" />\n          </PreviewLink>\n        ) : (\n          <RefreshPreviewButton\n            onClick={loadDeployPreview}\n            disabled={isFetching}\n            $spinning={isFetching}\n          >\n            <span>{t('editor.editorToolbar.deployPreviewPendingButtonLabel')}</span>\n            <Icon type=\"refresh\" size=\"xsmall\" />\n          </RefreshPreviewButton>\n        )}\n      </PreviewButtonContainer>\n    );\n  };\n\n  renderStatusInfoTooltip = () => {\n    const { t, currentStatus } = this.props;\n\n    const statusToLocaleKey = {\n      [status.get('DRAFT')]: 'statusInfoTooltipDraft',\n      [status.get('PENDING_REVIEW')]: 'statusInfoTooltipInReview',\n    };\n\n    const statusKey = Object.keys(statusToLocaleKey).find(key => key === currentStatus);\n    return (\n      <TooltipContainer>\n        <Tooltip>\n          <Icon type=\"info-circle\" size=\"small\" className=\"tooltip\" />\n        </Tooltip>\n        {statusKey && (\n          <TooltipText>{t(`editor.editorToolbar.${statusToLocaleKey[statusKey]}`)}</TooltipText>\n        )}\n      </TooltipContainer>\n    );\n  };\n\n  renderWorkflowStatusControls = () => {\n    const { isUpdatingStatus, onChangeStatus, currentStatus, t, useOpenAuthoring } = this.props;\n\n    const statusToTranslation = {\n      [status.get('DRAFT')]: t('editor.editorToolbar.draft'),\n      [status.get('PENDING_REVIEW')]: t('editor.editorToolbar.inReview'),\n      [status.get('PENDING_PUBLISH')]: t('editor.editorToolbar.ready'),\n    };\n\n    const buttonText = isUpdatingStatus\n      ? t('editor.editorToolbar.updating')\n      : t('editor.editorToolbar.status', { status: statusToTranslation[currentStatus] });\n\n    return (\n      <>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"120px\"\n          renderButton={() => <StatusButton>{buttonText}</StatusButton>}\n        >\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.draft')}\n            onClick={() => onChangeStatus('DRAFT')}\n            icon={currentStatus === status.get('DRAFT') ? 'check' : null}\n          />\n          <StatusDropdownItem\n            label={t('editor.editorToolbar.inReview')}\n            onClick={() => onChangeStatus('PENDING_REVIEW')}\n            icon={currentStatus === status.get('PENDING_REVIEW') ? 'check' : null}\n          />\n          {useOpenAuthoring ? (\n            ''\n          ) : (\n            <StatusDropdownItem\n              label={t('editor.editorToolbar.ready')}\n              onClick={() => onChangeStatus('PENDING_PUBLISH')}\n              icon={currentStatus === status.get('PENDING_PUBLISH') ? 'check' : null}\n            />\n          )}\n        </ToolbarDropdown>\n        {useOpenAuthoring && this.renderStatusInfoTooltip()}\n      </>\n    );\n  };\n\n  renderNewEntryWorkflowPublishControls = ({ canCreate, canPublish }) => {\n    const { isPublishing, onPublish, onPublishAndNew, onPublishAndDuplicate, t } = this.props;\n\n    return canPublish ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"200px\"\n        renderButton={() => (\n          <PublishButton>\n            {isPublishing\n              ? t('editor.editorToolbar.publishing')\n              : t('editor.editorToolbar.publish')}\n          </PublishButton>\n        )}\n      >\n        <PublishDropDownItem\n          label={t('editor.editorToolbar.publishNow')}\n          icon=\"arrow\"\n          iconDirection=\"right\"\n          onClick={onPublish}\n        />\n        {canCreate ? (\n          <>\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndCreateNew')}\n              icon=\"add\"\n              onClick={onPublishAndNew}\n            />\n            <PublishDropDownItem\n              label={t('editor.editorToolbar.publishAndDuplicate')}\n              icon=\"add\"\n              onClick={onPublishAndDuplicate}\n            />\n          </>\n        ) : null}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntryWorkflowPublishControls = ({ canCreate, canPublish, canDelete }) => {\n    const { unPublish, onDuplicate, isPersisting, t } = this.props;\n\n    return canPublish || canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        key=\"td-publish-create\"\n        renderButton={() => (\n          <PublishedToolbarButton>\n            {isPersisting\n              ? t('editor.editorToolbar.unpublishing')\n              : t('editor.editorToolbar.published')}\n          </PublishedToolbarButton>\n        )}\n      >\n        {canDelete && canPublish && (\n          <DropdownItem\n            label={t('editor.editorToolbar.unpublish')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={unPublish}\n          />\n        )}\n        {canCreate && (\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        )}\n      </ToolbarDropdown>\n    ) : (\n      ''\n    );\n  };\n\n  renderExistingEntrySimplePublishControls = ({ canCreate }) => {\n    const { onDuplicate, t } = this.props;\n    return canCreate ? (\n      <ToolbarDropdown\n        dropdownTopOverlap=\"40px\"\n        dropdownWidth=\"max-content\"\n        renderButton={() => (\n          <PublishedToolbarButton>{t('editor.editorToolbar.published')}</PublishedToolbarButton>\n        )}\n      >\n        {\n          <DropdownItem\n            label={t('editor.editorToolbar.duplicate')}\n            icon=\"add\"\n            onClick={onDuplicate}\n          />\n        }\n      </ToolbarDropdown>\n    ) : (\n      <PublishedButton>{t('editor.editorToolbar.published')}</PublishedButton>\n    );\n  };\n\n  renderNewEntrySimplePublishControls = ({ canCreate }) => {\n    const { onPersist, onPersistAndNew, onPersistAndDuplicate, isPersisting, t } = this.props;\n\n    return (\n      <div>\n        <ToolbarDropdown\n          dropdownTopOverlap=\"40px\"\n          dropdownWidth=\"max-content\"\n          renderButton={() => (\n            <PublishButton>\n              {isPersisting\n                ? t('editor.editorToolbar.publishing')\n                : t('editor.editorToolbar.publish')}\n            </PublishButton>\n          )}\n        >\n          <DropdownItem\n            label={t('editor.editorToolbar.publishNow')}\n            icon=\"arrow\"\n            iconDirection=\"right\"\n            onClick={onPersist}\n          />\n          {canCreate ? (\n            <>\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndCreateNew')}\n                icon=\"add\"\n                onClick={onPersistAndNew}\n              />\n              <DropdownItem\n                label={t('editor.editorToolbar.publishAndDuplicate')}\n                icon=\"add\"\n                onClick={onPersistAndDuplicate}\n              />\n            </>\n          ) : null}\n        </ToolbarDropdown>\n      </div>\n    );\n  };\n\n  renderSimpleDeployPreviewControls = () => {\n    const { hasChanged, isNewEntry, t } = this.props;\n\n    if (!isNewEntry && !hasChanged) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  renderWorkflowControls = () => {\n    const {\n      onPersist,\n      onDelete,\n      onDeleteUnpublishedChanges,\n      showDelete,\n      hasChanged,\n      hasUnpublishedChanges,\n      useOpenAuthoring,\n      isPersisting,\n      isDeleting,\n      isNewEntry,\n      isModification,\n      currentStatus,\n      collection,\n      t,\n    } = this.props;\n\n    const canCreate = collection.get('create');\n    const canPublish = collection.get('publish') && !useOpenAuthoring;\n    const canDelete = collection.get('delete', true);\n\n    const deleteLabel =\n      (hasUnpublishedChanges &&\n        isModification &&\n        t('editor.editorToolbar.deleteUnpublishedChanges')) ||\n      (hasUnpublishedChanges &&\n        (isNewEntry || !isModification) &&\n        t('editor.editorToolbar.deleteUnpublishedEntry')) ||\n      (!hasUnpublishedChanges && !isModification && t('editor.editorToolbar.deletePublishedEntry'));\n\n    return [\n      <SaveButton\n        disabled={!hasChanged}\n        key=\"save-button\"\n        onClick={() => hasChanged && onPersist()}\n      >\n        {isPersisting ? t('editor.editorToolbar.saving') : t('editor.editorToolbar.save')}\n      </SaveButton>,\n      currentStatus\n        ? [\n            <React.Fragment key=\"workflow-status-controls\">\n              {this.renderWorkflowStatusControls()}\n              {!hasChanged && this.renderNewEntryWorkflowPublishControls({ canCreate, canPublish })}\n            </React.Fragment>,\n          ]\n        : !isNewEntry && (\n            <React.Fragment key=\"existing-entry-workflow-publish-controls\">\n              {this.renderExistingEntryWorkflowPublishControls({\n                canCreate,\n                canPublish,\n                canDelete,\n              })}\n            </React.Fragment>\n          ),\n      (!showDelete || useOpenAuthoring) && !hasUnpublishedChanges && !isModification ? null : (\n        <DeleteButton\n          key=\"delete-button\"\n          onClick={hasUnpublishedChanges ? onDeleteUnpublishedChanges : onDelete}\n        >\n          {isDeleting ? t('editor.editorToolbar.deleting') : deleteLabel}\n        </DeleteButton>\n      ),\n    ];\n  };\n\n  renderWorkflowDeployPreviewControls = () => {\n    const { currentStatus, isNewEntry, t } = this.props;\n\n    if (currentStatus) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployPreviewButtonLabel'));\n    }\n\n    /**\n     * Publish control for published workflow entry.\n     */\n    if (!isNewEntry) {\n      return this.renderDeployPreviewControls(t('editor.editorToolbar.deployButtonLabel'));\n    }\n  };\n\n  render() {\n    const {\n      user,\n      hasChanged,\n      displayUrl,\n      collection,\n      hasWorkflow,\n      onLogoutClick,\n      t,\n      editorBackLink,\n    } = this.props;\n\n    return (\n      <ToolbarContainer>\n        <ToolbarSectionBackLink to={editorBackLink}>\n          <BackArrow>←</BackArrow>\n          <div>\n            <BackCollection>\n              {t('editor.editorToolbar.backCollection', {\n                collectionLabel: collection.get('label'),\n              })}\n            </BackCollection>\n            {hasChanged ? (\n              <BackStatusChanged>{t('editor.editorToolbar.unsavedChanges')}</BackStatusChanged>\n            ) : (\n              <BackStatusUnchanged>{t('editor.editorToolbar.changesSaved')}</BackStatusUnchanged>\n            )}\n          </div>\n        </ToolbarSectionBackLink>\n        <ToolbarSectionMain>\n          <ToolbarSubSectionFirst>\n            {hasWorkflow ? this.renderWorkflowControls() : this.renderSimpleControls()}\n          </ToolbarSubSectionFirst>\n          <ToolbarSubSectionLast>\n            {hasWorkflow\n              ? this.renderWorkflowDeployPreviewControls()\n              : this.renderSimpleDeployPreviewControls()}\n          </ToolbarSubSectionLast>\n        </ToolbarSectionMain>\n        <ToolbarSectionMeta>\n          <SettingsDropdown\n            displayUrl={displayUrl}\n            imageUrl={user?.avatar_url}\n            onLogoutClick={onLogoutClick}\n          />\n        </ToolbarSectionMeta>\n      </ToolbarContainer>\n    );\n  }\n}\n\nexport default translate()(EditorToolbar);\n"]} */"));
|
|
187
187
|
export class EditorToolbar extends React.Component {
|
|
188
188
|
static propTypes = {
|
|
189
189
|
isPersisting: PropTypes.bool,
|
|
@@ -226,8 +226,14 @@ export class EditorToolbar extends React.Component {
|
|
|
226
226
|
loadDeployPreview
|
|
227
227
|
} = this.props;
|
|
228
228
|
if (!isNewEntry) {
|
|
229
|
+
// 24 attempts × 5s interval = ~2 min polling window.
|
|
230
|
+
// With editorial workflow, saving remounts the component (navigates to
|
|
231
|
+
// the unpublished entry view), so componentDidMount is the primary
|
|
232
|
+
// polling trigger — not componentDidUpdate.
|
|
233
|
+
this._pollController = new AbortController();
|
|
229
234
|
loadDeployPreview({
|
|
230
|
-
maxAttempts:
|
|
235
|
+
maxAttempts: 24,
|
|
236
|
+
signal: this._pollController.signal
|
|
231
237
|
});
|
|
232
238
|
}
|
|
233
239
|
}
|
|
@@ -238,11 +244,21 @@ export class EditorToolbar extends React.Component {
|
|
|
238
244
|
loadDeployPreview
|
|
239
245
|
} = this.props;
|
|
240
246
|
if (!isNewEntry && prevProps.isPersisting && !isPersisting) {
|
|
247
|
+
// Abort any in-flight poll before starting a new one.
|
|
248
|
+
this._pollController?.abort();
|
|
249
|
+
this._pollController = new AbortController();
|
|
250
|
+
// Fires on subsequent saves when the component survives (no remount).
|
|
251
|
+
// In editorial workflow the first save remounts, so this mainly
|
|
252
|
+
// covers the second-save-and-beyond case.
|
|
241
253
|
loadDeployPreview({
|
|
242
|
-
maxAttempts: 3
|
|
254
|
+
maxAttempts: 3,
|
|
255
|
+
signal: this._pollController.signal
|
|
243
256
|
});
|
|
244
257
|
}
|
|
245
258
|
}
|
|
259
|
+
componentWillUnmount() {
|
|
260
|
+
this._pollController?.abort();
|
|
261
|
+
}
|
|
246
262
|
renderSimpleControls = () => {
|
|
247
263
|
const {
|
|
248
264
|
collection,
|
|
@@ -284,7 +300,9 @@ export class EditorToolbar extends React.Component {
|
|
|
284
300
|
type: "new-tab",
|
|
285
301
|
size: "xsmall"
|
|
286
302
|
})) : ___EmotionJSX(RefreshPreviewButton, {
|
|
287
|
-
onClick: loadDeployPreview
|
|
303
|
+
onClick: loadDeployPreview,
|
|
304
|
+
disabled: isFetching,
|
|
305
|
+
$spinning: isFetching
|
|
288
306
|
}, ___EmotionJSX("span", null, t('editor.editorToolbar.deployPreviewPendingButtonLabel')), ___EmotionJSX(Icon, {
|
|
289
307
|
type: "refresh",
|
|
290
308
|
size: "xsmall"
|