You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

AFPImageHandlerRenderedImage.java 9.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. /*
  2. * Licensed to the Apache Software Foundation (ASF) under one or more
  3. * contributor license agreements. See the NOTICE file distributed with
  4. * this work for additional information regarding copyright ownership.
  5. * The ASF licenses this file to You under the Apache License, Version 2.0
  6. * (the "License"); you may not use this file except in compliance with
  7. * the License. You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. /* $Id$ */
  18. package org.apache.fop.render.afp;
  19. import java.awt.Dimension;
  20. import java.awt.image.ColorModel;
  21. import java.awt.image.RenderedImage;
  22. import java.io.IOException;
  23. import org.apache.commons.io.output.ByteArrayOutputStream;
  24. import org.apache.commons.logging.Log;
  25. import org.apache.commons.logging.LogFactory;
  26. import org.apache.xmlgraphics.image.loader.ImageFlavor;
  27. import org.apache.xmlgraphics.image.loader.ImageInfo;
  28. import org.apache.xmlgraphics.image.loader.ImageSize;
  29. import org.apache.xmlgraphics.image.loader.impl.ImageRendered;
  30. import org.apache.xmlgraphics.ps.ImageEncodingHelper;
  31. import org.apache.xmlgraphics.util.MimeConstants;
  32. import org.apache.xmlgraphics.util.UnitConv;
  33. import org.apache.fop.afp.AFPDataObjectInfo;
  34. import org.apache.fop.afp.AFPImageObjectInfo;
  35. import org.apache.fop.afp.AFPObjectAreaInfo;
  36. import org.apache.fop.afp.AFPPaintingState;
  37. import org.apache.fop.afp.AFPResourceInfo;
  38. import org.apache.fop.afp.modca.ResourceObject;
  39. import org.apache.fop.util.bitmap.BitmapImageUtil;
  40. /**
  41. * PDFImageHandler implementation which handles RenderedImage instances.
  42. */
  43. public class AFPImageHandlerRenderedImage extends AFPImageHandler {
  44. /** logging instance */
  45. private static Log log = LogFactory.getLog(AFPImageHandlerRenderedImage.class);
  46. private static final ImageFlavor[] FLAVORS = new ImageFlavor[] {
  47. ImageFlavor.BUFFERED_IMAGE,
  48. ImageFlavor.RENDERED_IMAGE
  49. };
  50. /** {@inheritDoc} */
  51. public AFPDataObjectInfo generateDataObjectInfo(
  52. AFPRendererImageInfo rendererImageInfo) throws IOException {
  53. AFPImageObjectInfo imageObjectInfo
  54. = (AFPImageObjectInfo)super.generateDataObjectInfo(rendererImageInfo);
  55. AFPRendererContext rendererContext
  56. = (AFPRendererContext)rendererImageInfo.getRendererContext();
  57. AFPInfo afpInfo = rendererContext.getInfo();
  58. AFPResourceInfo resourceInfo = imageObjectInfo.getResourceInfo();
  59. if (!resourceInfo.levelChanged()) {
  60. resourceInfo.setLevel(afpInfo.getResourceManager().getResourceLevelDefaults()
  61. .getDefaultResourceLevel(ResourceObject.TYPE_IMAGE));
  62. }
  63. AFPPaintingState paintingState = afpInfo.getPaintingState();
  64. int resolution = paintingState.getResolution();
  65. int maxPixelSize = paintingState.getBitsPerPixel();
  66. if (paintingState.isColorImages()) {
  67. maxPixelSize *= 3; //RGB only at the moment
  68. }
  69. ImageRendered imageRendered = (ImageRendered) rendererImageInfo.img;
  70. RenderedImage renderedImage = imageRendered.getRenderedImage();
  71. ImageInfo imageInfo = rendererImageInfo.getImageInfo();
  72. ImageSize intrinsicSize = imageInfo.getSize();
  73. boolean useFS10 = (maxPixelSize == 1) || BitmapImageUtil.isMonochromeImage(renderedImage);
  74. boolean usePageSegments = useFS10
  75. && !resourceInfo.getLevel().isInline();
  76. ImageSize effIntrinsicSize = intrinsicSize;
  77. if (usePageSegments) {
  78. //Resize, optionally resample and convert image
  79. Dimension resampledDim = new Dimension(
  80. (int)Math.ceil(UnitConv.mpt2px(afpInfo.getWidth(), resolution)),
  81. (int)Math.ceil(UnitConv.mpt2px(afpInfo.getHeight(), resolution)));
  82. imageObjectInfo.setCreatePageSegment(true);
  83. imageObjectInfo.getResourceInfo().setImageDimension(resampledDim);
  84. //Only resample/downsample if image is smaller than its intrinsic size
  85. //to make print file smaller
  86. boolean resample = resampledDim.width < renderedImage.getWidth()
  87. && resampledDim.height < renderedImage.getHeight();
  88. if (resample) {
  89. if (log.isDebugEnabled()) {
  90. log.debug("Resample from " + intrinsicSize.getDimensionPx()
  91. + " to " + resampledDim);
  92. }
  93. renderedImage = BitmapImageUtil.convertToMonochrome(renderedImage, resampledDim);
  94. effIntrinsicSize = new ImageSize(
  95. resampledDim.width, resampledDim.height, resolution);
  96. }
  97. }
  98. if (useFS10) {
  99. imageObjectInfo.setMimeType(MimeConstants.MIME_AFP_IOCA_FS10);
  100. } else {
  101. imageObjectInfo.setMimeType(MimeConstants.MIME_AFP_IOCA_FS11);
  102. }
  103. imageObjectInfo.setDataHeightRes((int)Math.round(
  104. effIntrinsicSize.getDpiHorizontal() * 10));
  105. imageObjectInfo.setDataWidthRes((int)Math.round(
  106. effIntrinsicSize.getDpiVertical() * 10));
  107. int dataHeight = renderedImage.getHeight();
  108. imageObjectInfo.setDataHeight(dataHeight);
  109. int dataWidth = renderedImage.getWidth();
  110. imageObjectInfo.setDataWidth(dataWidth);
  111. ColorModel cm = renderedImage.getColorModel();
  112. if (log.isTraceEnabled()) {
  113. log.trace("ColorModel: " + cm);
  114. }
  115. int pixelSize = cm.getPixelSize();
  116. if (cm.hasAlpha()) {
  117. pixelSize -= 8;
  118. }
  119. //TODO Add support for CMYK images
  120. byte[] imageData = null;
  121. ByteArrayOutputStream baos = new ByteArrayOutputStream();
  122. boolean allowDirectEncoding = true;
  123. if (allowDirectEncoding && pixelSize <= maxPixelSize) {
  124. //Attempt to encode without resampling the image
  125. ImageEncodingHelper helper = new ImageEncodingHelper(renderedImage);
  126. ColorModel encodedColorModel = helper.getEncodedColorModel();
  127. boolean directEncode = true;
  128. if (helper.getEncodedColorModel().getPixelSize() > maxPixelSize) {
  129. directEncode = false; //pixel size needs to be reduced
  130. }
  131. if (BitmapImageUtil.getColorIndexSize(renderedImage) > 2) {
  132. directEncode = false; //Lookup tables are not implemented, yet
  133. }
  134. if (useFS10
  135. && BitmapImageUtil.isMonochromeImage(renderedImage)
  136. && BitmapImageUtil.isZeroBlack(renderedImage)) {
  137. directEncode = false;
  138. }
  139. if (directEncode) {
  140. log.debug("Encoding image directly...");
  141. imageObjectInfo.setBitsPerPixel(encodedColorModel.getPixelSize());
  142. if (BitmapImageUtil.isMonochromeImage(renderedImage)
  143. && BitmapImageUtil.isZeroBlack(renderedImage)) {
  144. log.trace("set subtractive mode");
  145. imageObjectInfo.setSubtractive(true);
  146. }
  147. helper.encode(baos);
  148. imageData = baos.toByteArray();
  149. }
  150. }
  151. if (imageData == null) {
  152. log.debug("Encoding image via RGB...");
  153. //Convert image to 24bit RGB
  154. ImageEncodingHelper.encodeRenderedImageAsRGB(renderedImage, baos);
  155. imageData = baos.toByteArray();
  156. boolean colorImages = paintingState.isColorImages();
  157. imageObjectInfo.setColor(colorImages);
  158. // convert to grayscale
  159. if (!colorImages) {
  160. log.debug("Converting RGB image to grayscale...");
  161. baos.reset();
  162. int bitsPerPixel = paintingState.getBitsPerPixel();
  163. imageObjectInfo.setBitsPerPixel(bitsPerPixel);
  164. //TODO this should be done off the RenderedImage to avoid buffering the
  165. //intermediate 24bit image
  166. ImageEncodingHelper.encodeRGBAsGrayScale(
  167. imageData, dataWidth, dataHeight, bitsPerPixel, baos);
  168. imageData = baos.toByteArray();
  169. if (bitsPerPixel == 1) {
  170. imageObjectInfo.setSubtractive(true);
  171. }
  172. }
  173. }
  174. imageObjectInfo.setData(imageData);
  175. // set object area info
  176. AFPObjectAreaInfo objectAreaInfo = imageObjectInfo.getObjectAreaInfo();
  177. objectAreaInfo.setWidthRes(resolution);
  178. objectAreaInfo.setHeightRes(resolution);
  179. return imageObjectInfo;
  180. }
  181. /** {@inheritDoc} */
  182. protected AFPDataObjectInfo createDataObjectInfo() {
  183. return new AFPImageObjectInfo();
  184. }
  185. /** {@inheritDoc} */
  186. public int getPriority() {
  187. return 300;
  188. }
  189. /** {@inheritDoc} */
  190. public Class getSupportedImageClass() {
  191. return ImageRendered.class;
  192. }
  193. /** {@inheritDoc} */
  194. public ImageFlavor[] getSupportedImageFlavors() {
  195. return FLAVORS;
  196. }
  197. }