UNPKG

@wordpress/block-library

Version:
406 lines (368 loc) 11.1 kB
/** * External dependencies */ import { filter, every } from 'lodash'; /** * WordPress dependencies */ import { createBlock } from '@wordpress/blocks'; import { createBlobURL } from '@wordpress/blob'; import { addFilter } from '@wordpress/hooks'; /** * Internal dependencies */ import { LINK_DESTINATION_ATTACHMENT, LINK_DESTINATION_NONE, LINK_DESTINATION_MEDIA } from './constants'; import { LINK_DESTINATION_ATTACHMENT as DEPRECATED_LINK_DESTINATION_ATTACHMENT, LINK_DESTINATION_MEDIA as DEPRECATED_LINK_DESTINATION_MEDIA } from './v1/constants'; import { pickRelevantMediaFiles, isGalleryV2Enabled } from './shared'; const parseShortcodeIds = ids => { if (!ids) { return []; } return ids.split(',').map(id => parseInt(id, 10)); }; /** * Third party block plugins don't have an easy way to detect if the * innerBlocks version of the Gallery is running when they run a * 3rdPartyBlock -> GalleryBlock transform so this tranform filter * will handle this. Once the innerBlocks version is the default * in a core release, this could be deprecated and removed after * plugin authors have been given time to update transforms. * * @typedef {Object} Attributes * @typedef {Object} Block * @property {Attributes} attributes The attributes of the block. * @param {Block} block The transformed block. * @return {Block} The transformed block. */ function updateThirdPartyTransformToGallery(block) { var _block$attributes; if (isGalleryV2Enabled() && block.name === 'core/gallery' && ((_block$attributes = block.attributes) === null || _block$attributes === void 0 ? void 0 : _block$attributes.images.length) > 0) { const innerBlocks = block.attributes.images.map(_ref => { let { url, id, alt } = _ref; return createBlock('core/image', { url, id: id ? parseInt(id, 10) : null, alt, sizeSlug: block.attributes.sizeSlug, linkDestination: block.attributes.linkDestination }); }); delete block.attributes.ids; delete block.attributes.images; block.innerBlocks = innerBlocks; } return block; } addFilter('blocks.switchToBlockType.transformedBlock', 'core/gallery/update-third-party-transform-to', updateThirdPartyTransformToGallery); /** * Third party block plugins don't have an easy way to detect if the * innerBlocks version of the Gallery is running when they run a * GalleryBlock -> 3rdPartyBlock transform so this transform filter * will handle this. Once the innerBlocks version is the default * in a core release, this could be deprecated and removed after * plugin authors have been given time to update transforms. * * @typedef {Object} Attributes * @typedef {Object} Block * @property {Attributes} attributes The attributes of the block. * @param {Block} toBlock The block to transform to. * @param {Block[]} fromBlocks The blocks to transform from. * @return {Block} The transformed block. */ function updateThirdPartyTransformFromGallery(toBlock, fromBlocks) { const from = Array.isArray(fromBlocks) ? fromBlocks : [fromBlocks]; const galleryBlock = from.find(transformedBlock => { var _transformedBlock$att; return transformedBlock.name === 'core/gallery' && transformedBlock.innerBlocks.length > 0 && !((_transformedBlock$att = transformedBlock.attributes.images) !== null && _transformedBlock$att !== void 0 && _transformedBlock$att.length) > 0 && !toBlock.name.includes('core/'); }); if (galleryBlock) { const images = galleryBlock.innerBlocks.map(_ref2 => { let { attributes: { url, id, alt } } = _ref2; return { url, id: id ? parseInt(id, 10) : null, alt }; }); const ids = images.map(_ref3 => { let { id } = _ref3; return id; }); galleryBlock.attributes.images = images; galleryBlock.attributes.ids = ids; } return toBlock; } addFilter('blocks.switchToBlockType.transformedBlock', 'core/gallery/update-third-party-transform-from', updateThirdPartyTransformFromGallery); const transforms = { from: [{ type: 'block', isMultiBlock: true, blocks: ['core/image'], transform: attributes => { // Init the align and size from the first item which may be either the placeholder or an image. let { align, sizeSlug } = attributes[0]; // Loop through all the images and check if they have the same align and size. align = every(attributes, ['align', align]) ? align : undefined; sizeSlug = every(attributes, ['sizeSlug', sizeSlug]) ? sizeSlug : undefined; const validImages = filter(attributes, _ref4 => { let { url } = _ref4; return url; }); if (isGalleryV2Enabled()) { const innerBlocks = validImages.map(image => { // Gallery images can't currently be resized so make sure height and width are undefined. image.width = undefined; image.height = undefined; return createBlock('core/image', image); }); return createBlock('core/gallery', { align, sizeSlug }, innerBlocks); } return createBlock('core/gallery', { images: validImages.map(_ref5 => { let { id, url, alt, caption } = _ref5; return { id: id.toString(), url, alt, caption }; }), ids: validImages.map(_ref6 => { let { id } = _ref6; return parseInt(id, 10); }), align, sizeSlug }); } }, { type: 'shortcode', tag: 'gallery', attributes: { images: { type: 'array', shortcode: _ref7 => { let { named: { ids } } = _ref7; if (!isGalleryV2Enabled()) { return parseShortcodeIds(ids).map(id => ({ id: id.toString() })); } } }, ids: { type: 'array', shortcode: _ref8 => { let { named: { ids } } = _ref8; if (!isGalleryV2Enabled()) { return parseShortcodeIds(ids); } } }, columns: { type: 'number', shortcode: _ref9 => { let { named: { columns = '3' } } = _ref9; return parseInt(columns, 10); } }, linkTo: { type: 'string', shortcode: _ref10 => { let { named: { link } } = _ref10; if (!isGalleryV2Enabled()) { switch (link) { case 'post': return DEPRECATED_LINK_DESTINATION_ATTACHMENT; case 'file': return DEPRECATED_LINK_DESTINATION_MEDIA; default: return DEPRECATED_LINK_DESTINATION_ATTACHMENT; } } switch (link) { case 'post': return LINK_DESTINATION_ATTACHMENT; case 'file': return LINK_DESTINATION_MEDIA; default: return LINK_DESTINATION_NONE; } } } }, transform(_ref11) { let { named: { ids, columns = 3, link } } = _ref11; const imageIds = parseShortcodeIds(ids).map(id => parseInt(id, 10)); let linkTo = LINK_DESTINATION_NONE; if (link === 'post') { linkTo = LINK_DESTINATION_ATTACHMENT; } else if (link === 'file') { linkTo = LINK_DESTINATION_MEDIA; } const galleryBlock = createBlock('core/gallery', { columns: parseInt(columns, 10), linkTo }, imageIds.map(imageId => createBlock('core/image', { id: imageId }))); return galleryBlock; }, isMatch(_ref12) { let { named } = _ref12; return undefined !== named.ids; } }, { // When created by drag and dropping multiple files on an insertion point. Because multiple // files must not be transformed to a gallery when dropped within a gallery there is another transform // within the image block to handle that case. Therefore this transform has to have priority 1 // set so that it overrrides the image block transformation when mulitple images are dropped outside // of a gallery block. type: 'files', priority: 1, isMatch(files) { return files.length !== 1 && every(files, file => file.type.indexOf('image/') === 0); }, transform(files) { if (isGalleryV2Enabled()) { const innerBlocks = files.map(file => createBlock('core/image', { url: createBlobURL(file) })); return createBlock('core/gallery', {}, innerBlocks); } const block = createBlock('core/gallery', { images: files.map(file => pickRelevantMediaFiles({ url: createBlobURL(file) })) }); return block; } }], to: [{ type: 'block', blocks: ['core/image'], transform: (_ref13, innerBlocks) => { let { align, images, ids, sizeSlug } = _ref13; if (isGalleryV2Enabled()) { if (innerBlocks.length > 0) { return innerBlocks.map(_ref14 => { let { attributes: { url, alt, caption, title, href, rel, linkClass, id, sizeSlug: imageSizeSlug, linkDestination, linkTarget, anchor, className } } = _ref14; return createBlock('core/image', { align, url, alt, caption, title, href, rel, linkClass, id, sizeSlug: imageSizeSlug, linkDestination, linkTarget, anchor, className }); }); } return createBlock('core/image', { align }); } if (images.length > 0) { return images.map((_ref15, index) => { let { url, alt, caption } = _ref15; return createBlock('core/image', { id: ids[index], url, alt, caption, align, sizeSlug }); }); } return createBlock('core/image', { align }); } }] }; export default transforms; //# sourceMappingURL=transforms.js.map