output.dart
3.59 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
/*
* Copyright (C) 2017, David PHAM-VAN <dev.nfet.net@gmail.com>
*
* 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.
*/
part of pdf;
class PdfOutput {
/// This is the actual [PdfStream] used to write to.
final PdfStream os;
/// This vector contains offsets of each object
List<PdfXref> offsets = [];
/// This is used to track the /Root object (catalog)
PdfObject rootID;
/// This is used to track the /Info object (info)
PdfObject infoID;
/// This creates a Pdf [PdfStream]
///
/// @param os The output stream to write the Pdf file to.
PdfOutput(this.os) {
os.putString("%PDF-1.4\n");
os.putBytes([0x25, 0xC2, 0xA5, 0xC2, 0xB1, 0xC3, 0xAB, 0x0A]);
}
/// This method writes a [PdfObject] to the stream.
///
/// @param ob [PdfObject] Object to write
void write(PdfObject ob) {
// Check the object to see if it's one that is needed in the trailer
// object
if (ob is PdfCatalog) rootID = ob;
if (ob is PdfInfo) infoID = ob;
offsets.add(PdfXref(ob.objser, os.offset));
ob._write(os);
}
/// This closes the Stream, writing the xref table
void close() {
int xref = os.offset;
os.putString("xref\n");
// Now a single subsection for object 0
//os.write("0 1\n0000000000 65535 f \n");
// Now scan through the offsets list. The should be in sequence,
// but just in case:
int firstid = 0; // First id in block
int lastid = -1; // The last id used
var block = []; // xrefs in this block
// We need block 0 to exist
block.add(PdfXref(0, 0, generation: 65535));
for (PdfXref x in offsets) {
if (firstid == -1) firstid = x.id;
// check to see if block is in range (-1 means empty)
if (lastid > -1 && x.id != (lastid + 1)) {
// no, so write this block, and reset
writeblock(firstid, block);
block = [];
firstid = -1;
}
// now add to block
block.add(x);
lastid = x.id;
}
// now write the last block
if (firstid > -1) writeblock(firstid, block);
// now the trailer object
os.putString("trailer\n<<\n");
// the number of entries (REQUIRED)
os.putString("/Size ");
os.putString((offsets.length + 1).toString());
os.putString("\n");
// the /Root catalog indirect reference (REQUIRED)
if (rootID != null) {
os.putString("/Root ");
os.putStream(rootID.ref());
os.putString("\n");
} else
throw Exception("Root object is not present in document");
// the /Info reference (OPTIONAL)
if (infoID != null) {
os.putString("/Info ");
os.putStream(infoID.ref());
os.putString("\n");
}
// end the trailer object
os.putString(">>\nstartxref\n$xref\n%%EOF\n");
}
/// Writes a block of references to the Pdf file
/// @param firstid ID of the first reference in this block
/// @param block Vector containing the references in this block
void writeblock(int firstid, var block) {
os.putString("$firstid ${block.length}\n");
//os.write("\n0000000000 65535 f\n");
for (PdfXref x in block) {
os.putString(x.ref());
os.putString("\n");
}
}
}