SimpleImageMediaMarkupBuilder.java
/*
* #%L
* wcm.io
* %%
* Copyright (C) 2014 wcm.io
* %%
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* #L%
*/
package io.wcm.handler.media.markup;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.stream.Collectors;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.builder.HashCodeBuilder;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.models.annotations.Model;
import org.jdom2.Element;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.osgi.annotation.versioning.ConsumerType;
import io.wcm.handler.commons.dom.Area;
import io.wcm.handler.commons.dom.HtmlElement;
import io.wcm.handler.commons.dom.Image;
import io.wcm.handler.commons.dom.Map;
import io.wcm.handler.commons.dom.Picture;
import io.wcm.handler.commons.dom.Source;
import io.wcm.handler.commons.dom.Span;
import io.wcm.handler.media.Asset;
import io.wcm.handler.media.Media;
import io.wcm.handler.media.MediaArgs;
import io.wcm.handler.media.MediaArgs.ImageSizes;
import io.wcm.handler.media.MediaArgs.PictureSource;
import io.wcm.handler.media.MediaArgs.WidthOption;
import io.wcm.handler.media.MediaNameConstants;
import io.wcm.handler.media.Rendition;
import io.wcm.handler.media.format.MediaFormat;
import io.wcm.handler.media.format.Ratio;
import io.wcm.handler.media.imagemap.ImageMapArea;
/**
* Basic implementation of {@link io.wcm.handler.media.spi.MediaMarkupBuilder} for images.
*
* <p>
* If image sizes or picture sources are set on the media handler this markup builder also
* generates markup for responsive images using <code>img</code> with <code>sizes</code> and <code>srcset</code>
* attributes or <code>picture</code> with <code>source</code> elements.
* </p>
*/
@Model(adaptables = {
SlingHttpServletRequest.class, Resource.class
})
@ConsumerType
public class SimpleImageMediaMarkupBuilder extends AbstractImageMediaMarkupBuilder {
@Override
public final boolean accepts(@NotNull Media media) {
// accept if rendition is an image rendition, and resolving was successful
Rendition rendition = media.getRendition();
return media.isValid()
&& rendition != null
&& rendition.isBrowserImage();
}
@Override
public final HtmlElement build(@NotNull Media media) {
// render media element for rendition
HtmlElement mediaElement = getMediaElement(media);
// further processing in edit or preview mode
applyWcmMarkup(mediaElement, media);
return mediaElement;
}
/**
* Create <code>img</code> or <code>picture</code> media element.
* @param media Media metadata
* @return Media element with properties or null if media metadata is invalid
*/
protected @Nullable HtmlElement getMediaElement(@NotNull Media media) {
PictureSource[] pictureSources = media.getMediaRequest().getMediaArgs().getPictureSources();
if (pictureSources != null && pictureSources.length > 0) {
return getPictureElement(media);
}
else {
return getImageElement(media);
}
}
/**
* Create an <code>img</code> element that displays the given rendition image.
* @param media Media metadata
* @return <code>img</code> element with properties or null if media metadata is invalid
*/
@SuppressWarnings("java:S3776") // ignore complexity
protected @Nullable HtmlElement getPictureElement(@NotNull Media media) {
PictureSource[] pictureSources = media.getMediaRequest().getMediaArgs().getPictureSources();
Picture picture = new Picture();
// add source elements (only if matching renditions found)
boolean foundAnySource = false;
if (pictureSources != null) {
for (PictureSource pictureSource : pictureSources) {
Source source = new Source();
if (pictureSource.getMedia() != null) {
source.setMedia(pictureSource.getMedia());
}
if (pictureSource.getSizes() != null) {
source.setSizes(pictureSource.getSizes());
}
MediaFormat mediaFormat = pictureSource.getMediaFormat();
if (mediaFormat != null) {
String srcSet = getSrcSetRenditions(media, mediaFormat, pictureSource.getWidthOptions(), pictureSource.hasDensityDescriptors());
if (srcSet != null) {
source.setSrcSet(srcSet);
picture.add(source);
foundAnySource = true;
}
}
}
}
// add image element
HtmlElement image = getImageElement(media);
if (image == null) {
return null;
}
if (foundAnySource) {
if (image instanceof Span) {
// if image was wrapped in span, add content of span element, not the span itself
for (Element element : List.copyOf(image.getChildren())) {
element.detach();
picture.addContent(element);
}
}
else {
picture.addContent(image);
}
return picture;
}
else {
return image;
}
}
/**
* Create an <code>img</code> element that displays the given rendition image.
* @param media Media metadata
* @return <code>img</code> element with properties or null if media metadata is invalid
*/
@SuppressWarnings({ "java:S3776", "java:S2589" }) // ignore complexity
protected @Nullable HtmlElement getImageElement(@NotNull Media media) {
Image img = null;
MediaArgs mediaArgs = media.getMediaRequest().getMediaArgs();
Asset asset = media.getAsset();
Rendition rendition = media.getRendition();
String url = null;
if (rendition != null) {
url = rendition.getUrl();
}
if (url != null) {
img = new Image(url);
// Alternative text
String altText = null;
if (asset != null) {
altText = asset.getAltText();
}
if (altText != null) {
img.setAlt(altText);
}
// set width/height
if (rendition != null
&& !rendition.isVectorImage()
&& mediaArgs.getImageSizes() == null
&& mediaArgs.getPictureSources() == null) {
long height = rendition.getHeight();
if (height > 0) {
img.setHeight(height);
}
long width = rendition.getWidth();
if (width > 0) {
img.setWidth(width);
}
}
// set image sizes/srcset
ImageSizes imageSizes = mediaArgs.getImageSizes();
if (imageSizes != null) {
MediaFormat primaryMediaFormat = getFirstMediaFormat(media);
if (primaryMediaFormat != null) {
String srcSet = getSrcSetRenditions(media, primaryMediaFormat, imageSizes.getWidthOptions(), imageSizes.hasDensityDescriptors());
if (srcSet != null) {
img.setSrcSet(srcSet);
img.setSizes(imageSizes.getSizes());
}
}
}
}
// set additional attributes
setAdditionalAttributes(img, media);
// apply image map markup
return applyImageMap(img, media);
}
/**
* Generate srcset list from the resolved renditions for the ratio of the given media formats and the given widths.
* Widths that have no match are ignored.
* @param media Media
* @param mediaFormat Media format
* @param widthOptions width options
* @param hasDensityDescriptors render density descriptors instead of width descriptors
* @return srcset String or null if no matching renditions found
*/
protected @Nullable String getSrcSetRenditions(@NotNull Media media, @NotNull MediaFormat mediaFormat,
@NotNull WidthOption @Nullable [] widthOptions, boolean hasDensityDescriptors) {
if (widthOptions == null || widthOptions.length == 0) {
return null;
}
String srcset = Arrays.stream(widthOptions)
.map(widthOption -> getSrcSetRenditionUrl(media, mediaFormat, widthOption, hasDensityDescriptors))
.filter(Objects::nonNull)
.collect(Collectors.joining(", "));
return !srcset.isEmpty() ? srcset : null;
}
/**
* Generate a rendition URL with descriptor (width or density) for the given media format and width option
* @param media Media
* @param mediaFormat Media format
* @param widthOption width option
* @param hasDensityDescriptors render density descriptors instead of width descriptors
* @return a media URL with descriptor, or null if no matching renditions found
*/
protected @Nullable String getSrcSetRenditionUrl(@NotNull Media media, @NotNull MediaFormat mediaFormat, @NotNull WidthOption widthOption, boolean hasDensityDescriptors) {
String descriptor = hasDensityDescriptors ? widthOption.getDensityDescriptor() : widthOption.getWidthDescriptor();
return media.getRenditions().stream()
.filter(rendition -> (Ratio.matches(rendition.getRatio(), mediaFormat.getRatio())
|| Ratio.matches(mediaFormat.getRatio(), 0d))
&& rendition.getWidth() == widthOption.getWidth())
.map(Rendition::getUrl)
.findFirst()
.map(url -> url + (!descriptor.isEmpty() ? " " + descriptor : ""))
.orElse(null);
}
/**
* Generate srcset list from the resolved renditions for the ratio of the given media formats and the given widths.
* Widths that have no match are ignored.
* @param media Media
* @param mediaFormat Media format
* @param widths widths
* @return srcset String or null if no matching renditions found
*/
protected @Nullable String getSrcSetRenditions(@NotNull Media media, @NotNull MediaFormat mediaFormat,
@NotNull WidthOption @Nullable... widths) {
if (widths == null) {
return null;
}
return getSrcSetRenditions(media, mediaFormat, Arrays.stream(widths)
.mapToLong(WidthOption::getWidth)
.toArray());
}
/**
* Generate srcset list from the resolved renditions for the ratio of the given media formats and the given widths.
* Widths that have no match are ignored.
* @param media Media
* @param mediaFormat Media format
* @param widths widths
* @return srcset String or null if no matching renditions found
*/
protected @Nullable String getSrcSetRenditions(@NotNull Media media, @NotNull MediaFormat mediaFormat,
long @NotNull... widths) {
StringBuilder srcset = new StringBuilder();
for (long width : widths) {
Optional<String> url = media.getRenditions().stream()
.filter(rendition -> (Ratio.matches(rendition.getRatio(), mediaFormat.getRatio())
|| Ratio.matches(mediaFormat.getRatio(), 0d))
&& rendition.getWidth() == width)
.map(Rendition::getUrl)
.findFirst();
if (url.isPresent()) {
if (srcset.length() > 0) {
srcset.append(", ");
}
srcset.append(url.get()).append(" ").append(Long.toString(width)).append("w");
}
}
if (srcset.length() > 0) {
return srcset.toString();
}
else {
return null;
}
}
/**
* Get first media format from the media formats of the media args that has a ratio set.
* @param media Media
* @return Media format or null if none found
*/
protected final @Nullable MediaFormat getFirstMediaFormatWithRatio(@NotNull Media media) {
MediaFormat[] mediaFormats = media.getMediaRequest().getMediaArgs().getMediaFormats();
if (mediaFormats != null) {
for (MediaFormat mediaFormat : mediaFormats) {
if (mediaFormat.hasRatio()) {
return mediaFormat;
}
}
}
return null;
}
/**
* Get first media format from the resolved media renditions.
* @param media Media
* @return Media format or null if none found
*/
@SuppressWarnings("null")
protected final @Nullable MediaFormat getFirstMediaFormat(@NotNull Media media) {
return media.getRenditions().stream()
.map(Rendition::getMediaFormat)
.filter(Objects::nonNull)
.findFirst()
.orElse(null);
}
/**
* If a image map was resolved apply map markup to given image element. As a result both image
* and map markup are wrapped in a span element.
* @param element Image Element
* @param media Media
* @return Unchanged element or wrapped element with map
*/
protected final @Nullable HtmlElement applyImageMap(@Nullable HtmlElement element, @NotNull Media media) {
List<ImageMapArea> mapData = media.getMap();
if (!(element instanceof Image) || mapData == null) {
return element;
}
// build unique name for map
String mapName = buildImageMapName(mapData, media);
// build wrapper element that will contain both image and map element
Span span = new Span();
((Image)element).setUseMap("#" + mapName);
span.addContent(element);
// build image map markup
Map map = new Map();
map.setMapName(mapName);
for (ImageMapArea areaData : mapData) {
Area area = new Area();
area.setShape(areaData.getShape());
area.setCoords(areaData.getCoordinates());
area.setHRef(areaData.getLinkUrl());
if (areaData.getLinkWindowTarget() != null) {
area.setTarget(areaData.getLinkWindowTarget());
}
if (areaData.getAltText() != null) {
area.setAlt(areaData.getAltText());
}
map.addContent(area);
}
span.addContent(map);
return span;
}
/**
* Builds an ID for the image map that is unique within the page.
* @param map Map data
* @param media Media
* @return Unique ID
*/
protected final @NotNull String buildImageMapName(@NotNull List<ImageMapArea> map, @NotNull Media media) {
HashCodeBuilder builder = new HashCodeBuilder();
for (ImageMapArea area : map) {
builder.append(area);
}
return "map-" + builder.hashCode();
}
@Override
public final boolean isValidMedia(@NotNull HtmlElement element) {
if (element instanceof Image) {
Image img = (Image)element;
return StringUtils.isNotEmpty(img.getSrc())
&& !StringUtils.contains(img.getCssClass(), MediaNameConstants.CSS_DUMMYIMAGE);
}
if (element instanceof Picture) {
Element imgChild = element.getChild("img");
if (imgChild instanceof Image) {
Image img = (Image)imgChild;
return StringUtils.isNotEmpty(img.getSrc())
&& !StringUtils.contains(element.getCssClass(), MediaNameConstants.CSS_DUMMYIMAGE);
}
}
if (element instanceof Span) {
Optional<Element> firstChild = element.getChildren().stream().findFirst();
if (firstChild.isPresent()) {
return isValidMedia((HtmlElement)firstChild.get());
}
}
return false;
}
}