decap-cms-core 3.6.2 → 3.7.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 +26 -26
- package/dist/decap-cms-core.js.LICENSE.txt +22 -4
- package/dist/decap-cms-core.js.map +1 -1
- package/dist/esm/actions/config.js +57 -49
- package/dist/esm/actions/editorialWorkflow.js +4 -4
- package/dist/esm/actions/entries.js +8 -14
- package/dist/esm/actions/mediaLibrary.js +6 -11
- package/dist/esm/actions/search.js +2 -2
- package/dist/esm/actions/status.js +2 -8
- package/dist/esm/backend.js +70 -79
- package/dist/esm/bootstrap.js +2 -2
- package/dist/esm/components/App/App.js +28 -34
- package/dist/esm/components/App/Header.js +32 -39
- package/dist/esm/components/Collection/Collection.js +45 -48
- package/dist/esm/components/Collection/CollectionSearch.js +76 -81
- package/dist/esm/components/Collection/CollectionTop.js +1 -2
- package/dist/esm/components/Collection/Entries/Entries.js +2 -4
- package/dist/esm/components/Collection/Entries/EntriesCollection.js +25 -29
- package/dist/esm/components/Collection/Entries/EntriesSearch.js +34 -38
- package/dist/esm/components/Collection/Entries/EntryCard.js +8 -13
- package/dist/esm/components/Collection/Entries/EntryListing.js +72 -76
- package/dist/esm/components/Collection/FilterControl.js +1 -1
- package/dist/esm/components/Collection/GroupControl.js +1 -1
- package/dist/esm/components/Collection/NestedCollection.js +50 -53
- package/dist/esm/components/Collection/Sidebar.js +35 -38
- package/dist/esm/components/Collection/SortControl.js +3 -3
- package/dist/esm/components/Collection/ViewStyleControl.js +1 -2
- package/dist/esm/components/Editor/Editor.js +197 -201
- package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +79 -87
- package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +75 -86
- package/dist/esm/components/Editor/EditorControlPane/Widget.js +226 -228
- package/dist/esm/components/Editor/EditorInterface.js +69 -80
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +1 -2
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +20 -28
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +163 -161
- package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +4 -8
- package/dist/esm/components/Editor/EditorToolbar.js +335 -347
- package/dist/esm/components/Editor/withWorkflow.js +5 -6
- package/dist/esm/components/MediaLibrary/MediaLibrary.js +304 -294
- package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +40 -46
- package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +1 -2
- package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +8 -13
- package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +3 -3
- package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +1 -2
- package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +3 -6
- package/dist/esm/components/UI/DragDrop.js +15 -23
- package/dist/esm/components/UI/ErrorBoundary.js +23 -25
- package/dist/esm/components/UI/Modal.js +10 -12
- package/dist/esm/components/UI/Notifications.js +4 -8
- package/dist/esm/components/UI/SettingsDropdown.js +4 -8
- package/dist/esm/components/Workflow/Workflow.js +19 -20
- package/dist/esm/components/Workflow/WorkflowCard.js +2 -4
- package/dist/esm/components/Workflow/WorkflowList.js +105 -113
- package/dist/esm/constants/configSchema.js +18 -16
- package/dist/esm/formats/formats.js +11 -12
- package/dist/esm/formats/frontmatter.js +17 -21
- package/dist/esm/formats/yaml.js +2 -6
- package/dist/esm/index.js +3 -7
- package/dist/esm/integrations/providers/algolia/implementation.js +12 -14
- package/dist/esm/integrations/providers/assetStore/implementation.js +10 -12
- package/dist/esm/lib/formatters.js +13 -17
- package/dist/esm/lib/i18n.js +35 -33
- package/dist/esm/lib/phrases.js +2 -2
- package/dist/esm/lib/registry.js +35 -35
- package/dist/esm/lib/serializeEntryValues.js +3 -3
- package/dist/esm/lib/stega.js +142 -0
- package/dist/esm/lib/urlHelper.js +16 -18
- package/dist/esm/mediaLibrary.js +3 -4
- package/dist/esm/reducers/collections.js +26 -42
- package/dist/esm/reducers/combinedReducer.js +3 -6
- package/dist/esm/reducers/config.js +3 -7
- package/dist/esm/reducers/editorialWorkflow.js +5 -9
- package/dist/esm/reducers/entries.js +33 -35
- package/dist/esm/reducers/entryDraft.js +2 -2
- package/dist/esm/reducers/integrations.js +8 -14
- package/dist/esm/reducers/mediaLibrary.js +18 -20
- package/dist/esm/reducers/notifications.js +4 -8
- package/dist/esm/types/immutable.js +7 -1
- package/dist/esm/valueObjects/AssetProxy.js +1 -9
- package/dist/esm/valueObjects/EditorComponent.js +18 -25
- package/dist/esm/valueObjects/Entry.js +2 -2
- package/index.d.ts +2 -0
- package/package.json +8 -11
- package/src/actions/__tests__/config.spec.js +3 -3
- package/src/actions/config.ts +3 -1
- package/src/actions/editorialWorkflow.ts +1 -1
- package/src/actions/entries.ts +1 -1
- package/src/actions/search.ts +1 -1
- package/src/backend.ts +8 -1
- package/src/components/App/App.js +5 -0
- package/src/components/App/Header.js +3 -0
- package/src/components/Collection/Collection.js +5 -0
- package/src/components/Collection/CollectionSearch.js +5 -0
- package/src/components/Collection/Entries/EntriesCollection.js +4 -1
- package/src/components/Collection/Entries/EntriesSearch.js +4 -1
- package/src/components/Collection/Entries/EntryListing.js +5 -0
- package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +0 -4
- package/src/components/Collection/NestedCollection.js +6 -1
- package/src/components/Collection/Sidebar.js +5 -0
- package/src/components/Editor/Editor.js +4 -1
- package/src/components/Editor/EditorControlPane/EditorControl.js +7 -1
- package/src/components/Editor/EditorControlPane/Widget.js +5 -0
- package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +1 -1
- package/src/components/Editor/EditorToolbar.js +6 -3
- package/src/components/Editor/__tests__/Editor.spec.js +3 -4
- package/src/components/Editor/__tests__/__snapshots__/Editor.spec.js.snap +5 -5
- package/src/components/Editor/__tests__/__snapshots__/EditorToolbar.spec.js.snap +708 -393
- package/src/components/MediaLibrary/MediaLibrary.js +5 -1
- package/src/components/MediaLibrary/MediaLibraryModal.js +1 -1
- package/src/components/UI/ErrorBoundary.js +6 -1
- package/src/components/UI/Modal.js +3 -0
- package/src/components/Workflow/Workflow.js +3 -0
- package/src/components/Workflow/WorkflowList.js +5 -0
- package/src/constants/__tests__/configSchema.spec.js +1 -1
- package/src/formats/formats.ts +1 -1
- package/src/integrations/providers/algolia/implementation.js +2 -2
- package/src/integrations/providers/assetStore/implementation.js +2 -1
- package/src/lib/formatters.ts +4 -1
- package/src/lib/i18n.ts +3 -1
- package/src/lib/phrases.js +1 -1
- package/src/lib/serializeEntryValues.js +1 -1
- package/src/lib/stega.ts +145 -0
- package/src/lib/urlHelper.ts +4 -1
- package/src/mediaLibrary.ts +1 -1
- package/src/reducers/collections.ts +2 -1
- package/src/reducers/editorialWorkflow.ts +1 -1
- package/src/reducers/entries.ts +6 -1
- package/src/reducers/entryDraft.js +1 -1
- package/src/types/immutable.ts +10 -0
- package/src/types/redux.ts +2 -0
- package/src/valueObjects/EditorComponent.js +1 -1
- package/src/valueObjects/Entry.ts +1 -1
|
@@ -1,15 +1,5 @@
|
|
|
1
1
|
import _styled from "@emotion/styled/base";
|
|
2
|
-
|
|
3
|
-
import _partial from "lodash/partial";
|
|
4
|
-
const _excluded = ["node"];
|
|
5
|
-
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
|
6
|
-
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
|
|
7
|
-
function _extends() { _extends = Object.assign ? Object.assign.bind() : function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }
|
|
8
|
-
function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }
|
|
9
|
-
function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; }
|
|
10
|
-
function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
11
|
-
function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : String(i); }
|
|
12
|
-
function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
|
|
2
|
+
function _extends() { return _extends = Object.assign ? Object.assign.bind() : function (n) { for (var e = 1; e < arguments.length; e++) { var t = arguments[e]; for (var r in t) ({}).hasOwnProperty.call(t, r) && (n[r] = t[r]); } return n; }, _extends.apply(null, arguments); }
|
|
13
3
|
function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
|
|
14
4
|
import React from 'react';
|
|
15
5
|
import { bindActionCreators } from 'redux';
|
|
@@ -17,6 +7,8 @@ import PropTypes from 'prop-types';
|
|
|
17
7
|
import ImmutablePropTypes from 'react-immutable-proptypes';
|
|
18
8
|
import { translate } from 'react-polyglot';
|
|
19
9
|
import { ClassNames, Global, css as coreCss } from '@emotion/react';
|
|
10
|
+
import partial from 'lodash/partial';
|
|
11
|
+
import uniqueId from 'lodash/uniqueId';
|
|
20
12
|
import { connect } from 'react-redux';
|
|
21
13
|
import { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';
|
|
22
14
|
import ReactMarkdown from 'react-markdown';
|
|
@@ -80,8 +72,7 @@ const ControlContainer = /*#__PURE__*/_styled("div", {
|
|
|
80
72
|
styles: "margin-top:16px;&:first-of-type{margin-top:36px;}"
|
|
81
73
|
} : {
|
|
82
74
|
name: "gpgisz",
|
|
83
|
-
styles: "margin-top:16px;&:first-of-type{margin-top:36px;}",
|
|
84
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AAsEmC","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { partial, uniqueId } from 'lodash';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */",
|
|
75
|
+
styles: "margin-top:16px;&:first-of-type{margin-top:36px;}/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AAuEmC","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport partial from 'lodash/partial';\nimport uniqueId from 'lodash/uniqueId';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorControl.propTypes, this.props, 'prop', 'EditorControl');\n  }\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */",
|
|
85
76
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
86
77
|
});
|
|
87
78
|
const ControlTopbar = /*#__PURE__*/_styled("div", {
|
|
@@ -92,18 +83,17 @@ const ControlTopbar = /*#__PURE__*/_styled("div", {
|
|
|
92
83
|
styles: "display:flex;justify-content:space-between;gap:20px;align-items:end"
|
|
93
84
|
} : {
|
|
94
85
|
name: "hvtmtt",
|
|
95
|
-
styles: "display:flex;justify-content:space-between;gap:20px;align-items:end",
|
|
96
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AA8EgC","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { partial, uniqueId } from 'lodash';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */",
|
|
86
|
+
styles: "display:flex;justify-content:space-between;gap:20px;align-items:end/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AA+EgC","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport partial from 'lodash/partial';\nimport uniqueId from 'lodash/uniqueId';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorControl.propTypes, this.props, 'prop', 'EditorControl');\n  }\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */",
|
|
97
87
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
98
88
|
});
|
|
99
89
|
const ControlErrorsList = /*#__PURE__*/_styled("ul", {
|
|
100
90
|
target: "ec1nk3l1",
|
|
101
91
|
label: "ControlErrorsList"
|
|
102
|
-
})("list-style-type:none;font-size:12px;color:", colors.errorText, ";text-align:right;text-transform:uppercase;font-weight:600;margin:0;padding:2px 0 3px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AAoFmC","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { partial, uniqueId } from 'lodash';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */"));
|
|
92
|
+
})("list-style-type:none;font-size:12px;color:", colors.errorText, ";text-align:right;text-transform:uppercase;font-weight:600;margin:0;padding:2px 0 3px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AAqFmC","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport partial from 'lodash/partial';\nimport uniqueId from 'lodash/uniqueId';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorControl.propTypes, this.props, 'prop', 'EditorControl');\n  }\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */"));
|
|
103
93
|
export const ControlHint = /*#__PURE__*/_styled("p", {
|
|
104
94
|
target: "ec1nk3l0",
|
|
105
95
|
label: "ControlHint"
|
|
106
|
-
})("margin-bottom:0;padding:6px 0 0;font-size:12px;color:", props => props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel, ";transition:color ", transitions.main, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AA+FmC","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { partial, uniqueId } from 'lodash';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */"));
|
|
96
|
+
})("margin-bottom:0;padding:6px 0 0;font-size:12px;color:", props => props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel, ";transition:color ", transitions.main, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AAgGmC","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport partial from 'lodash/partial';\nimport uniqueId from 'lodash/uniqueId';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorControl.propTypes, this.props, 'prop', 'EditorControl');\n  }\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */"));
|
|
107
97
|
function LabelComponent({
|
|
108
98
|
field,
|
|
109
99
|
isActive,
|
|
@@ -121,22 +111,59 @@ function LabelComponent({
|
|
|
121
111
|
return labelComponent;
|
|
122
112
|
}
|
|
123
113
|
class EditorControl extends React.Component {
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
114
|
+
static propTypes = {
|
|
115
|
+
value: PropTypes.oneOfType([PropTypes.node, PropTypes.object, PropTypes.string, PropTypes.bool]),
|
|
116
|
+
field: ImmutablePropTypes.map.isRequired,
|
|
117
|
+
fieldsMetaData: ImmutablePropTypes.map,
|
|
118
|
+
fieldsErrors: ImmutablePropTypes.map,
|
|
119
|
+
mediaPaths: ImmutablePropTypes.map.isRequired,
|
|
120
|
+
boundGetAsset: PropTypes.func.isRequired,
|
|
121
|
+
onChange: PropTypes.func.isRequired,
|
|
122
|
+
openMediaLibrary: PropTypes.func.isRequired,
|
|
123
|
+
addAsset: PropTypes.func.isRequired,
|
|
124
|
+
removeInsertedMedia: PropTypes.func.isRequired,
|
|
125
|
+
persistMedia: PropTypes.func.isRequired,
|
|
126
|
+
onValidate: PropTypes.func,
|
|
127
|
+
controlRef: PropTypes.func,
|
|
128
|
+
query: PropTypes.func.isRequired,
|
|
129
|
+
queryHits: PropTypes.object,
|
|
130
|
+
isFetching: PropTypes.bool,
|
|
131
|
+
clearSearch: PropTypes.func.isRequired,
|
|
132
|
+
clearFieldErrors: PropTypes.func.isRequired,
|
|
133
|
+
loadEntry: PropTypes.func.isRequired,
|
|
134
|
+
t: PropTypes.func.isRequired,
|
|
135
|
+
isEditorComponent: PropTypes.bool,
|
|
136
|
+
isNewEditorComponent: PropTypes.bool,
|
|
137
|
+
parentIds: PropTypes.arrayOf(PropTypes.string),
|
|
138
|
+
entry: ImmutablePropTypes.map.isRequired,
|
|
139
|
+
collection: ImmutablePropTypes.map.isRequired,
|
|
140
|
+
isDisabled: PropTypes.bool,
|
|
141
|
+
isHidden: PropTypes.bool,
|
|
142
|
+
isFieldDuplicate: PropTypes.func,
|
|
143
|
+
isFieldHidden: PropTypes.func,
|
|
144
|
+
locale: PropTypes.string,
|
|
145
|
+
isParentListCollapsed: PropTypes.bool
|
|
146
|
+
};
|
|
147
|
+
static defaultProps = {
|
|
148
|
+
parentIds: []
|
|
149
|
+
};
|
|
150
|
+
state = {
|
|
151
|
+
activeLabel: false
|
|
152
|
+
};
|
|
153
|
+
uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);
|
|
154
|
+
componentDidMount() {
|
|
155
|
+
// Manually validate PropTypes - React 19 breaking change
|
|
156
|
+
PropTypes.checkPropTypes(EditorControl.propTypes, this.props, 'prop', 'EditorControl');
|
|
139
157
|
}
|
|
158
|
+
isAncestorOfFieldError = () => {
|
|
159
|
+
const {
|
|
160
|
+
fieldsErrors
|
|
161
|
+
} = this.props;
|
|
162
|
+
if (fieldsErrors && fieldsErrors.size > 0) {
|
|
163
|
+
return Object.values(fieldsErrors.toJS()).some(arr => arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)));
|
|
164
|
+
}
|
|
165
|
+
return false;
|
|
166
|
+
};
|
|
140
167
|
render() {
|
|
141
168
|
const {
|
|
142
169
|
value,
|
|
@@ -197,7 +224,7 @@ class EditorControl extends React.Component {
|
|
|
197
224
|
${isHidden && styleStrings.hidden};
|
|
198
225
|
`
|
|
199
226
|
}, ___EmotionJSX(ControlTopbar, null, widget.globalStyles && ___EmotionJSX(Global, {
|
|
200
|
-
styles: /*#__PURE__*/coreCss(widget.globalStyles, ";;label:EditorControl;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AAsP6D","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport { partial, uniqueId } from 'lodash';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */"))
|
|
227
|
+
styles: /*#__PURE__*/coreCss(widget.globalStyles, ";;label:EditorControl;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorControlPane/EditorControl.js"],"names":[],"mappings":"AA4P6D","file":"../../../../../src/components/Editor/EditorControlPane/EditorControl.js","sourcesContent":["import React from 'react';\nimport { bindActionCreators } from 'redux';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport { translate } from 'react-polyglot';\nimport { ClassNames, Global, css as coreCss } from '@emotion/react';\nimport styled from '@emotion/styled';\nimport partial from 'lodash/partial';\nimport uniqueId from 'lodash/uniqueId';\nimport { connect } from 'react-redux';\nimport { FieldLabel, colors, transitions, lengths, borders } from 'decap-cms-ui-default';\nimport ReactMarkdown from 'react-markdown';\nimport gfm from 'remark-gfm';\n\nimport { resolveWidget, getEditorComponents } from '../../../lib/registry';\nimport { clearFieldErrors, tryLoadEntry, validateMetaField } from '../../../actions/entries';\nimport { addAsset, boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { query, clearSearch } from '../../../actions/search';\nimport {\n  openMediaLibrary,\n  removeInsertedMedia,\n  clearMediaControl,\n  removeMediaControl,\n  persistMedia,\n} from '../../../actions/mediaLibrary';\nimport Widget from './Widget';\n\n/**\n * This is a necessary bridge as we are still passing classnames to widgets\n * for styling. Once that changes we can stop storing raw style strings like\n * this.\n */\nconst styleStrings = {\n  widget: `\n    display: block;\n    width: 100%;\n    padding: ${lengths.inputPadding};\n    margin: 0;\n    border: ${borders.textField};\n    border-radius: ${lengths.borderRadius};\n    border-top-left-radius: 0;\n    outline: 0;\n    box-shadow: none;\n    background-color: ${colors.inputBackground};\n    color: #444a57;\n    transition: border-color ${transitions.main};\n    position: relative;\n    font-size: 15px;\n    line-height: 1.5;\n\n    select& {\n      text-indent: 14px;\n      height: 58px;\n    }\n  `,\n  widgetActive: `\n    border-color: ${colors.active};\n  `,\n  widgetError: `\n    border-color: ${colors.errorText};\n  `,\n  disabled: `\n    pointer-events: none;\n    opacity: 0.5;\n  `,\n  hidden: `\n    visibility: hidden;\n  `,\n};\n\nconst ControlContainer = styled.div`\n  margin-top: 16px;\n\n  &:first-of-type {\n    margin-top: 36px;\n  }\n`;\n\nconst ControlTopbar = styled.div`\n  display: flex;\n  justify-content: space-between;\n  gap: 20px;\n  align-items: end;\n`;\nconst ControlErrorsList = styled.ul`\n  list-style-type: none;\n  font-size: 12px;\n  color: ${colors.errorText};\n  text-align: right;\n  text-transform: uppercase;\n  font-weight: 600;\n  margin: 0;\n  padding: 2px 0 3px;\n`;\n\nexport const ControlHint = styled.p`\n  margin-bottom: 0;\n  padding: 6px 0 0;\n  font-size: 12px;\n  color: ${props =>\n    props.error ? colors.errorText : props.active ? colors.active : colors.controlLabel};\n  transition: color ${transitions.main};\n`;\n\nfunction LabelComponent({ field, isActive, hasErrors, uniqueFieldId, isFieldOptional, t }) {\n  const label = `${field.get('label', field.get('name'))}`;\n  const labelComponent = (\n    <FieldLabel isActive={isActive} hasErrors={hasErrors} htmlFor={uniqueFieldId}>\n      {isFieldOptional ? (\n        <>\n          {label}\n          <span>{` (${t('editor.editorControl.field.optional')})`}</span>\n        </>\n      ) : (\n        label\n      )}\n    </FieldLabel>\n  );\n\n  return labelComponent;\n}\n\nclass EditorControl extends React.Component {\n  static propTypes = {\n    value: PropTypes.oneOfType([\n      PropTypes.node,\n      PropTypes.object,\n      PropTypes.string,\n      PropTypes.bool,\n    ]),\n    field: ImmutablePropTypes.map.isRequired,\n    fieldsMetaData: ImmutablePropTypes.map,\n    fieldsErrors: ImmutablePropTypes.map,\n    mediaPaths: ImmutablePropTypes.map.isRequired,\n    boundGetAsset: PropTypes.func.isRequired,\n    onChange: PropTypes.func.isRequired,\n    openMediaLibrary: PropTypes.func.isRequired,\n    addAsset: PropTypes.func.isRequired,\n    removeInsertedMedia: PropTypes.func.isRequired,\n    persistMedia: PropTypes.func.isRequired,\n    onValidate: PropTypes.func,\n    controlRef: PropTypes.func,\n    query: PropTypes.func.isRequired,\n    queryHits: PropTypes.object,\n    isFetching: PropTypes.bool,\n    clearSearch: PropTypes.func.isRequired,\n    clearFieldErrors: PropTypes.func.isRequired,\n    loadEntry: PropTypes.func.isRequired,\n    t: PropTypes.func.isRequired,\n    isEditorComponent: PropTypes.bool,\n    isNewEditorComponent: PropTypes.bool,\n    parentIds: PropTypes.arrayOf(PropTypes.string),\n    entry: ImmutablePropTypes.map.isRequired,\n    collection: ImmutablePropTypes.map.isRequired,\n    isDisabled: PropTypes.bool,\n    isHidden: PropTypes.bool,\n    isFieldDuplicate: PropTypes.func,\n    isFieldHidden: PropTypes.func,\n    locale: PropTypes.string,\n    isParentListCollapsed: PropTypes.bool,\n  };\n\n  static defaultProps = {\n    parentIds: [],\n  };\n\n  state = {\n    activeLabel: false,\n  };\n\n  uniqueFieldId = uniqueId(`${this.props.field.get('name')}-field-`);\n\n  componentDidMount() {\n    // Manually validate PropTypes - React 19 breaking change\n    PropTypes.checkPropTypes(EditorControl.propTypes, this.props, 'prop', 'EditorControl');\n  }\n\n  isAncestorOfFieldError = () => {\n    const { fieldsErrors } = this.props;\n\n    if (fieldsErrors && fieldsErrors.size > 0) {\n      return Object.values(fieldsErrors.toJS()).some(arr =>\n        arr.some(err => err.parentIds && err.parentIds.includes(this.uniqueFieldId)),\n      );\n    }\n    return false;\n  };\n\n  render() {\n    const {\n      value,\n      entry,\n      collection,\n      config,\n      field,\n      fieldsMetaData,\n      fieldsErrors,\n      mediaPaths,\n      boundGetAsset,\n      onChange,\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      addAsset,\n      removeInsertedMedia,\n      persistMedia,\n      onValidate,\n      controlRef,\n      query,\n      queryHits,\n      isFetching,\n      clearSearch,\n      clearFieldErrors,\n      loadEntry,\n      className,\n      isSelected,\n      isEditorComponent,\n      isNewEditorComponent,\n      parentIds,\n      t,\n      validateMetaField,\n      isLoadingAsset,\n      isDisabled,\n      isHidden,\n      isFieldDuplicate,\n      isFieldHidden,\n      locale,\n      isParentListCollapsed,\n    } = this.props;\n\n    const widgetName = field.get('widget');\n    const widget = resolveWidget(widgetName);\n    const fieldName = field.get('name');\n    const fieldHint = field.get('hint');\n    const isFieldOptional = field.get('required') === false;\n    const onValidateObject = onValidate;\n    const metadata = fieldsMetaData && fieldsMetaData.get(fieldName);\n    const errors = fieldsErrors && fieldsErrors.get(this.uniqueFieldId);\n    const childErrors = this.isAncestorOfFieldError();\n    const hasErrors = !!errors || childErrors;\n\n    return (\n      <ClassNames>\n        {({ css, cx }) => (\n          <ControlContainer\n            className={className}\n            css={css`\n              ${isHidden && styleStrings.hidden};\n            `}\n          >\n            <ControlTopbar>\n              {widget.globalStyles && <Global styles={coreCss`${widget.globalStyles}`} />}\n              <LabelComponent\n                field={field}\n                isActive={isSelected || this.state.styleActive}\n                hasErrors={hasErrors}\n                uniqueFieldId={this.uniqueFieldId}\n                isFieldOptional={isFieldOptional}\n                t={t}\n              />\n              {errors && (\n                <ControlErrorsList>\n                  {errors.map(\n                    error =>\n                      error.message &&\n                      typeof error.message === 'string' && (\n                        <li key={error.message.trim().replace(/[^a-z0-9]+/gi, '-')}>\n                          {error.message}\n                        </li>\n                      ),\n                  )}\n                </ControlErrorsList>\n              )}\n            </ControlTopbar>\n            <Widget\n              classNameWrapper={cx(\n                css`\n                  ${styleStrings.widget};\n                `,\n                {\n                  [css`\n                    ${styleStrings.widgetActive};\n                  `]: isSelected || this.state.styleActive,\n                },\n                {\n                  [css`\n                    ${styleStrings.widgetError};\n                  `]: hasErrors,\n                },\n                {\n                  [css`\n                    ${styleStrings.disabled}\n                  `]: isDisabled,\n                },\n              )}\n              classNameWidget={css`\n                ${styleStrings.widget};\n              `}\n              classNameWidgetActive={css`\n                ${styleStrings.widgetActive};\n              `}\n              classNameLabel={css`\n                ${styleStrings.label};\n              `}\n              classNameLabelActive={css`\n                ${styleStrings.labelActive};\n              `}\n              controlComponent={widget.control}\n              entry={entry}\n              collection={collection}\n              config={config}\n              field={field}\n              uniqueFieldId={this.uniqueFieldId}\n              value={value}\n              mediaPaths={mediaPaths}\n              metadata={metadata}\n              onChange={(newValue, newMetadata) => {\n                onChange(field, newValue, newMetadata);\n                clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля\n              }}\n              onValidate={onValidate && partial(onValidate, this.uniqueFieldId)}\n              onOpenMediaLibrary={openMediaLibrary}\n              onClearMediaControl={clearMediaControl}\n              onRemoveMediaControl={removeMediaControl}\n              onRemoveInsertedMedia={removeInsertedMedia}\n              onPersistMedia={persistMedia}\n              onAddAsset={addAsset}\n              getAsset={boundGetAsset}\n              hasActiveStyle={isSelected || this.state.styleActive}\n              setActiveStyle={() => this.setState({ styleActive: true })}\n              setInactiveStyle={() => this.setState({ styleActive: false })}\n              resolveWidget={resolveWidget}\n              widget={widget}\n              getEditorComponents={getEditorComponents}\n              controlRef={controlRef}\n              editorControl={ConnectedEditorControl}\n              query={query}\n              loadEntry={loadEntry}\n              queryHits={queryHits[this.uniqueFieldId] || []}\n              clearSearch={clearSearch}\n              clearFieldErrors={clearFieldErrors}\n              isFetching={isFetching}\n              fieldsErrors={fieldsErrors}\n              onValidateObject={onValidateObject}\n              isEditorComponent={isEditorComponent}\n              isNewEditorComponent={isNewEditorComponent}\n              parentIds={parentIds}\n              t={t}\n              validateMetaField={validateMetaField}\n              isDisabled={isDisabled}\n              isFieldDuplicate={isFieldDuplicate}\n              isFieldHidden={isFieldHidden}\n              isLoadingAsset={isLoadingAsset}\n              locale={locale}\n              isParentListCollapsed={isParentListCollapsed}\n            />\n            {fieldHint && (\n              <ControlHint active={isSelected || this.state.styleActive} error={hasErrors}>\n                <ReactMarkdown\n                  remarkPlugins={[gfm]}\n                  allowedElements={['a', 'strong', 'em', 'del']}\n                  unwrapDisallowed={true}\n                  components={{\n                    // eslint-disable-next-line no-unused-vars\n                    a: ({ node, ...props }) => (\n                      <a\n                        {...props}\n                        target=\"_blank\"\n                        rel=\"noopener noreferrer\"\n                        style={{ color: 'inherit' }}\n                      />\n                    ),\n                  }}\n                >\n                  {fieldHint}\n                </ReactMarkdown>\n              </ControlHint>\n            )}\n          </ControlContainer>\n        )}\n      </ClassNames>\n    );\n  }\n}\n\nfunction mapStateToProps(state) {\n  const { collections, entryDraft } = state;\n  const entry = entryDraft.get('entry');\n  const collection = collections.get(entryDraft.getIn(['entry', 'collection']));\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n\n  async function loadEntry(collectionName, slug) {\n    const targetCollection = collections.get(collectionName);\n    if (targetCollection) {\n      const loadedEntry = await tryLoadEntry(state, targetCollection, slug);\n      return loadedEntry;\n    } else {\n      throw new Error(`Can't find collection '${collectionName}'`);\n    }\n  }\n\n  return {\n    mediaPaths: state.mediaLibrary.get('controlMedia'),\n    isFetching: state.search.isFetching,\n    queryHits: state.search.queryHits,\n    config: state.config,\n    entry,\n    collection,\n    isLoadingAsset,\n    loadEntry,\n    validateMetaField: (field, value, t) => validateMetaField(state, collection, field, value, t),\n  };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  const creators = bindActionCreators(\n    {\n      openMediaLibrary,\n      clearMediaControl,\n      removeMediaControl,\n      removeInsertedMedia,\n      persistMedia,\n      addAsset,\n      query,\n      clearSearch,\n      clearFieldErrors,\n    },\n    dispatch,\n  );\n  return {\n    ...creators,\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry),\n  };\n}\n\nconst ConnectedEditorControl = connect(\n  mapStateToProps,\n  mapDispatchToProps,\n  mergeProps,\n)(translate()(EditorControl));\n\nexport default ConnectedEditorControl;\n"]} */"))
|
|
201
228
|
}), ___EmotionJSX(LabelComponent, {
|
|
202
229
|
field: field,
|
|
203
230
|
isActive: isSelected || this.state.styleActive,
|
|
@@ -248,7 +275,7 @@ class EditorControl extends React.Component {
|
|
|
248
275
|
onChange(field, newValue, newMetadata);
|
|
249
276
|
clearFieldErrors(this.uniqueFieldId); // Видаляємо помилки лише для цього поля
|
|
250
277
|
},
|
|
251
|
-
onValidate: onValidate &&
|
|
278
|
+
onValidate: onValidate && partial(onValidate, this.uniqueFieldId),
|
|
252
279
|
onOpenMediaLibrary: openMediaLibrary,
|
|
253
280
|
onClearMediaControl: clearMediaControl,
|
|
254
281
|
onRemoveMediaControl: removeMediaControl,
|
|
@@ -296,59 +323,20 @@ class EditorControl extends React.Component {
|
|
|
296
323
|
unwrapDisallowed: true,
|
|
297
324
|
components: {
|
|
298
325
|
// eslint-disable-next-line no-unused-vars
|
|
299
|
-
a:
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
}
|
|
310
|
-
}));
|
|
311
|
-
}
|
|
326
|
+
a: ({
|
|
327
|
+
node,
|
|
328
|
+
...props
|
|
329
|
+
}) => ___EmotionJSX("a", _extends({}, props, {
|
|
330
|
+
target: "_blank",
|
|
331
|
+
rel: "noopener noreferrer",
|
|
332
|
+
style: {
|
|
333
|
+
color: 'inherit'
|
|
334
|
+
}
|
|
335
|
+
}))
|
|
312
336
|
}
|
|
313
337
|
}, fieldHint))));
|
|
314
338
|
}
|
|
315
339
|
}
|
|
316
|
-
_defineProperty(EditorControl, "propTypes", {
|
|
317
|
-
value: PropTypes.oneOfType([PropTypes.node, PropTypes.object, PropTypes.string, PropTypes.bool]),
|
|
318
|
-
field: ImmutablePropTypes.map.isRequired,
|
|
319
|
-
fieldsMetaData: ImmutablePropTypes.map,
|
|
320
|
-
fieldsErrors: ImmutablePropTypes.map,
|
|
321
|
-
mediaPaths: ImmutablePropTypes.map.isRequired,
|
|
322
|
-
boundGetAsset: PropTypes.func.isRequired,
|
|
323
|
-
onChange: PropTypes.func.isRequired,
|
|
324
|
-
openMediaLibrary: PropTypes.func.isRequired,
|
|
325
|
-
addAsset: PropTypes.func.isRequired,
|
|
326
|
-
removeInsertedMedia: PropTypes.func.isRequired,
|
|
327
|
-
persistMedia: PropTypes.func.isRequired,
|
|
328
|
-
onValidate: PropTypes.func,
|
|
329
|
-
controlRef: PropTypes.func,
|
|
330
|
-
query: PropTypes.func.isRequired,
|
|
331
|
-
queryHits: PropTypes.object,
|
|
332
|
-
isFetching: PropTypes.bool,
|
|
333
|
-
clearSearch: PropTypes.func.isRequired,
|
|
334
|
-
clearFieldErrors: PropTypes.func.isRequired,
|
|
335
|
-
loadEntry: PropTypes.func.isRequired,
|
|
336
|
-
t: PropTypes.func.isRequired,
|
|
337
|
-
isEditorComponent: PropTypes.bool,
|
|
338
|
-
isNewEditorComponent: PropTypes.bool,
|
|
339
|
-
parentIds: PropTypes.arrayOf(PropTypes.string),
|
|
340
|
-
entry: ImmutablePropTypes.map.isRequired,
|
|
341
|
-
collection: ImmutablePropTypes.map.isRequired,
|
|
342
|
-
isDisabled: PropTypes.bool,
|
|
343
|
-
isHidden: PropTypes.bool,
|
|
344
|
-
isFieldDuplicate: PropTypes.func,
|
|
345
|
-
isFieldHidden: PropTypes.func,
|
|
346
|
-
locale: PropTypes.string,
|
|
347
|
-
isParentListCollapsed: PropTypes.bool
|
|
348
|
-
});
|
|
349
|
-
_defineProperty(EditorControl, "defaultProps", {
|
|
350
|
-
parentIds: []
|
|
351
|
-
});
|
|
352
340
|
function mapStateToProps(state) {
|
|
353
341
|
const {
|
|
354
342
|
collections,
|
|
@@ -390,14 +378,18 @@ function mapDispatchToProps(dispatch) {
|
|
|
390
378
|
clearSearch,
|
|
391
379
|
clearFieldErrors
|
|
392
380
|
}, dispatch);
|
|
393
|
-
return
|
|
381
|
+
return {
|
|
382
|
+
...creators,
|
|
394
383
|
boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry)
|
|
395
|
-
}
|
|
384
|
+
};
|
|
396
385
|
}
|
|
397
386
|
function mergeProps(stateProps, dispatchProps, ownProps) {
|
|
398
|
-
return
|
|
387
|
+
return {
|
|
388
|
+
...stateProps,
|
|
389
|
+
...dispatchProps,
|
|
390
|
+
...ownProps,
|
|
399
391
|
boundGetAsset: dispatchProps.boundGetAsset(stateProps.collection, stateProps.entry)
|
|
400
|
-
}
|
|
392
|
+
};
|
|
401
393
|
}
|
|
402
394
|
const ConnectedEditorControl = connect(mapStateToProps, mapDispatchToProps, mergeProps)(translate()(EditorControl));
|
|
403
395
|
export default ConnectedEditorControl;
|