wp/wp-includes/js/dist/script-modules/block-library/image/view.js
changeset 22 8c2e4d02f4ef
equal deleted inserted replaced
21:48c4eec2b7e6 22:8c2e4d02f4ef
       
     1 import * as __WEBPACK_EXTERNAL_MODULE__wordpress_interactivity_8e89b257__ from "@wordpress/interactivity";
       
     2 /******/ // The require scope
       
     3 /******/ var __webpack_require__ = {};
       
     4 /******/ 
       
     5 /************************************************************************/
       
     6 /******/ /* webpack/runtime/define property getters */
       
     7 /******/ (() => {
       
     8 /******/ 	// define getter functions for harmony exports
       
     9 /******/ 	__webpack_require__.d = (exports, definition) => {
       
    10 /******/ 		for(var key in definition) {
       
    11 /******/ 			if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {
       
    12 /******/ 				Object.defineProperty(exports, key, { enumerable: true, get: definition[key] });
       
    13 /******/ 			}
       
    14 /******/ 		}
       
    15 /******/ 	};
       
    16 /******/ })();
       
    17 /******/ 
       
    18 /******/ /* webpack/runtime/hasOwnProperty shorthand */
       
    19 /******/ (() => {
       
    20 /******/ 	__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))
       
    21 /******/ })();
       
    22 /******/ 
       
    23 /************************************************************************/
       
    24 var __webpack_exports__ = {};
       
    25 
       
    26 ;// external "@wordpress/interactivity"
       
    27 var x = (y) => {
       
    28 	var x = {}; __webpack_require__.d(x, y); return x
       
    29 } 
       
    30 var y = (x) => (() => (x))
       
    31 const interactivity_namespaceObject = x({ ["getContext"]: () => (__WEBPACK_EXTERNAL_MODULE__wordpress_interactivity_8e89b257__.getContext), ["getElement"]: () => (__WEBPACK_EXTERNAL_MODULE__wordpress_interactivity_8e89b257__.getElement), ["store"]: () => (__WEBPACK_EXTERNAL_MODULE__wordpress_interactivity_8e89b257__.store), ["withSyncEvent"]: () => (__WEBPACK_EXTERNAL_MODULE__wordpress_interactivity_8e89b257__.withSyncEvent) });
       
    32 ;// ./node_modules/@wordpress/block-library/build-module/image/view.js
       
    33 /**
       
    34  * WordPress dependencies
       
    35  */
       
    36 
       
    37 
       
    38 /**
       
    39  * Tracks whether user is touching screen; used to differentiate behavior for
       
    40  * touch and mouse input.
       
    41  *
       
    42  * @type {boolean}
       
    43  */
       
    44 let isTouching = false;
       
    45 
       
    46 /**
       
    47  * Tracks the last time the screen was touched; used to differentiate behavior
       
    48  * for touch and mouse input.
       
    49  *
       
    50  * @type {number}
       
    51  */
       
    52 let lastTouchTime = 0;
       
    53 const {
       
    54   state,
       
    55   actions,
       
    56   callbacks
       
    57 } = (0,interactivity_namespaceObject.store)('core/image', {
       
    58   state: {
       
    59     currentImageId: null,
       
    60     get currentImage() {
       
    61       return state.metadata[state.currentImageId];
       
    62     },
       
    63     get overlayOpened() {
       
    64       return state.currentImageId !== null;
       
    65     },
       
    66     get roleAttribute() {
       
    67       return state.overlayOpened ? 'dialog' : null;
       
    68     },
       
    69     get ariaModal() {
       
    70       return state.overlayOpened ? 'true' : null;
       
    71     },
       
    72     get enlargedSrc() {
       
    73       return state.currentImage.uploadedSrc || '';
       
    74     },
       
    75     get figureStyles() {
       
    76       return state.overlayOpened && `${state.currentImage.figureStyles?.replace(/margin[^;]*;?/g, '')};`;
       
    77     },
       
    78     get imgStyles() {
       
    79       return state.overlayOpened && `${state.currentImage.imgStyles?.replace(/;$/, '')}; object-fit:cover;`;
       
    80     },
       
    81     get imageButtonRight() {
       
    82       const {
       
    83         imageId
       
    84       } = (0,interactivity_namespaceObject.getContext)();
       
    85       return state.metadata[imageId].imageButtonRight;
       
    86     },
       
    87     get imageButtonTop() {
       
    88       const {
       
    89         imageId
       
    90       } = (0,interactivity_namespaceObject.getContext)();
       
    91       return state.metadata[imageId].imageButtonTop;
       
    92     },
       
    93     get isContentHidden() {
       
    94       const ctx = (0,interactivity_namespaceObject.getContext)();
       
    95       return state.overlayEnabled && state.currentImageId === ctx.imageId;
       
    96     },
       
    97     get isContentVisible() {
       
    98       const ctx = (0,interactivity_namespaceObject.getContext)();
       
    99       return !state.overlayEnabled && state.currentImageId === ctx.imageId;
       
   100     }
       
   101   },
       
   102   actions: {
       
   103     showLightbox() {
       
   104       const {
       
   105         imageId
       
   106       } = (0,interactivity_namespaceObject.getContext)();
       
   107 
       
   108       // Bails out if the image has not loaded yet.
       
   109       if (!state.metadata[imageId].imageRef?.complete) {
       
   110         return;
       
   111       }
       
   112 
       
   113       // Stores the positions of the scroll to fix it until the overlay is
       
   114       // closed.
       
   115       state.scrollTopReset = document.documentElement.scrollTop;
       
   116       state.scrollLeftReset = document.documentElement.scrollLeft;
       
   117 
       
   118       // Sets the current expanded image in the state and enables the overlay.
       
   119       state.overlayEnabled = true;
       
   120       state.currentImageId = imageId;
       
   121 
       
   122       // Computes the styles of the overlay for the animation.
       
   123       callbacks.setOverlayStyles();
       
   124     },
       
   125     hideLightbox() {
       
   126       if (state.overlayEnabled) {
       
   127         // Starts the overlay closing animation. The showClosingAnimation
       
   128         // class is used to avoid showing it on page load.
       
   129         state.showClosingAnimation = true;
       
   130         state.overlayEnabled = false;
       
   131 
       
   132         // Waits until the close animation has completed before allowing a
       
   133         // user to scroll again. The duration of this animation is defined in
       
   134         // the `styles.scss` file, but in any case we should wait a few
       
   135         // milliseconds longer than the duration, otherwise a user may scroll
       
   136         // too soon and cause the animation to look sloppy.
       
   137         setTimeout(function () {
       
   138           // Delays before changing the focus. Otherwise the focus ring will
       
   139           // appear on Firefox before the image has finished animating, which
       
   140           // looks broken.
       
   141           state.currentImage.buttonRef.focus({
       
   142             preventScroll: true
       
   143           });
       
   144 
       
   145           // Resets the current image id to mark the overlay as closed.
       
   146           state.currentImageId = null;
       
   147         }, 450);
       
   148       }
       
   149     },
       
   150     handleKeydown: (0,interactivity_namespaceObject.withSyncEvent)(event => {
       
   151       if (state.overlayEnabled) {
       
   152         // Focuses the close button when the user presses the tab key.
       
   153         if (event.key === 'Tab') {
       
   154           event.preventDefault();
       
   155           const {
       
   156             ref
       
   157           } = (0,interactivity_namespaceObject.getElement)();
       
   158           ref.querySelector('button').focus();
       
   159         }
       
   160         // Closes the lightbox when the user presses the escape key.
       
   161         if (event.key === 'Escape') {
       
   162           actions.hideLightbox();
       
   163         }
       
   164       }
       
   165     }),
       
   166     handleTouchMove: (0,interactivity_namespaceObject.withSyncEvent)(event => {
       
   167       // On mobile devices, prevents triggering the scroll event because
       
   168       // otherwise the page jumps around when it resets the scroll position.
       
   169       // This also means that closing the lightbox requires that a user
       
   170       // perform a simple tap. This may be changed in the future if there is a
       
   171       // better alternative to override or reset the scroll position during
       
   172       // swipe actions.
       
   173       if (state.overlayEnabled) {
       
   174         event.preventDefault();
       
   175       }
       
   176     }),
       
   177     handleTouchStart() {
       
   178       isTouching = true;
       
   179     },
       
   180     handleTouchEnd() {
       
   181       // Waits a few milliseconds before resetting to ensure that pinch to
       
   182       // zoom works consistently on mobile devices when the lightbox is open.
       
   183       lastTouchTime = Date.now();
       
   184       isTouching = false;
       
   185     },
       
   186     handleScroll() {
       
   187       // Prevents scrolling behaviors that trigger content shift while the
       
   188       // lightbox is open. It would be better to accomplish through CSS alone,
       
   189       // but using overflow: hidden is currently the only way to do so and
       
   190       // that causes a layout to shift and prevents the zoom animation from
       
   191       // working in some cases because it's not possible to account for the
       
   192       // layout shift when doing the animation calculations. Instead, it uses
       
   193       // JavaScript to prevent and reset the scrolling behavior.
       
   194       if (state.overlayOpened) {
       
   195         // Avoids overriding the scroll behavior on mobile devices because
       
   196         // doing so breaks the pinch to zoom functionality, and users should
       
   197         // be able to zoom in further on the high-res image.
       
   198         if (!isTouching && Date.now() - lastTouchTime > 450) {
       
   199           // It doesn't rely on `event.preventDefault()` to prevent scrolling
       
   200           // because the scroll event can't be canceled, so it resets the
       
   201           // position instead.
       
   202           window.scrollTo(state.scrollLeftReset, state.scrollTopReset);
       
   203         }
       
   204       }
       
   205     }
       
   206   },
       
   207   callbacks: {
       
   208     setOverlayStyles() {
       
   209       if (!state.overlayEnabled) {
       
   210         return;
       
   211       }
       
   212       let {
       
   213         naturalWidth,
       
   214         naturalHeight,
       
   215         offsetWidth: originalWidth,
       
   216         offsetHeight: originalHeight
       
   217       } = state.currentImage.imageRef;
       
   218       let {
       
   219         x: screenPosX,
       
   220         y: screenPosY
       
   221       } = state.currentImage.imageRef.getBoundingClientRect();
       
   222 
       
   223       // Natural ratio of the image clicked to open the lightbox.
       
   224       const naturalRatio = naturalWidth / naturalHeight;
       
   225       // Original ratio of the image clicked to open the lightbox.
       
   226       let originalRatio = originalWidth / originalHeight;
       
   227 
       
   228       // If it has object-fit: contain, recalculates the original sizes
       
   229       // and the screen position without the blank spaces.
       
   230       if (state.currentImage.scaleAttr === 'contain') {
       
   231         if (naturalRatio > originalRatio) {
       
   232           const heightWithoutSpace = originalWidth / naturalRatio;
       
   233           // Recalculates screen position without the top space.
       
   234           screenPosY += (originalHeight - heightWithoutSpace) / 2;
       
   235           originalHeight = heightWithoutSpace;
       
   236         } else {
       
   237           const widthWithoutSpace = originalHeight * naturalRatio;
       
   238           // Recalculates screen position without the left space.
       
   239           screenPosX += (originalWidth - widthWithoutSpace) / 2;
       
   240           originalWidth = widthWithoutSpace;
       
   241         }
       
   242       }
       
   243       originalRatio = originalWidth / originalHeight;
       
   244 
       
   245       // Typically, it uses the image's full-sized dimensions. If those
       
   246       // dimensions have not been set (i.e. an external image with only one
       
   247       // size), the image's dimensions in the lightbox are the same
       
   248       // as those of the image in the content.
       
   249       let imgMaxWidth = parseFloat(state.currentImage.targetWidth !== 'none' ? state.currentImage.targetWidth : naturalWidth);
       
   250       let imgMaxHeight = parseFloat(state.currentImage.targetHeight !== 'none' ? state.currentImage.targetHeight : naturalHeight);
       
   251 
       
   252       // Ratio of the biggest image stored in the database.
       
   253       let imgRatio = imgMaxWidth / imgMaxHeight;
       
   254       let containerMaxWidth = imgMaxWidth;
       
   255       let containerMaxHeight = imgMaxHeight;
       
   256       let containerWidth = imgMaxWidth;
       
   257       let containerHeight = imgMaxHeight;
       
   258 
       
   259       // Checks if the target image has a different ratio than the original
       
   260       // one (thumbnail). Recalculates the width and height.
       
   261       if (naturalRatio.toFixed(2) !== imgRatio.toFixed(2)) {
       
   262         if (naturalRatio > imgRatio) {
       
   263           // If the width is reached before the height, it keeps the maxWidth
       
   264           // and recalculates the height unless the difference between the
       
   265           // maxHeight and the reducedHeight is higher than the maxWidth,
       
   266           // where it keeps the reducedHeight and recalculate the width.
       
   267           const reducedHeight = imgMaxWidth / naturalRatio;
       
   268           if (imgMaxHeight - reducedHeight > imgMaxWidth) {
       
   269             imgMaxHeight = reducedHeight;
       
   270             imgMaxWidth = reducedHeight * naturalRatio;
       
   271           } else {
       
   272             imgMaxHeight = imgMaxWidth / naturalRatio;
       
   273           }
       
   274         } else {
       
   275           // If the height is reached before the width, it keeps the maxHeight
       
   276           // and recalculate the width unlesss the difference between the
       
   277           // maxWidth and the reducedWidth is higher than the maxHeight, where
       
   278           // it keeps the reducedWidth and recalculate the height.
       
   279           const reducedWidth = imgMaxHeight * naturalRatio;
       
   280           if (imgMaxWidth - reducedWidth > imgMaxHeight) {
       
   281             imgMaxWidth = reducedWidth;
       
   282             imgMaxHeight = reducedWidth / naturalRatio;
       
   283           } else {
       
   284             imgMaxWidth = imgMaxHeight * naturalRatio;
       
   285           }
       
   286         }
       
   287         containerWidth = imgMaxWidth;
       
   288         containerHeight = imgMaxHeight;
       
   289         imgRatio = imgMaxWidth / imgMaxHeight;
       
   290 
       
   291         // Calculates the max size of the container.
       
   292         if (originalRatio > imgRatio) {
       
   293           containerMaxWidth = imgMaxWidth;
       
   294           containerMaxHeight = containerMaxWidth / originalRatio;
       
   295         } else {
       
   296           containerMaxHeight = imgMaxHeight;
       
   297           containerMaxWidth = containerMaxHeight * originalRatio;
       
   298         }
       
   299       }
       
   300 
       
   301       // If the image has been pixelated on purpose, it keeps that size.
       
   302       if (originalWidth > containerWidth || originalHeight > containerHeight) {
       
   303         containerWidth = originalWidth;
       
   304         containerHeight = originalHeight;
       
   305       }
       
   306 
       
   307       // Calculates the final lightbox image size and the scale factor.
       
   308       // MaxWidth is either the window container (accounting for padding) or
       
   309       // the image resolution.
       
   310       let horizontalPadding = 0;
       
   311       if (window.innerWidth > 480) {
       
   312         horizontalPadding = 80;
       
   313       } else if (window.innerWidth > 1920) {
       
   314         horizontalPadding = 160;
       
   315       }
       
   316       const verticalPadding = 80;
       
   317       const targetMaxWidth = Math.min(window.innerWidth - horizontalPadding, containerWidth);
       
   318       const targetMaxHeight = Math.min(window.innerHeight - verticalPadding, containerHeight);
       
   319       const targetContainerRatio = targetMaxWidth / targetMaxHeight;
       
   320       if (originalRatio > targetContainerRatio) {
       
   321         // If targetMaxWidth is reached before targetMaxHeight.
       
   322         containerWidth = targetMaxWidth;
       
   323         containerHeight = containerWidth / originalRatio;
       
   324       } else {
       
   325         // If targetMaxHeight is reached before targetMaxWidth.
       
   326         containerHeight = targetMaxHeight;
       
   327         containerWidth = containerHeight * originalRatio;
       
   328       }
       
   329       const containerScale = originalWidth / containerWidth;
       
   330       const lightboxImgWidth = imgMaxWidth * (containerWidth / containerMaxWidth);
       
   331       const lightboxImgHeight = imgMaxHeight * (containerHeight / containerMaxHeight);
       
   332 
       
   333       // As of this writing, using the calculations above will render the
       
   334       // lightbox with a small, erroneous whitespace on the left side of the
       
   335       // image in iOS Safari, perhaps due to an inconsistency in how browsers
       
   336       // handle absolute positioning and CSS transformation. In any case,
       
   337       // adding 1 pixel to the container width and height solves the problem,
       
   338       // though this can be removed if the issue is fixed in the future.
       
   339       state.overlayStyles = `
       
   340 					--wp--lightbox-initial-top-position: ${screenPosY}px;
       
   341 					--wp--lightbox-initial-left-position: ${screenPosX}px;
       
   342 					--wp--lightbox-container-width: ${containerWidth + 1}px;
       
   343 					--wp--lightbox-container-height: ${containerHeight + 1}px;
       
   344 					--wp--lightbox-image-width: ${lightboxImgWidth}px;
       
   345 					--wp--lightbox-image-height: ${lightboxImgHeight}px;
       
   346 					--wp--lightbox-scale: ${containerScale};
       
   347 					--wp--lightbox-scrollbar-width: ${window.innerWidth - document.documentElement.clientWidth}px;
       
   348 				`;
       
   349     },
       
   350     setButtonStyles() {
       
   351       const {
       
   352         imageId
       
   353       } = (0,interactivity_namespaceObject.getContext)();
       
   354       const {
       
   355         ref
       
   356       } = (0,interactivity_namespaceObject.getElement)();
       
   357       state.metadata[imageId].imageRef = ref;
       
   358       state.metadata[imageId].currentSrc = ref.currentSrc;
       
   359       const {
       
   360         naturalWidth,
       
   361         naturalHeight,
       
   362         offsetWidth,
       
   363         offsetHeight
       
   364       } = ref;
       
   365 
       
   366       // If the image isn't loaded yet, it can't calculate where the button
       
   367       // should be.
       
   368       if (naturalWidth === 0 || naturalHeight === 0) {
       
   369         return;
       
   370       }
       
   371       const figure = ref.parentElement;
       
   372       const figureWidth = ref.parentElement.clientWidth;
       
   373 
       
   374       // It needs special handling for the height because a caption will cause
       
   375       // the figure to be taller than the image, which means it needs to
       
   376       // account for that when calculating the placement of the button in the
       
   377       // top right corner of the image.
       
   378       let figureHeight = ref.parentElement.clientHeight;
       
   379       const caption = figure.querySelector('figcaption');
       
   380       if (caption) {
       
   381         const captionComputedStyle = window.getComputedStyle(caption);
       
   382         if (!['absolute', 'fixed'].includes(captionComputedStyle.position)) {
       
   383           figureHeight = figureHeight - caption.offsetHeight - parseFloat(captionComputedStyle.marginTop) - parseFloat(captionComputedStyle.marginBottom);
       
   384         }
       
   385       }
       
   386       const buttonOffsetTop = figureHeight - offsetHeight;
       
   387       const buttonOffsetRight = figureWidth - offsetWidth;
       
   388       let imageButtonTop = buttonOffsetTop + 16;
       
   389       let imageButtonRight = buttonOffsetRight + 16;
       
   390 
       
   391       // In the case of an image with object-fit: contain, the size of the
       
   392       // <img> element can be larger than the image itself, so it needs to
       
   393       // calculate where to place the button.
       
   394       if (state.metadata[imageId].scaleAttr === 'contain') {
       
   395         // Natural ratio of the image.
       
   396         const naturalRatio = naturalWidth / naturalHeight;
       
   397         // Offset ratio of the image.
       
   398         const offsetRatio = offsetWidth / offsetHeight;
       
   399         if (naturalRatio >= offsetRatio) {
       
   400           // If it reaches the width first, it keeps the width and compute the
       
   401           // height.
       
   402           const referenceHeight = offsetWidth / naturalRatio;
       
   403           imageButtonTop = (offsetHeight - referenceHeight) / 2 + buttonOffsetTop + 16;
       
   404           imageButtonRight = buttonOffsetRight + 16;
       
   405         } else {
       
   406           // If it reaches the height first, it keeps the height and compute
       
   407           // the width.
       
   408           const referenceWidth = offsetHeight * naturalRatio;
       
   409           imageButtonTop = buttonOffsetTop + 16;
       
   410           imageButtonRight = (offsetWidth - referenceWidth) / 2 + buttonOffsetRight + 16;
       
   411         }
       
   412       }
       
   413       state.metadata[imageId].imageButtonTop = imageButtonTop;
       
   414       state.metadata[imageId].imageButtonRight = imageButtonRight;
       
   415     },
       
   416     setOverlayFocus() {
       
   417       if (state.overlayEnabled) {
       
   418         // Moves the focus to the dialog when it opens.
       
   419         const {
       
   420           ref
       
   421         } = (0,interactivity_namespaceObject.getElement)();
       
   422         ref.focus();
       
   423       }
       
   424     },
       
   425     initTriggerButton() {
       
   426       const {
       
   427         imageId
       
   428       } = (0,interactivity_namespaceObject.getContext)();
       
   429       const {
       
   430         ref
       
   431       } = (0,interactivity_namespaceObject.getElement)();
       
   432       state.metadata[imageId].buttonRef = ref;
       
   433     }
       
   434   }
       
   435 }, {
       
   436   lock: true
       
   437 });
       
   438