KickJava   Java API By Example, From Geeks To Geeks.

Java > Open Source Codes > com > lowagie > text > pdf > PdfNameTree


1 /*
2  * Copyright 2004 by Paulo Soares.
3  *
4  * The contents of this file are subject to the Mozilla Public License Version 1.1
5  * (the "License"); you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at http://www.mozilla.org/MPL/
7  *
8  * Software distributed under the License is distributed on an "AS IS" basis,
9  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
10  * for the specific language governing rights and limitations under the License.
11  *
12  * The Original Code is 'iText, a free JAVA-PDF library'.
13  *
14  * The Initial Developer of the Original Code is Bruno Lowagie. Portions created by
15  * the Initial Developer are Copyright (C) 1999, 2000, 2001, 2002 by Bruno Lowagie.
16  * All Rights Reserved.
17  * Co-Developer of the code is Paulo Soares. Portions created by the Co-Developer
18  * are Copyright (C) 2000, 2001, 2002 by Paulo Soares. All Rights Reserved.
19  *
20  * Contributor(s): all the names of the contributors are added in the source code
21  * where applicable.
22  *
23  * Alternatively, the contents of this file may be used under the terms of the
24  * LGPL license (the "GNU LIBRARY GENERAL PUBLIC LICENSE"), in which case the
25  * provisions of LGPL are applicable instead of those above. If you wish to
26  * allow use of your version of this file only under the terms of the LGPL
27  * License and not to allow others to use your version of this file under
28  * the MPL, indicate your decision by deleting the provisions above and
29  * replace them with the notice and other provisions required by the LGPL.
30  * If you do not delete the provisions above, a recipient may use your version
31  * of this file under either the MPL or the GNU LIBRARY GENERAL PUBLIC LICENSE.
32  *
33  * This library is free software; you can redistribute it and/or modify it
34  * under the terms of the MPL as stated above or under the terms of the GNU
35  * Library General Public License as published by the Free Software Foundation;
36  * either version 2 of the License, or any later version.
37  *
38  * This library is distributed in the hope that it will be useful, but WITHOUT
39  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
40  * FOR A PARTICULAR PURPOSE. See the GNU Library general Public License for more
41  * details.
42  *
43  * If you didn't download this code from the following link, you should check if
44  * you aren't using an obsolete version:
45  * http://www.lowagie.com/iText/
46  */

47 package com.lowagie.text.pdf;
48
49 import java.io.IOException JavaDoc;
50 import java.util.ArrayList JavaDoc;
51 import java.util.Arrays JavaDoc;
52 import java.util.HashMap JavaDoc;
53
54 /**
55  * Creates a name tree.
56  * @author Paulo Soares (psoares@consiste.pt)
57  */

58 public class PdfNameTree {
59     
60     private static final int leafSize = 64;
61     
62     /**
63      * Creates a name tree.
64      * @param items the item of the name tree. The key is a <CODE>String</CODE>
65      * and the value is a <CODE>PdfObject</CODE>. Note that although the
66      * keys are strings only the lower byte is used and no check is made for chars
67      * with the same lower byte and different upper byte. This will generate a wrong
68      * tree name.
69      * @param writer the writer
70      * @throws IOException on error
71      * @return the dictionary with the name tree. This dictionary is the one
72      * generally pointed to by the key /Dests, for example
73      */

74     public static PdfDictionary writeTree(HashMap JavaDoc items, PdfWriter writer) throws IOException JavaDoc {
75         if (items.isEmpty())
76             return null;
77         String JavaDoc names[] = new String JavaDoc[items.size()];
78         names = (String JavaDoc[])items.keySet().toArray(names);
79         Arrays.sort(names);
80         if (names.length <= leafSize) {
81             PdfDictionary dic = new PdfDictionary();
82             PdfArray ar = new PdfArray();
83             for (int k = 0; k < names.length; ++k) {
84                 ar.add(new PdfString(names[k], null));
85                 ar.add((PdfObject)items.get(names[k]));
86             }
87             dic.put(PdfName.NAMES, ar);
88             return dic;
89         }
90         int skip = leafSize;
91         PdfIndirectReference kids[] = new PdfIndirectReference[(names.length + leafSize - 1) / leafSize];
92         for (int k = 0; k < kids.length; ++k) {
93             int offset = k * leafSize;
94             int end = Math.min(offset + leafSize, names.length);
95             PdfDictionary dic = new PdfDictionary();
96             PdfArray arr = new PdfArray();
97             arr.add(new PdfString(names[offset], null));
98             arr.add(new PdfString(names[end - 1], null));
99             dic.put(PdfName.LIMITS, arr);
100             arr = new PdfArray();
101             for (; offset < end; ++offset) {
102                 arr.add(new PdfString(names[offset], null));
103                 arr.add((PdfObject)items.get(names[offset]));
104             }
105             dic.put(PdfName.NAMES, arr);
106             kids[k] = writer.addToBody(dic).getIndirectReference();
107         }
108         int top = kids.length;
109         while (true) {
110             if (top <= leafSize) {
111                 PdfArray arr = new PdfArray();
112                 for (int k = 0; k < top; ++k)
113                     arr.add(kids[k]);
114                 PdfDictionary dic = new PdfDictionary();
115                 dic.put(PdfName.KIDS, arr);
116                 return dic;
117             }
118             skip *= leafSize;
119             int tt = (names.length + skip - 1 )/ skip;
120             for (int k = 0; k < tt; ++k) {
121                 int offset = k * leafSize;
122                 int end = Math.min(offset + leafSize, top);
123                 PdfDictionary dic = new PdfDictionary();
124                 PdfArray arr = new PdfArray();
125                 arr.add(new PdfString(names[k * skip], null));
126                 arr.add(new PdfString(names[Math.min((k + 1) * skip, names.length) - 1], null));
127                 dic.put(PdfName.LIMITS, arr);
128                 arr = new PdfArray();
129                 for (; offset < end; ++offset) {
130                     arr.add(kids[offset]);
131                 }
132                 dic.put(PdfName.KIDS, arr);
133                 kids[k] = writer.addToBody(dic).getIndirectReference();
134             }
135             top = tt;
136         }
137     }
138     
139     private static void iterateItems(PdfDictionary dic, HashMap JavaDoc items) {
140         PdfArray nn = (PdfArray)PdfReader.getPdfObjectRelease(dic.get(PdfName.NAMES));
141         if (nn != null) {
142             ArrayList JavaDoc arr = nn.getArrayList();
143             for (int k = 0; k < arr.size(); ++k) {
144                 PdfString s = (PdfString)PdfReader.getPdfObjectRelease((PdfObject)arr.get(k++));
145                 items.put(PdfEncodings.convertToString(s.getBytes(), null), arr.get(k));
146             }
147         }
148         else if ((nn = (PdfArray)PdfReader.getPdfObjectRelease(dic.get(PdfName.KIDS))) != null) {
149             ArrayList JavaDoc arr = nn.getArrayList();
150             for (int k = 0; k < arr.size(); ++k) {
151                 PdfDictionary kid = (PdfDictionary)PdfReader.getPdfObjectRelease((PdfObject)arr.get(k));
152                 iterateItems(kid, items);
153             }
154         }
155     }
156     
157     public static HashMap JavaDoc readTree(PdfDictionary dic) {
158         HashMap JavaDoc items = new HashMap JavaDoc();
159         if (dic != null)
160             iterateItems(dic, items);
161         return items;
162     }
163 }
Popular Tags