#region PDFsharp - A .NET library for processing PDF // // Authors: // Stefan Lange // // Copyright (c) 2005-2017 empira Software GmbH, Cologne Area (Germany) // // http://www.pdfsharp.com // http://sourceforge.net/projects/pdfsharp // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included // in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. #endregion using PdfSharp.Drawing; using PdfSharp.Pdf.Filters; using PdfSharp.Fonts.OpenType; namespace PdfSharp.Pdf.Advanced { /// /// Represents a CIDFont dictionary. /// internal class PdfCIDFont : PdfFont { public PdfCIDFont(PdfDocument document) : base(document) { } public PdfCIDFont(PdfDocument document, PdfFontDescriptor fontDescriptor, XFont font) : base(document) { Elements.SetName(Keys.Type, "/Font"); Elements.SetName(Keys.Subtype, "/CIDFontType2"); PdfDictionary cid = new PdfDictionary(); cid.Elements.SetString("/Ordering", "Identity"); cid.Elements.SetString("/Registry", "Adobe"); cid.Elements.SetInteger("/Supplement", 0); Elements.SetValue(Keys.CIDSystemInfo, cid); FontDescriptor = fontDescriptor; // ReSharper disable once DoNotCallOverridableMethodsInConstructor Owner._irefTable.Add(fontDescriptor); Elements[Keys.FontDescriptor] = fontDescriptor.Reference; FontEncoding = font.PdfOptions.FontEncoding; } public PdfCIDFont(PdfDocument document, PdfFontDescriptor fontDescriptor, byte[] fontData) : base(document) { Elements.SetName(Keys.Type, "/Font"); Elements.SetName(Keys.Subtype, "/CIDFontType2"); PdfDictionary cid = new PdfDictionary(); cid.Elements.SetString("/Ordering", "Identity"); cid.Elements.SetString("/Registry", "Adobe"); cid.Elements.SetInteger("/Supplement", 0); Elements.SetValue(Keys.CIDSystemInfo, cid); FontDescriptor = fontDescriptor; // ReSharper disable once DoNotCallOverridableMethodsInConstructor Owner._irefTable.Add(fontDescriptor); Elements[Keys.FontDescriptor] = fontDescriptor.Reference; FontEncoding = PdfFontEncoding.Unicode; } public string BaseFont { get { return Elements.GetName(Keys.BaseFont); } set { Elements.SetName(Keys.BaseFont, value); } } /// /// Prepares the object to get saved. /// internal override void PrepareForSave() { base.PrepareForSave(); #if DEBUG_ if (FontDescriptor._descriptor.FontFace.loca == null) { GetType(); } #endif // CID fonts must be always embedded. PDFsharp embeds automatically a subset. OpenTypeFontface subSet = null; if (FontDescriptor._descriptor.FontFace.loca == null) subSet = FontDescriptor._descriptor.FontFace; else subSet = FontDescriptor._descriptor.FontFace.CreateFontSubSet(_cmapInfo.GlyphIndices, true); byte[] fontData = subSet.FontSource.Bytes; PdfDictionary fontStream = new PdfDictionary(Owner); Owner.Internals.AddObject(fontStream); FontDescriptor.Elements[PdfFontDescriptor.Keys.FontFile2] = fontStream.Reference; fontStream.Elements["/Length1"] = new PdfInteger(fontData.Length); if (!Owner.Options.NoCompression) { fontData = Filtering.FlateDecode.Encode(fontData, _document.Options.FlateEncodeMode); fontStream.Elements["/Filter"] = new PdfName("/FlateDecode"); } fontStream.Elements["/Length"] = new PdfInteger(fontData.Length); fontStream.CreateStream(fontData); } /// /// Predefined keys of this dictionary. /// public new sealed class Keys : PdfFont.Keys { /// /// (Required) The type of PDF object that this dictionary describes; /// must be Font for a CIDFont dictionary. /// [KeyInfo(KeyType.Name | KeyType.Required, FixedValue = "Font")] public new const string Type = "/Type"; /// /// (Required) The type of CIDFont; CIDFontType0 or CIDFontType2. /// [KeyInfo(KeyType.Name | KeyType.Required)] public new const string Subtype = "/Subtype"; /// /// (Required) The PostScript name of the CIDFont. For Type 0 CIDFonts, this /// is usually the value of the CIDFontName entry in the CIDFont program. For /// Type 2 CIDFonts, it is derived the same way as for a simple TrueType font; /// In either case, the name can have a subset prefix if appropriate. /// [KeyInfo(KeyType.Name | KeyType.Required)] public new const string BaseFont = "/BaseFont"; /// /// (Required) A dictionary containing entries that define the character collection /// of the CIDFont. /// [KeyInfo(KeyType.Dictionary | KeyType.Required)] public const string CIDSystemInfo = "/CIDSystemInfo"; /// /// (Required; must be an indirect reference) A font descriptor describing the /// CIDFont’s default metrics other than its glyph widths. /// [KeyInfo(KeyType.Dictionary | KeyType.MustBeIndirect, typeof(PdfFontDescriptor))] public new const string FontDescriptor = "/FontDescriptor"; /// /// (Optional) The default width for glyphs in the CIDFont. /// Default value: 1000. /// [KeyInfo(KeyType.Integer)] public const string DW = "/DW"; /// /// (Optional) A description of the widths for the glyphs in the CIDFont. The /// array’s elements have a variable format that can specify individual widths /// for consecutive CIDs or one width for a range of CIDs. /// Default value: none (the DW value is used for all glyphs). /// [KeyInfo(KeyType.Array, typeof(PdfArray))] public const string W = "/W"; /// /// (Optional; applies only to CIDFonts used for vertical writing) An array of two /// numbers specifying the default metrics for vertical writing. /// Default value: [880 −1000]. /// [KeyInfo(KeyType.Array)] public const string DW2 = "/DW2"; /// /// (Optional; applies only to CIDFonts used for vertical writing) A description /// of the metrics for vertical writing for the glyphs in the CIDFont. /// Default value: none (the DW2 value is used for all glyphs). /// [KeyInfo(KeyType.Array, typeof(PdfArray))] public const string W2 = "/W2"; /// /// (Optional; Type 2 CIDFonts only) A specification of the mapping from CIDs /// to glyph indices. If the value is a stream, the bytes in the stream contain the /// mapping from CIDs to glyph indices: the glyph index for a particular CID /// value c is a 2-byte value stored in bytes 2 × c and 2 × c + 1, where the first /// byte is the high-order byte. If the value of CIDToGIDMap is a name, it must /// be Identity, indicating that the mapping between CIDs and glyph indices is /// the identity mapping. /// Default value: Identity. /// This entry may appear only in a Type 2 CIDFont whose associated True-Type font /// program is embedded in the PDF file. /// [KeyInfo(KeyType.Dictionary | KeyType.StreamOrName)] public const string CIDToGIDMap = "/CIDToGIDMap"; /// /// Gets the KeysMeta for these keys. /// internal static DictionaryMeta Meta { get { return _meta ?? (_meta = CreateMeta(typeof(Keys))); } } static DictionaryMeta _meta; } /// /// Gets the KeysMeta of this dictionary type. /// internal override DictionaryMeta Meta { get { return Keys.Meta; } } } }