MediaHandlerConfig.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.spi;
import java.util.Collections;
import java.util.EnumSet;
import java.util.List;
import java.util.Set;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.osgi.annotation.versioning.ConsumerType;
import com.day.cq.wcm.api.Page;
import io.wcm.handler.media.MediaFileType;
import io.wcm.handler.media.MediaNameConstants;
import io.wcm.handler.media.markup.DummyImageMediaMarkupBuilder;
import io.wcm.handler.media.markup.MediaMarkupBuilderUtil;
import io.wcm.handler.media.markup.SimpleImageMediaMarkupBuilder;
import io.wcm.handler.mediasource.dam.AemRenditionType;
import io.wcm.handler.mediasource.dam.DamMediaSource;
import io.wcm.sling.commons.caservice.ContextAwareService;
/**
* {@link MediaHandlerConfig} OSGi services provide application-specific configuration for media handling.
* Applications can set service properties or bundle headers as defined in {@link ContextAwareService} to apply this
* configuration only for resources that match the relevant resource paths.
*/
@ConsumerType
public abstract class MediaHandlerConfig implements ContextAwareService {
/**
* Default image quality for images with lossy compressions (e.g. JPEG).
*/
public static final double DEFAULT_IMAGE_QUALITY = 0.85d;
/**
* Default value for JPEG quality.
* @deprecated Use {@link #DEFAULT_IMAGE_QUALITY} instead.
*/
@Deprecated(since = "2.0.0")
public static final double DEFAULT_JPEG_QUALITY = DEFAULT_IMAGE_QUALITY;
private static final List<Class<? extends MediaSource>> DEFAULT_MEDIA_SOURCES = List.of(
DamMediaSource.class);
private static final List<Class<? extends MediaMarkupBuilder>> DEFAULT_MEDIA_MARKUP_BUILDERS = List.of(
SimpleImageMediaMarkupBuilder.class,
DummyImageMediaMarkupBuilder.class);
/**
* @return Supported media sources
*/
public @NotNull List<Class<? extends MediaSource>> getSources() {
return DEFAULT_MEDIA_SOURCES;
}
/**
* @return Available media markup builders
*/
public @NotNull List<Class<? extends MediaMarkupBuilder>> getMarkupBuilders() {
return DEFAULT_MEDIA_MARKUP_BUILDERS;
}
/**
* @return List of media metadata pre processors (optional). The processors are applied in list order.
*/
public @NotNull List<Class<? extends MediaProcessor>> getPreProcessors() {
// no processors
return Collections.emptyList();
}
/**
* @return List of media metadata post processors (optional). The processors are applied in list order.
*/
public @NotNull List<Class<? extends MediaProcessor>> getPostProcessors() {
// no processors
return Collections.emptyList();
}
/**
* Get the default quality for images.
* The meaning of the quality parameter for the different image formats is described in
* {@link com.day.image.Layer#write(String, double, java.io.OutputStream)}.
* @param contentType MIME-type of the output format
* @return Quality factor
*/
public double getDefaultImageQuality(@Nullable String contentType) {
MediaFileType mediaFileType = MediaFileType.getByContentType(contentType);
if (mediaFileType != null && mediaFileType.isImageQualityPercentage()) {
return getDefaultImageQualityPercentage();
}
else if (mediaFileType == MediaFileType.GIF) {
return 256d; // 256 colors
}
// return quality "1" for all other mime types
return 1d;
}
/**
* Get the default quality for images.
* This parameter only applies to images with lossy compression (e.g. JPEG).
* @return Quality percentage (0..1)
*/
public double getDefaultImageQualityPercentage() {
return DEFAULT_IMAGE_QUALITY;
}
/**
* With this switch it's possible to switch all used property and node names from (legacy) wcm.io
* Handler standard to Adobe Standard (as used e.g. in Adobe Core WCM Components) - e.g.
* using "fileReference" instead of property name "mediaRef" for the asset reference.
*
* <p>
* The benefit of the wcm.io Handler standard was that it supported storage multiple asset references
* in one single node - but this it not well supported by the Touch UI anyway, so it's not of much
* use nowadays.
* </p>
*
* <p>
* For new projects it is recommended to always use the Adobe standard names. But for backward compatibility
* the default values is false.
* </p>
* @return If true, Adobe standard property and node names are used.
*/
public boolean useAdobeStandardNames() {
return false;
}
/**
* @return Default property name for reference to media library item
*/
public @NotNull String getMediaRefProperty() {
if (useAdobeStandardNames()) {
return MediaNameConstants.PN_MEDIA_REF_STANDARD;
}
else {
return MediaNameConstants.PN_MEDIA_REF;
}
}
/**
* @return Default property name for cropping parameters
*/
public @NotNull String getMediaCropProperty() {
if (useAdobeStandardNames()) {
return MediaNameConstants.PN_MEDIA_CROP_STANDARD;
}
else {
return MediaNameConstants.PN_MEDIA_CROP;
}
}
/**
* @return Default property name for rotate parameter
*/
public @NotNull String getMediaRotationProperty() {
if (useAdobeStandardNames()) {
return MediaNameConstants.PN_MEDIA_ROTATION_STANDARD;
}
else {
return MediaNameConstants.PN_MEDIA_ROTATION;
}
}
/**
* @return Default property name for map parameter
*/
public @NotNull String getMediaMapProperty() {
if (useAdobeStandardNames()) {
return MediaNameConstants.PN_MEDIA_MAP_STANDARD;
}
else {
return MediaNameConstants.PN_MEDIA_MAP;
}
}
/**
* @return Default property name for media alt. text
*/
public @NotNull String getMediaAltTextProperty() {
if (useAdobeStandardNames()) {
return MediaNameConstants.PN_MEDIA_ALTTEXT_STANDARD;
}
else {
return MediaNameConstants.PN_MEDIA_ALTTEXT;
}
}
/**
* @return Default property name for forcing reading alt. text from DAM asset description
*/
public @NotNull String getMediaForceAltTextFromAssetProperty() {
if (useAdobeStandardNames()) {
return MediaNameConstants.PN_MEDIA_FORCE_ALTTEXT_FROM_ASSET_STANDARD;
}
else {
return MediaNameConstants.PN_MEDIA_FORCE_ALTTEXT_FROM_ASSET;
}
}
/**
* @return Default property name for marking image as "decorative" - requiring no alt. text
*/
public @NotNull String getMediaIsDecorativeProperty() {
if (useAdobeStandardNames()) {
return MediaNameConstants.PN_MEDIA_IS_DECORATIVE_STANDARD;
}
else {
return MediaNameConstants.PN_MEDIA_IS_DECORATIVE;
}
}
/**
* @return Default node name for inline media item stored in node within the content page
*/
public @NotNull String getMediaInlineNodeName() {
if (useAdobeStandardNames()) {
return MediaNameConstants.NN_MEDIA_INLINE_STANDARD;
}
else {
return MediaNameConstants.NN_MEDIA_INLINE;
}
}
/**
* @return If set to true, web renditions generated by AEM (with <code>cq5dam.web.</code> prefix) are
* taken into account by default when trying to resolve the media request.
* @deprecated Use {@link #getIncludeAssetAemRenditionsByDefault()} instead.
*/
@Deprecated(since = "2.0.0")
public boolean includeAssetWebRenditionsByDefault() {
return false;
}
/**
* Set of renditions auto-generated by AEM (with <code>cq5dam.</code> prefix) which are taken into account
* by default when trying to resolve the media request.
* @return Set or rendition types
*/
@SuppressWarnings("java:S1874") // ignore use of deprecated method
public @NotNull Set<AemRenditionType> getIncludeAssetAemRenditionsByDefault() {
if (includeAssetWebRenditionsByDefault()) {
return EnumSet.of(AemRenditionType.WEB_RENDITION, AemRenditionType.VIDEO_RENDITION);
}
else {
return EnumSet.of(AemRenditionType.VIDEO_RENDITION);
}
}
/**
* Enforce to generate only virtual renditions.
*
* <p>
* By default, virtual renditions (rendered on-the-fly via <code>ImageFileServet</code>) are only
* generated if there is a need to re-scale or crop or transform an image. Otherwise direct references
* to renditions or original stored in DAM are returned when there is an direct match with the requested ratio and
* resolution.
* </p>
*
* <p>
* When this flag is set to <code>true</code>, even if there is a direct match a virtual rendition is returned.
* This ensures that the default quality setting e.g. for JPEG images is always respected, regardless
* in which quality the original images was uploaded.
* </p>
* @return Enforce always returning virtual renditions for images.
*/
public boolean enforceVirtualRenditions() {
return false;
}
/**
* @return Allowed editor types for image IPE (in-place editor).
* By default, only the OOTB "image" editor type is supported.
*/
public @NotNull Set<String> allowedIpeEditorTypes() {
return MediaMarkupBuilderUtil.DEFAULT_ALLOWED_IPE_EDITOR_TYPES;
}
/**
* Get root path for picking assets using path field widgets.
* @param page Context page
* @return DAM root path
*/
public @NotNull String getDamRootPath(@NotNull Page page) {
return "/content/dam";
}
}