/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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. */ /* $Id$ */ package org.apache.fop.pdf; import java.io.IOException; import java.io.OutputStream; import java.util.LinkedHashMap; import java.util.LinkedHashSet; import java.util.Map; import java.util.Set; import org.apache.xmlgraphics.java2d.color.profile.ColorProfileUtil; import org.apache.fop.fonts.FontDescriptor; import org.apache.fop.fonts.FontInfo; import org.apache.fop.fonts.Typeface; import org.apache.fop.fonts.base14.Symbol; import org.apache.fop.fonts.base14.ZapfDingbats; /** * Class representing a /Resources object. * * /Resources object contain a list of references to the fonts, patterns, * shadings, etc., for the document. */ public class PDFResources extends PDFDictionary { /** * /Font objects keyed by their internal name */ protected Map fonts = new LinkedHashMap(); /** * Set of XObjects */ protected Set xObjects = new LinkedHashSet(); /** * Set of patterns */ protected Set patterns = new LinkedHashSet(); /** * Set of shadings */ protected Set shadings = new LinkedHashSet(); /** * Set of ExtGStates */ protected Set gstates = new LinkedHashSet(); /** Map of color spaces (key: color space name) */ protected Map colorSpaces = new LinkedHashMap(); /** Map of ICC color spaces (key: ICC profile description) */ protected Map iccColorSpaces = new LinkedHashMap(); /** Named properties */ protected Map properties = new LinkedHashMap(); /** * create a /Resources object. * * @param objnum the object's number */ public PDFResources(int objnum) { /* generic creation of object */ super(); setObjectNumber(objnum); } /** * add font object to resources list. * * @param font the PDFFont to add */ public void addFont(PDFFont font) { this.fonts.put(font.getName(), font); } /** * Add the fonts in the font info to this PDF document's Font Resources. * * @param doc PDF document to add fonts to * @param fontInfo font info object to get font information from */ public void addFonts(PDFDocument doc, FontInfo fontInfo) { Map usedFonts = fontInfo.getUsedFonts(); for (String f : usedFonts.keySet()) { Typeface font = usedFonts.get(f); //Check if the font actually had any mapping operations. If not, it is an indication //that it has never actually been used and therefore doesn't have to be embedded. if (font.hadMappingOperations()) { FontDescriptor desc = null; if (font instanceof FontDescriptor) { desc = (FontDescriptor)font; } String encoding = font.getEncodingName(); if (font instanceof Symbol || font instanceof ZapfDingbats) { encoding = null; //Symbolic fonts shouldn't specify an encoding value in PDF } addFont(doc.getFactory().makeFont( f, font.getEmbedFontName(), encoding, font, desc)); } } } /** * Add a PDFGState to the resources. * * @param gs the PDFGState to add */ public void addGState(PDFGState gs) { this.gstates.add(gs); } /** * Add a Shading to the resources. * * @param theShading the shading to add */ public void addShading(PDFShading theShading) { this.shadings.add(theShading); } /** * Add the pattern to the resources. * * @param thePattern the pattern to add */ public void addPattern(PDFPattern thePattern) { this.patterns.add(thePattern); } /** * Add an XObject to the resources. * * @param xObject the XObject to add */ public void addXObject(PDFXObject xObject) { this.xObjects.add(xObject); } /** * Add a ColorSpace dictionary to the resources. * @param colorSpace the color space */ public void addColorSpace(PDFColorSpace colorSpace) { this.colorSpaces.put(new PDFName(colorSpace.getName()), colorSpace); if (colorSpace instanceof PDFICCBasedColorSpace) { PDFICCBasedColorSpace icc = (PDFICCBasedColorSpace)colorSpace; String desc = ColorProfileUtil.getICCProfileDescription( icc.getICCStream().getICCProfile()); this.iccColorSpaces.put(desc, icc); } } /** * Returns a ICCBased color space by profile name. * @param desc the name of the color space * @return the requested color space or null if it wasn't found */ public PDFICCBasedColorSpace getICCColorSpaceByProfileName(String desc) { PDFICCBasedColorSpace cs = this.iccColorSpaces.get(desc); return cs; } /** * Returns a color space by name. * @param name the name of the color space * @return the requested color space or null if it wasn't found */ public PDFColorSpace getColorSpace(PDFName name) { PDFColorSpace cs = this.colorSpaces.get(name); return cs; } /** * Add a named property. * * @param name name of property * @param property reference to property value */ public void addProperty(String name, PDFReference property) { this.properties.put(name, property); } /** * Get a named property. * * @param name name of property */ public PDFReference getProperty(String name) { return this.properties.get(name); } @Override public int output(OutputStream stream) throws IOException { populateDictionary(); return super.output(stream); } private void populateDictionary() { if (!this.fonts.isEmpty()) { PDFDictionary dict = new PDFDictionary(this); /* construct PDF dictionary of font object references */ for (Map.Entry entry : fonts.entrySet()) { dict.put(entry.getKey(), entry.getValue()); } put("Font", dict); } if (!this.shadings.isEmpty()) { PDFDictionary dict = new PDFDictionary(this); for (PDFShading shading : shadings) { dict.put(shading.getName(), shading); } put("Shading", dict); } if (!this.patterns.isEmpty()) { PDFDictionary dict = new PDFDictionary(this); for (PDFPattern pattern : patterns) { dict.put(pattern.getName(), pattern); } put("Pattern", dict); } PDFArray procset = new PDFArray(this); procset.add(new PDFName("PDF")); procset.add(new PDFName("ImageB")); procset.add(new PDFName("ImageC")); procset.add(new PDFName("Text")); put("ProcSet", procset); if (this.xObjects != null && !this.xObjects.isEmpty()) { PDFDictionary dict = new PDFDictionary(this); for (PDFXObject xObject : xObjects) { dict.put(xObject.getName().toString(), xObject); } put("XObject", dict); } if (!this.gstates.isEmpty()) { PDFDictionary dict = new PDFDictionary(this); for (PDFGState gstate : gstates) { dict.put(gstate.getName(), gstate); } put("ExtGState", dict); } if (!this.colorSpaces.isEmpty()) { PDFDictionary dict = new PDFDictionary(this); for (PDFColorSpace colorSpace : colorSpaces.values()) { dict.put(colorSpace.getName(), colorSpace); } put("ColorSpace", dict); } if (!properties.isEmpty()) { PDFDictionary dict = new PDFDictionary(this); for (String name : properties.keySet()) { dict.put(name, properties.get(name)); } put("Properties", dict); } } }