"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = withFileControl; var _base = _interopRequireDefault(require("@emotion/styled/base")); var _once2 = _interopRequireDefault(require("lodash/once")); var _react = _interopRequireDefault(require("react")); var _propTypes = _interopRequireDefault(require("prop-types")); var _reactImmutableProptypes = _interopRequireDefault(require("react-immutable-proptypes")); var _react2 = require("@emotion/react"); var _immutable = require("immutable"); var _uuid = require("uuid"); var _commonTags = require("common-tags"); var _decapCmsUiDefault = require("decap-cms-ui-default"); var _decapCmsLibUtil = require("decap-cms-lib-util"); var _arrayMove = require("array-move"); var _core = require("@dnd-kit/core"); var _sortable = require("@dnd-kit/sortable"); var _utilities = require("@dnd-kit/utilities"); var _modifiers = require("@dnd-kit/modifiers"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } 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; } function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : String(i); } 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); } 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); } 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)."; } const MAX_DISPLAY_LENGTH = 50; const ImageWrapper = /*#__PURE__*/(0, _base.default)("div", { target: "e1hax4ql7", label: "ImageWrapper" })("flex-basis:155px;width:155px;height:100px;margin-right:20px;margin-bottom:20px;border:", _decapCmsUiDefault.borders.textField, ";border-radius:", _decapCmsUiDefault.lengths.borderRadius, ";overflow:hidden;", _decapCmsUiDefault.effects.checkerboard, ";", _decapCmsUiDefault.shadows.inset, ";cursor:", props => props.sortable ? 'pointer' : 'auto', ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AAkC+B","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */")); const SortableImageButtonsWrapper = /*#__PURE__*/(0, _base.default)("div", { target: "e1hax4ql6", label: "SortableImageButtonsWrapper" })(process.env.NODE_ENV === "production" ? { name: "1ekgmip", styles: "display:flex;justify-content:center;column-gap:10px;margin-right:20px;margin-top:-10px;margin-bottom:10px" } : { name: "1ekgmip", styles: "display:flex;justify-content:center;column-gap:10px;margin-right:20px;margin-top:-10px;margin-bottom:10px", map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AAgD8C","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */", toString: _EMOTION_STRINGIFIED_CSS_ERROR__ }); const StyledImage = /*#__PURE__*/(0, _base.default)("img", { target: "e1hax4ql5", label: "StyledImage" })(process.env.NODE_ENV === "production" ? { name: "ukfjzf", styles: "width:100%;height:100%;object-fit:contain" } : { name: "ukfjzf", styles: "width:100%;height:100%;object-fit:contain", map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AAyD8B","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */", toString: _EMOTION_STRINGIFIED_CSS_ERROR__ }); function Image(props) { return (0, _react2.jsx)(StyledImage, _extends({ role: "presentation" }, props)); } function SortableImageButtons({ onRemove, onReplace }) { return (0, _react2.jsx)(SortableImageButtonsWrapper, null, (0, _react2.jsx)(_decapCmsUiDefault.IconButton, { size: "small", type: "media", onClick: onReplace }), (0, _react2.jsx)(_decapCmsUiDefault.IconButton, { size: "small", type: "close", onClick: onRemove })); } function SortableImage(props) { const { attributes, listeners, setNodeRef, transform, transition } = (0, _sortable.useSortable)({ id: props.id }); const style = { transform: _utilities.CSS.Transform.toString(transform), transition }; const { itemValue, getAsset, field, onRemove, onReplace } = props; return (0, _react2.jsx)("div", _extends({ ref: setNodeRef, style: style }, attributes, listeners), (0, _react2.jsx)(ImageWrapper, { sortable: true }, (0, _react2.jsx)(Image, { src: getAsset(itemValue, field) || '' })), (0, _react2.jsx)(SortableImageButtons, { item: itemValue, onRemove: onRemove, onReplace: onReplace })); } var _ref = process.env.NODE_ENV === "production" ? { name: "a42x49-SortableMultiImageWrapper", styles: "display:flex;flex-wrap:wrap;label:SortableMultiImageWrapper;" } : { name: "a42x49-SortableMultiImageWrapper", styles: "display:flex;flex-wrap:wrap;label:SortableMultiImageWrapper;", map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AA8Hc","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */", toString: _EMOTION_STRINGIFIED_CSS_ERROR__ }; function SortableMultiImageWrapper({ items, getAsset, field, onSortEnd, onRemoveOne, onReplaceOne }) { const activationConstraint = { distance: 4 }; const sensors = (0, _core.useSensors)((0, _core.useSensor)(_core.MouseSensor, { activationConstraint }), (0, _core.useSensor)(_core.TouchSensor, { activationConstraint })); function handleSortEnd({ active, over }) { onSortEnd({ oldIndex: items.findIndex(item => item.id === active.id), newIndex: items.findIndex(item => item.id === over.id) }); } return (0, _react2.jsx)("div", { // eslint-disable-next-line react/no-unknown-property css: _ref }, (0, _react2.jsx)(_core.DndContext, { modifiers: [_modifiers.restrictToParentElement], collisionDetection: _core.closestCenter, sensors: sensors, onDragEnd: handleSortEnd }, (0, _react2.jsx)(_sortable.SortableContext, { items: items }, items.map((item, index) => (0, _react2.jsx)(SortableImage, { key: item.id, id: item.id, index: index, itemValue: item.value, getAsset: getAsset, field: field, onRemove: onRemoveOne(index), onReplace: onReplaceOne(index) }))))); } const FileLink = /*#__PURE__*/(0, _base.default)("a", { target: "e1hax4ql4", label: "FileLink" })(process.env.NODE_ENV === "production" ? { name: "7mbjrw", styles: "margin-bottom:20px;font-weight:normal;color:inherit;&:hover,&:active,&:focus{text-decoration:underline;}" } : { name: "7mbjrw", styles: "margin-bottom:20px;font-weight:normal;color:inherit;&:hover,&:active,&:focus{text-decoration:underline;}", map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AA4JyB","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */", toString: _EMOTION_STRINGIFIED_CSS_ERROR__ }); const FileLinks = /*#__PURE__*/(0, _base.default)("div", { target: "e1hax4ql3", label: "FileLinks" })(process.env.NODE_ENV === "production" ? { name: "cn3xcj", styles: "margin-bottom:12px" } : { name: "cn3xcj", styles: "margin-bottom:12px", map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AAwK4B","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */", toString: _EMOTION_STRINGIFIED_CSS_ERROR__ }); const FileLinkList = /*#__PURE__*/(0, _base.default)("ul", { target: "e1hax4ql2", label: "FileLinkList" })(process.env.NODE_ENV === "production" ? { name: "ffhm6p", styles: "list-style-type:none" } : { name: "ffhm6p", styles: "list-style-type:none", map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AA4K8B","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */", toString: _EMOTION_STRINGIFIED_CSS_ERROR__ }); const FileWidgetButton = /*#__PURE__*/(0, _base.default)("button", { target: "e1hax4ql1", label: "FileWidgetButton" })(_decapCmsUiDefault.buttons.button, ";", _decapCmsUiDefault.components.badge, ";margin-bottom:12px;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AAgLsC","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */")); const FileWidgetButtonRemove = /*#__PURE__*/(0, _base.default)("button", { target: "e1hax4ql0", label: "FileWidgetButtonRemove" })(_decapCmsUiDefault.buttons.button, ";", _decapCmsUiDefault.components.badgeDanger, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/withFileControl.js"],"names":[],"mappings":"AAsL4C","file":"../../src/withFileControl.js","sourcesContent":["import React from 'react';\nimport PropTypes from 'prop-types';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport styled from '@emotion/styled';\nimport { css } from '@emotion/react';\nimport { Map, List } from 'immutable';\nimport { once } from 'lodash';\nimport { v4 as uuid } from 'uuid';\nimport { oneLine } from 'common-tags';\nimport {\n  lengths,\n  components,\n  buttons,\n  borders,\n  effects,\n  shadows,\n  IconButton,\n} from 'decap-cms-ui-default';\nimport { basename } from 'decap-cms-lib-util';\nimport { arrayMoveImmutable as arrayMove } from 'array-move';\nimport {\n  DndContext,\n  MouseSensor,\n  TouchSensor,\n  closestCenter,\n  useSensor,\n  useSensors,\n} from '@dnd-kit/core';\nimport { SortableContext, useSortable } from '@dnd-kit/sortable';\nimport { CSS } from '@dnd-kit/utilities';\nimport { restrictToParentElement } from '@dnd-kit/modifiers';\n\nconst MAX_DISPLAY_LENGTH = 50;\n\nconst ImageWrapper = styled.div`\n  flex-basis: 155px;\n  width: 155px;\n  height: 100px;\n  margin-right: 20px;\n  margin-bottom: 20px;\n  border: ${borders.textField};\n  border-radius: ${lengths.borderRadius};\n  overflow: hidden;\n  ${effects.checkerboard};\n  ${shadows.inset};\n  cursor: ${props => (props.sortable ? 'pointer' : 'auto')};\n`;\n\nconst SortableImageButtonsWrapper = styled.div`\n  display: flex;\n  justify-content: center;\n  column-gap: 10px;\n  margin-right: 20px;\n  margin-top: -10px;\n  margin-bottom: 10px;\n`;\n\nconst StyledImage = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: contain;\n`;\n\nfunction Image(props) {\n  return <StyledImage role=\"presentation\" {...props} />;\n}\n\nfunction SortableImageButtons({ onRemove, onReplace }) {\n  return (\n    <SortableImageButtonsWrapper>\n      <IconButton size=\"small\" type=\"media\" onClick={onReplace}></IconButton>\n      <IconButton size=\"small\" type=\"close\" onClick={onRemove}></IconButton>\n    </SortableImageButtonsWrapper>\n  );\n}\n\nfunction SortableImage(props) {\n  const { attributes, listeners, setNodeRef, transform, transition } = useSortable({\n    id: props.id,\n  });\n\n  const style = {\n    transform: CSS.Transform.toString(transform),\n    transition,\n  };\n\n  const { itemValue, getAsset, field, onRemove, onReplace } = props;\n\n  return (\n    <div ref={setNodeRef} style={style} {...attributes} {...listeners}>\n      <ImageWrapper sortable>\n        <Image src={getAsset(itemValue, field) || ''} />\n      </ImageWrapper>\n      <SortableImageButtons\n        item={itemValue}\n        onRemove={onRemove}\n        onReplace={onReplace}\n      ></SortableImageButtons>\n    </div>\n  );\n}\n\nfunction SortableMultiImageWrapper({\n  items,\n  getAsset,\n  field,\n  onSortEnd,\n  onRemoveOne,\n  onReplaceOne,\n}) {\n  const activationConstraint = { distance: 4 };\n  const sensors = useSensors(\n    useSensor(MouseSensor, { activationConstraint }),\n    useSensor(TouchSensor, { activationConstraint }),\n  );\n\n  function handleSortEnd({ active, over }) {\n    onSortEnd({\n      oldIndex: items.findIndex(item => item.id === active.id),\n      newIndex: items.findIndex(item => item.id === over.id),\n    });\n  }\n\n  return (\n    <div\n      // eslint-disable-next-line react/no-unknown-property\n      css={css`\n        display: flex;\n        flex-wrap: wrap;\n      `}\n    >\n      <DndContext\n        modifiers={[restrictToParentElement]}\n        collisionDetection={closestCenter}\n        sensors={sensors}\n        onDragEnd={handleSortEnd}\n      >\n        <SortableContext items={items}>\n          {items.map((item, index) => (\n            <SortableImage\n              key={item.id}\n              id={item.id}\n              index={index}\n              itemValue={item.value}\n              getAsset={getAsset}\n              field={field}\n              onRemove={onRemoveOne(index)}\n              onReplace={onReplaceOne(index)}\n            ></SortableImage>\n          ))}\n        </SortableContext>\n      </DndContext>\n    </div>\n  );\n}\n\nconst FileLink = styled.a`\n  margin-bottom: 20px;\n  font-weight: normal;\n  color: inherit;\n\n  &:hover,\n  &:active,\n  &:focus {\n    text-decoration: underline;\n  }\n`;\n\nconst FileLinks = styled.div`\n  margin-bottom: 12px;\n`;\n\nconst FileLinkList = styled.ul`\n  list-style-type: none;\n`;\n\nconst FileWidgetButton = styled.button`\n  ${buttons.button};\n  ${components.badge};\n  margin-bottom: 12px;\n`;\n\nconst FileWidgetButtonRemove = styled.button`\n  ${buttons.button};\n  ${components.badgeDanger};\n`;\n\nfunction isMultiple(value) {\n  return Array.isArray(value) || List.isList(value);\n}\n\nfunction sizeOfValue(value) {\n  if (Array.isArray(value)) {\n    return value.length;\n  }\n\n  if (List.isList(value)) {\n    return value.size;\n  }\n\n  return value ? 1 : 0;\n}\n\nfunction valueListToArray(value) {\n  return List.isList(value) ? value.toArray() : value ?? '';\n}\n\nfunction valueListToSortableArray(value) {\n  if (!isMultiple(value)) {\n    return value;\n  }\n\n  const valueArray = valueListToArray(value).map(value => ({\n    id: uuid(),\n    value,\n  }));\n\n  return valueArray;\n}\n\nconst warnDeprecatedOptions = once(field =>\n  console.warn(oneLine`\n  Decap CMS config: ${field.get('name')} field: property \"options\" has been deprecated for the\n  ${field.get('widget')} widget and will be removed in the next major release. Rather than\n  \\`field.options.media_library\\`, apply media library options for this widget under\n  \\`field.media_library\\`.\n`),\n);\n\nexport default function withFileControl({ forImage } = {}) {\n  return class FileControl extends React.Component {\n    static propTypes = {\n      field: PropTypes.object.isRequired,\n      getAsset: PropTypes.func.isRequired,\n      mediaPaths: ImmutablePropTypes.map.isRequired,\n      onAddAsset: PropTypes.func.isRequired,\n      onChange: PropTypes.func.isRequired,\n      onRemoveInsertedMedia: PropTypes.func.isRequired,\n      onOpenMediaLibrary: PropTypes.func.isRequired,\n      onClearMediaControl: PropTypes.func.isRequired,\n      onRemoveMediaControl: PropTypes.func.isRequired,\n      classNameWrapper: PropTypes.string.isRequired,\n      value: PropTypes.oneOfType([\n        PropTypes.string,\n        PropTypes.arrayOf(PropTypes.string),\n        ImmutablePropTypes.listOf(PropTypes.string),\n      ]),\n      t: PropTypes.func.isRequired,\n    };\n\n    static defaultProps = {\n      value: '',\n    };\n\n    constructor(props) {\n      super(props);\n      this.controlID = uuid();\n    }\n\n    shouldComponentUpdate(nextProps) {\n      /**\n       * Always update if the value or getAsset changes.\n       */\n      if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) {\n        return true;\n      }\n\n      /**\n       * If there is a media path for this control in the state object, and that\n       * path is different than the value in `nextProps`, update.\n       */\n      const mediaPath = nextProps.mediaPaths.get(this.controlID);\n      if (mediaPath && nextProps.value !== mediaPath) {\n        return true;\n      }\n\n      return false;\n    }\n\n    componentDidUpdate() {\n      const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props;\n      const mediaPath = mediaPaths.get(this.controlID);\n      if (mediaPath && mediaPath !== value) {\n        onChange(mediaPath);\n      } else if (mediaPath && mediaPath === value) {\n        onRemoveInsertedMedia(this.controlID);\n      }\n    }\n\n    componentWillUnmount() {\n      this.props.onRemoveMediaControl(this.controlID);\n    }\n\n    handleChange = e => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      e.preventDefault();\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true),\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    handleUrl = subject => e => {\n      e.preventDefault();\n\n      const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`));\n\n      if (url) {\n        return this.props.onChange(url);\n      }\n    };\n\n    handleRemove = e => {\n      e.preventDefault();\n      this.props.onClearMediaControl(this.controlID);\n      return this.props.onChange('');\n    };\n\n    onRemoveOne = index => () => {\n      const value = valueListToArray(this.props.value);\n      value.splice(index, 1);\n      return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null);\n    };\n\n    onReplaceOne = index => () => {\n      const { field, onOpenMediaLibrary, value } = this.props;\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n\n      return onOpenMediaLibrary({\n        controlID: this.controlID,\n        forImage,\n        privateUpload: field.get('private'),\n        value: valueListToArray(value),\n        replaceIndex: index,\n        allowMultiple: false,\n        config: mediaLibraryFieldOptions.get('config'),\n        field,\n      });\n    };\n\n    getMediaLibraryFieldOptions = () => {\n      const { field } = this.props;\n\n      if (field.hasIn(['options', 'media_library'])) {\n        warnDeprecatedOptions(field);\n        return field.getIn(['options', 'media_library'], Map());\n      }\n\n      return field.get('media_library', Map());\n    };\n\n    allowsMultiple = () => {\n      const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions();\n      return (\n        mediaLibraryFieldOptions.get('config', false) &&\n        mediaLibraryFieldOptions.get('config').get('multiple', false)\n      );\n    };\n\n    onSortEnd = ({ oldIndex, newIndex }) => {\n      const { value } = this.props;\n      const newValue = arrayMove(value, oldIndex, newIndex);\n      return this.props.onChange(newValue);\n    };\n\n    getValidateValue = () => {\n      const { value } = this.props;\n      if (value) {\n        return isMultiple(value) ? value.map(v => basename(v)) : basename(value);\n      }\n\n      return value;\n    };\n\n    renderFileLink = value => {\n      const size = MAX_DISPLAY_LENGTH;\n      if (!value || value.length <= size) {\n        return value;\n      }\n      const text = `${value.slice(0, size / 2)}\\u2026${value.slice(-(size / 2) + 1)}`;\n      return (\n        <FileLink href={value} rel=\"noopener\" target=\"_blank\">\n          {text}\n        </FileLink>\n      );\n    };\n\n    renderFileLinks = () => {\n      const { value } = this.props;\n\n      if (isMultiple(value)) {\n        return (\n          <FileLinks>\n            <FileLinkList>\n              {value.map(val => (\n                <li key={val}>{this.renderFileLink(val)}</li>\n              ))}\n            </FileLinkList>\n          </FileLinks>\n        );\n      }\n      return <FileLinks>{this.renderFileLink(value)}</FileLinks>;\n    };\n\n    renderImages = () => {\n      const { getAsset, value, field } = this.props;\n      const items = valueListToSortableArray(value);\n      if (isMultiple(value)) {\n        return (\n          <SortableMultiImageWrapper\n            items={items}\n            onSortEnd={this.onSortEnd}\n            onRemoveOne={this.onRemoveOne}\n            onReplaceOne={this.onReplaceOne}\n            distance={4}\n            getAsset={getAsset}\n            field={field}\n            axis=\"xy\"\n            lockToContainerEdges={true}\n          ></SortableMultiImageWrapper>\n        );\n      }\n\n      const src = getAsset(value, field);\n      return (\n        <ImageWrapper>\n          <Image src={src || ''} />\n        </ImageWrapper>\n      );\n    };\n\n    renderSelection = subject => {\n      const { t, field } = this.props;\n      const allowsMultiple = this.allowsMultiple();\n      return (\n        <div>\n          {forImage ? this.renderImages() : null}\n          <div>\n            {forImage ? null : this.renderFileLinks()}\n            <FileWidgetButton onClick={this.handleChange}>\n              {t(\n                `editor.editorWidgets.${subject}.${\n                  this.allowsMultiple() ? 'addMore' : 'chooseDifferent'\n                }`,\n              )}\n            </FileWidgetButton>\n            {field.get('choose_url', true) && !this.allowsMultiple() ? (\n              <FileWidgetButton onClick={this.handleUrl(subject)}>\n                {t(`editor.editorWidgets.${subject}.replaceUrl`)}\n              </FileWidgetButton>\n            ) : null}\n            <FileWidgetButtonRemove onClick={this.handleRemove}>\n              {t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)}\n            </FileWidgetButtonRemove>\n          </div>\n        </div>\n      );\n    };\n\n    renderNoSelection = subject => {\n      const { t, field } = this.props;\n      return (\n        <>\n          <FileWidgetButton onClick={this.handleChange}>\n            {t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)}\n          </FileWidgetButton>\n          {field.get('choose_url', true) ? (\n            <FileWidgetButton onClick={this.handleUrl(subject)}>\n              {t(`editor.editorWidgets.${subject}.chooseUrl`)}\n            </FileWidgetButton>\n          ) : null}\n        </>\n      );\n    };\n\n    render() {\n      const { value, classNameWrapper } = this.props;\n      const subject = forImage ? 'image' : 'file';\n\n      return (\n        <div className={classNameWrapper}>\n          <span>{value ? this.renderSelection(subject) : this.renderNoSelection(subject)}</span>\n        </div>\n      );\n    }\n  };\n}\n"]} */")); function isMultiple(value) { return Array.isArray(value) || _immutable.List.isList(value); } function sizeOfValue(value) { if (Array.isArray(value)) { return value.length; } if (_immutable.List.isList(value)) { return value.size; } return value ? 1 : 0; } function valueListToArray(value) { return _immutable.List.isList(value) ? value.toArray() : value !== null && value !== void 0 ? value : ''; } function valueListToSortableArray(value) { if (!isMultiple(value)) { return value; } const valueArray = valueListToArray(value).map(value => ({ id: (0, _uuid.v4)(), value })); return valueArray; } const warnDeprecatedOptions = (0, _once2.default)(field => console.warn((0, _commonTags.oneLine)` Decap CMS config: ${field.get('name')} field: property "options" has been deprecated for the ${field.get('widget')} widget and will be removed in the next major release. Rather than \`field.options.media_library\`, apply media library options for this widget under \`field.media_library\`. `)); function withFileControl({ forImage } = {}) { var _class; return _class = class FileControl extends _react.default.Component { constructor(props) { super(props); _defineProperty(this, "handleChange", e => { const { field, onOpenMediaLibrary, value } = this.props; e.preventDefault(); const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions(); return onOpenMediaLibrary({ controlID: this.controlID, forImage, privateUpload: field.get('private'), value: valueListToArray(value), allowMultiple: !!mediaLibraryFieldOptions.get('allow_multiple', true), config: mediaLibraryFieldOptions.get('config'), field }); }); _defineProperty(this, "handleUrl", subject => e => { e.preventDefault(); const url = window.prompt(this.props.t(`editor.editorWidgets.${subject}.promptUrl`)); if (url) { return this.props.onChange(url); } }); _defineProperty(this, "handleRemove", e => { e.preventDefault(); this.props.onClearMediaControl(this.controlID); return this.props.onChange(''); }); _defineProperty(this, "onRemoveOne", index => () => { const value = valueListToArray(this.props.value); value.splice(index, 1); return this.props.onChange(sizeOfValue(value) > 0 ? [...value] : null); }); _defineProperty(this, "onReplaceOne", index => () => { const { field, onOpenMediaLibrary, value } = this.props; const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions(); return onOpenMediaLibrary({ controlID: this.controlID, forImage, privateUpload: field.get('private'), value: valueListToArray(value), replaceIndex: index, allowMultiple: false, config: mediaLibraryFieldOptions.get('config'), field }); }); _defineProperty(this, "getMediaLibraryFieldOptions", () => { const { field } = this.props; if (field.hasIn(['options', 'media_library'])) { warnDeprecatedOptions(field); return field.getIn(['options', 'media_library'], (0, _immutable.Map)()); } return field.get('media_library', (0, _immutable.Map)()); }); _defineProperty(this, "allowsMultiple", () => { const mediaLibraryFieldOptions = this.getMediaLibraryFieldOptions(); return mediaLibraryFieldOptions.get('config', false) && mediaLibraryFieldOptions.get('config').get('multiple', false); }); _defineProperty(this, "onSortEnd", ({ oldIndex, newIndex }) => { const { value } = this.props; const newValue = (0, _arrayMove.arrayMoveImmutable)(value, oldIndex, newIndex); return this.props.onChange(newValue); }); _defineProperty(this, "getValidateValue", () => { const { value } = this.props; if (value) { return isMultiple(value) ? value.map(v => (0, _decapCmsLibUtil.basename)(v)) : (0, _decapCmsLibUtil.basename)(value); } return value; }); _defineProperty(this, "renderFileLink", value => { const size = MAX_DISPLAY_LENGTH; if (!value || value.length <= size) { return value; } const text = `${value.slice(0, size / 2)}\u2026${value.slice(-(size / 2) + 1)}`; return (0, _react2.jsx)(FileLink, { href: value, rel: "noopener", target: "_blank" }, text); }); _defineProperty(this, "renderFileLinks", () => { const { value } = this.props; if (isMultiple(value)) { return (0, _react2.jsx)(FileLinks, null, (0, _react2.jsx)(FileLinkList, null, value.map(val => (0, _react2.jsx)("li", { key: val }, this.renderFileLink(val))))); } return (0, _react2.jsx)(FileLinks, null, this.renderFileLink(value)); }); _defineProperty(this, "renderImages", () => { const { getAsset, value, field } = this.props; const items = valueListToSortableArray(value); if (isMultiple(value)) { return (0, _react2.jsx)(SortableMultiImageWrapper, { items: items, onSortEnd: this.onSortEnd, onRemoveOne: this.onRemoveOne, onReplaceOne: this.onReplaceOne, distance: 4, getAsset: getAsset, field: field, axis: "xy", lockToContainerEdges: true }); } const src = getAsset(value, field); return (0, _react2.jsx)(ImageWrapper, null, (0, _react2.jsx)(Image, { src: src || '' })); }); _defineProperty(this, "renderSelection", subject => { const { t, field } = this.props; const allowsMultiple = this.allowsMultiple(); return (0, _react2.jsx)("div", null, forImage ? this.renderImages() : null, (0, _react2.jsx)("div", null, forImage ? null : this.renderFileLinks(), (0, _react2.jsx)(FileWidgetButton, { onClick: this.handleChange }, t(`editor.editorWidgets.${subject}.${this.allowsMultiple() ? 'addMore' : 'chooseDifferent'}`)), field.get('choose_url', true) && !this.allowsMultiple() ? (0, _react2.jsx)(FileWidgetButton, { onClick: this.handleUrl(subject) }, t(`editor.editorWidgets.${subject}.replaceUrl`)) : null, (0, _react2.jsx)(FileWidgetButtonRemove, { onClick: this.handleRemove }, t(`editor.editorWidgets.${subject}.remove${allowsMultiple ? 'All' : ''}`)))); }); _defineProperty(this, "renderNoSelection", subject => { const { t, field } = this.props; return (0, _react2.jsx)(_react.default.Fragment, null, (0, _react2.jsx)(FileWidgetButton, { onClick: this.handleChange }, t(`editor.editorWidgets.${subject}.choose${this.allowsMultiple() ? 'Multiple' : ''}`)), field.get('choose_url', true) ? (0, _react2.jsx)(FileWidgetButton, { onClick: this.handleUrl(subject) }, t(`editor.editorWidgets.${subject}.chooseUrl`)) : null); }); this.controlID = (0, _uuid.v4)(); } shouldComponentUpdate(nextProps) { /** * Always update if the value or getAsset changes. */ if (this.props.value !== nextProps.value || this.props.getAsset !== nextProps.getAsset) { return true; } /** * If there is a media path for this control in the state object, and that * path is different than the value in `nextProps`, update. */ const mediaPath = nextProps.mediaPaths.get(this.controlID); if (mediaPath && nextProps.value !== mediaPath) { return true; } return false; } componentDidUpdate() { const { mediaPaths, value, onRemoveInsertedMedia, onChange } = this.props; const mediaPath = mediaPaths.get(this.controlID); if (mediaPath && mediaPath !== value) { onChange(mediaPath); } else if (mediaPath && mediaPath === value) { onRemoveInsertedMedia(this.controlID); } } componentWillUnmount() { this.props.onRemoveMediaControl(this.controlID); } render() { const { value, classNameWrapper } = this.props; const subject = forImage ? 'image' : 'file'; return (0, _react2.jsx)("div", { className: classNameWrapper }, (0, _react2.jsx)("span", null, value ? this.renderSelection(subject) : this.renderNoSelection(subject))); } }, _defineProperty(_class, "propTypes", { field: _propTypes.default.object.isRequired, getAsset: _propTypes.default.func.isRequired, mediaPaths: _reactImmutableProptypes.default.map.isRequired, onAddAsset: _propTypes.default.func.isRequired, onChange: _propTypes.default.func.isRequired, onRemoveInsertedMedia: _propTypes.default.func.isRequired, onOpenMediaLibrary: _propTypes.default.func.isRequired, onClearMediaControl: _propTypes.default.func.isRequired, onRemoveMediaControl: _propTypes.default.func.isRequired, classNameWrapper: _propTypes.default.string.isRequired, value: _propTypes.default.oneOfType([_propTypes.default.string, _propTypes.default.arrayOf(_propTypes.default.string), _reactImmutableProptypes.default.listOf(_propTypes.default.string)]), t: _propTypes.default.func.isRequired }), _defineProperty(_class, "defaultProps", { value: '' }), _class; }