All checks were successful
Publish To Prod / deploy_and_publish (push) Successful in 35s
470 lines
182 KiB
JavaScript
470 lines
182 KiB
JavaScript
"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;
|
|
} |