blob: d5641b2847a15e2e94f126cdf3e943e297eede96 [file] [log] [blame]
edisonn@google.com1be794f2013-06-21 21:43:09 +00001#ifndef __DEFINED__SkPdfFont
2#define __DEFINED__SkPdfFont
3
4#include "SkPdfHeaders_autogen.h"
edisonn@google.com3aac1f92013-07-02 22:42:53 +00005#include "SkPdfMapper_autogen.h"
edisonn@google.com1be794f2013-06-21 21:43:09 +00006
7#include <map>
8#include <string>
9
edisonn@google.com131d4ee2013-06-26 17:48:12 +000010#include "SkTypeface.h"
edisonn@google.com1be794f2013-06-21 21:43:09 +000011#include "SkUtils.h"
edisonn@google.comb857a0c2013-06-25 20:45:40 +000012#include "SkPdfBasics.h"
13#include "SkPdfUtils.h"
14
edisonn@google.com1be794f2013-06-21 21:43:09 +000015
16class SkPdfType0Font;
17class SkPdfType1Font;
18class SkPdfType3Font;
19class SkPdfTrueTypeFont;
edisonn@google.com1be794f2013-06-21 21:43:09 +000020class SkPdfMultiMasterFont;
21class SkPdfFont;
22
edisonn@google.com1be794f2013-06-21 21:43:09 +000023struct SkPdfStandardFontEntry {
edisonn@google.com596d2e22013-07-10 17:44:55 +000024 // We don't own this pointer!
edisonn@google.com1be794f2013-06-21 21:43:09 +000025 const char* fName;
26 bool fIsBold;
27 bool fIsItalic;
edisonn@google.com596d2e22013-07-10 17:44:55 +000028 SkPdfStandardFontEntry()
29 : fName(NULL),
30 fIsBold(false),
31 fIsItalic(false) {}
32
33 SkPdfStandardFontEntry(const char* name, bool bold, bool italic)
34 : fName(name),
35 fIsBold(bold),
36 fIsItalic(italic) {}
edisonn@google.com1be794f2013-06-21 21:43:09 +000037};
38
39std::map<std::string, SkPdfStandardFontEntry>& getStandardFonts();
40SkTypeface* SkTypefaceFromPdfStandardFont(const char* fontName, bool bold, bool italic);
edisonn@google.com571c70b2013-07-10 17:09:50 +000041SkPdfFont* fontFromName(SkNativeParsedPDF* doc, SkPdfObject* obj, const char* fontName);
edisonn@google.com1be794f2013-06-21 21:43:09 +000042
43struct SkUnencodedText {
44 void* text;
45 int len;
46
47public:
edisonn@google.com3aac1f92013-07-02 22:42:53 +000048 SkUnencodedText(const SkPdfString* obj) {
49 text = (void*)obj->c_str();
edisonn@google.come878e722013-07-29 19:10:58 +000050 len = obj->lenstr();
edisonn@google.com1be794f2013-06-21 21:43:09 +000051 }
52};
53
54struct SkDecodedText {
55 uint16_t* text;
56 int len;
edisonn@google.comb857a0c2013-06-25 20:45:40 +000057public:
58 unsigned int operator[](int i) const { return text[i]; }
59 int size() const { return len; }
edisonn@google.com1be794f2013-06-21 21:43:09 +000060};
61
62struct SkUnicodeText {
63 uint16_t* text;
64 int len;
65
66public:
67 unsigned int operator[](int i) const { return text[i]; }
68 int size() const { return len; }
69};
70
71class SkPdfEncoding {
72public:
73 virtual bool decodeText(const SkUnencodedText& textIn, SkDecodedText* textOut) const = 0;
edisonn@google.comb857a0c2013-06-25 20:45:40 +000074 static SkPdfEncoding* fromName(const char* name);
edisonn@google.com1be794f2013-06-21 21:43:09 +000075};
76
edisonn@google.comb857a0c2013-06-25 20:45:40 +000077std::map<std::string, SkPdfEncoding*>& getStandardEncodings();
78
79class SkPdfToUnicode {
edisonn@google.com571c70b2013-07-10 17:09:50 +000080 SkNativeParsedPDF* fParsed;
edisonn@google.comb857a0c2013-06-25 20:45:40 +000081 // TODO(edisonn): hide public members
82public:
83 unsigned short* fCMapEncoding;
84 unsigned char* fCMapEncodingFlag;
85
edisonn@google.com571c70b2013-07-10 17:09:50 +000086 SkPdfToUnicode(SkNativeParsedPDF* parsed, SkPdfStream* stream);
edisonn@google.comb857a0c2013-06-25 20:45:40 +000087};
88
89
edisonn@google.com1be794f2013-06-21 21:43:09 +000090class SkPdfIdentityHEncoding : public SkPdfEncoding {
91public:
92 virtual bool decodeText(const SkUnencodedText& textIn, SkDecodedText* textOut) const {
93 // TODO(edisonn): SkASSERT(textIn.len % 2 == 0); or report error?
94
95 uint16_t* text = (uint16_t*)textIn.text;
96 textOut->text = new uint16_t[textIn.len / 2];
97 textOut->len = textIn.len / 2;
98
99 for (int i = 0; i < textOut->len; i++) {
100 textOut->text[i] = ((text[i] << 8) & 0xff00) | ((text[i] >> 8) & 0x00ff);
101 }
102
103 return true;
104 }
105
106 static SkPdfIdentityHEncoding* instance() {
107 static SkPdfIdentityHEncoding* inst = new SkPdfIdentityHEncoding();
108 return inst;
109 }
110};
111
edisonn@google.com6e49c342013-06-27 20:03:43 +0000112// TODO(edisonn): using this one when no encoding is specified
113class SkPdfDefaultEncoding : public SkPdfEncoding {
114public:
115 virtual bool decodeText(const SkUnencodedText& textIn, SkDecodedText* textOut) const {
116 // TODO(edisonn): SkASSERT(textIn.len % 2 == 0); or report error?
117
118 unsigned char* text = (unsigned char*)textIn.text;
119 textOut->text = new uint16_t[textIn.len];
120 textOut->len = textIn.len;
121
122 for (int i = 0; i < textOut->len; i++) {
123 textOut->text[i] = text[i];
124 }
125
126 return true;
127 }
128
129 static SkPdfDefaultEncoding* instance() {
130 static SkPdfDefaultEncoding* inst = new SkPdfDefaultEncoding();
131 return inst;
132 }
133};
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000134
135class SkPdfCIDToGIDMapIdentityEncoding : public SkPdfEncoding {
136public:
137 virtual bool decodeText(const SkUnencodedText& textIn, SkDecodedText* textOut) const {
138 // TODO(edisonn): SkASSERT(textIn.len % 2 == 0); or report error?
139
edisonn@google.com571c70b2013-07-10 17:09:50 +0000140 uint16_t* text = (uint16_t*)textIn.text;
141 textOut->text = new uint16_t[textIn.len / 2];
142 textOut->len = textIn.len / 2;
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000143
144 for (int i = 0; i < textOut->len; i++) {
edisonn@google.com571c70b2013-07-10 17:09:50 +0000145 textOut->text[i] = ((text[i] << 8) & 0xff00) | ((text[i] >> 8) & 0x00ff);
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000146 }
147
148 return true;
149 }
150
151 static SkPdfCIDToGIDMapIdentityEncoding* instance() {
152 static SkPdfCIDToGIDMapIdentityEncoding* inst = new SkPdfCIDToGIDMapIdentityEncoding();
153 return inst;
154 }
155};
156
edisonn@google.com1be794f2013-06-21 21:43:09 +0000157class SkPdfFont {
158public:
159 SkPdfFont* fBaseFont;
160 SkPdfEncoding* fEncoding;
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000161 SkPdfToUnicode* fToUnicode;
162
edisonn@google.com1be794f2013-06-21 21:43:09 +0000163
164public:
edisonn@google.com6e49c342013-06-27 20:03:43 +0000165 SkPdfFont() : fBaseFont(NULL), fEncoding(SkPdfDefaultEncoding::instance()), fToUnicode(NULL) {}
edisonn@google.com1be794f2013-06-21 21:43:09 +0000166
edisonn@google.com7ee12ca2013-08-05 18:18:06 +0000167 virtual ~SkPdfFont() {
168 // TODO(edisonn): NYI (will leak for now)
169 }
170
edisonn@google.com1be794f2013-06-21 21:43:09 +0000171 const SkPdfEncoding* encoding() const {return fEncoding;}
172
edisonn@google.com6e49c342013-06-27 20:03:43 +0000173 void drawText(const SkDecodedText& text, SkPaint* paint, PdfContext* pdfContext, SkCanvas* canvas) {
edisonn@google.com1be794f2013-06-21 21:43:09 +0000174 for (int i = 0 ; i < text.size(); i++) {
edisonn@google.com6e49c342013-06-27 20:03:43 +0000175 double width = drawOneChar(text[i], paint, pdfContext, canvas);
176 pdfContext->fGraphicsState.fMatrixTm.preTranslate(SkDoubleToScalar(width), SkDoubleToScalar(0.0));
177 canvas->translate(SkDoubleToScalar(width), SkDoubleToScalar(0.0));
edisonn@google.com1be794f2013-06-21 21:43:09 +0000178 }
179 }
180
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000181 void ToUnicode(const SkDecodedText& textIn, SkUnicodeText* textOut) const {
182 if (fToUnicode) {
183 textOut->text = new uint16_t[textIn.len];
184 textOut->len = textIn.len;
185 for (int i = 0; i < textIn.len; i++) {
186 textOut->text[i] = fToUnicode->fCMapEncoding[textIn.text[i]];
187 }
188 } else {
189 textOut->text = textIn.text;
190 textOut->len = textIn.len;
191 }
192 };
193
194 inline unsigned int ToUnicode(unsigned int ch) const {
195 if (fToUnicode) {
196 return fToUnicode->fCMapEncoding[ch];
197 } else {
198 return ch;
199 }
edisonn@google.com1be794f2013-06-21 21:43:09 +0000200 };
201
edisonn@google.com571c70b2013-07-10 17:09:50 +0000202 static SkPdfFont* fontFromPdfDictionary(SkNativeParsedPDF* doc, SkPdfFontDictionary* dict);
203 static SkPdfFont* Default() {return fontFromName(NULL, NULL, "TimesNewRoman");}
edisonn@google.com1be794f2013-06-21 21:43:09 +0000204
edisonn@google.com571c70b2013-07-10 17:09:50 +0000205 static SkPdfType0Font* fontFromType0FontDictionary(SkNativeParsedPDF* doc, SkPdfType0FontDictionary* dict);
206 static SkPdfType1Font* fontFromType1FontDictionary(SkNativeParsedPDF* doc, SkPdfType1FontDictionary* dict);
207 static SkPdfType3Font* fontFromType3FontDictionary(SkNativeParsedPDF* doc, SkPdfType3FontDictionary* dict);
208 static SkPdfTrueTypeFont* fontFromTrueTypeFontDictionary(SkNativeParsedPDF* doc, SkPdfTrueTypeFontDictionary* dict);
209 static SkPdfMultiMasterFont* fontFromMultiMasterFontDictionary(SkNativeParsedPDF* doc, SkPdfMultiMasterFontDictionary* dict);
edisonn@google.com1be794f2013-06-21 21:43:09 +0000210
edisonn@google.com571c70b2013-07-10 17:09:50 +0000211 static SkPdfFont* fontFromFontDescriptor(SkNativeParsedPDF* doc, SkPdfFontDescriptorDictionary* fd, bool loadFromName = true);
edisonn@google.com6e49c342013-06-27 20:03:43 +0000212
edisonn@google.com1be794f2013-06-21 21:43:09 +0000213public:
edisonn@google.com6e49c342013-06-27 20:03:43 +0000214 virtual double drawOneChar(unsigned int ch, SkPaint* paint, PdfContext* pdfContext, SkCanvas* canvas) = 0;
edisonn@google.com1be794f2013-06-21 21:43:09 +0000215 virtual void afterWord(SkPaint* paint, SkMatrix* matrix) = 0;
edisonn@google.com571c70b2013-07-10 17:09:50 +0000216
217private:
218 static SkPdfFont* fontFromPdfDictionaryOnce(SkNativeParsedPDF* doc, SkPdfFontDictionary* dict);
edisonn@google.com1be794f2013-06-21 21:43:09 +0000219};
220
221class SkPdfStandardFont : public SkPdfFont {
222 SkTypeface* fTypeface;
223
224public:
225 SkPdfStandardFont(SkTypeface* typeface) : fTypeface(typeface) {}
226
227public:
edisonn@google.com6e49c342013-06-27 20:03:43 +0000228 virtual double drawOneChar(unsigned int ch, SkPaint* paint, PdfContext* pdfContext, SkCanvas* canvas) {
edisonn@google.com1be794f2013-06-21 21:43:09 +0000229 paint->setTypeface(fTypeface);
230 paint->setTextEncoding(SkPaint::kUTF8_TextEncoding);
231
232 unsigned long ch4 = ch;
233 char utf8[10];
234 int len = SkUTF8_FromUnichar(ch4, utf8);
235
236 canvas->drawText(utf8, len, SkDoubleToScalar(0), SkDoubleToScalar(0), *paint);
237
238 SkScalar textWidth = paint->measureText(utf8, len);
edisonn@google.com6e49c342013-06-27 20:03:43 +0000239 return SkScalarToDouble(textWidth);
edisonn@google.com1be794f2013-06-21 21:43:09 +0000240 }
241
edisonn@google.com1be794f2013-06-21 21:43:09 +0000242 virtual void afterWord(SkPaint* paint, SkMatrix* matrix) {}
243};
244
edisonn@google.com1be794f2013-06-21 21:43:09 +0000245class SkPdfType0Font : public SkPdfFont {
edisonn@google.com1be794f2013-06-21 21:43:09 +0000246public:
edisonn@google.com571c70b2013-07-10 17:09:50 +0000247 SkPdfType0Font(SkNativeParsedPDF* doc, SkPdfType0FontDictionary* dict);
edisonn@google.com1be794f2013-06-21 21:43:09 +0000248
249public:
edisonn@google.com1be794f2013-06-21 21:43:09 +0000250
edisonn@google.com6e49c342013-06-27 20:03:43 +0000251 virtual double drawOneChar(unsigned int ch, SkPaint* paint, PdfContext* pdfContext, SkCanvas* canvas) {
252 return fBaseFont->drawOneChar(ToUnicode(ch), paint, pdfContext, canvas);
edisonn@google.com1be794f2013-06-21 21:43:09 +0000253 }
254
255 virtual void afterWord(SkPaint* paint, SkMatrix* matrix) {
edisonn@google.com1be794f2013-06-21 21:43:09 +0000256 }
257};
258
edisonn@google.com1be794f2013-06-21 21:43:09 +0000259class SkPdfType1Font : public SkPdfFont {
260public:
edisonn@google.com571c70b2013-07-10 17:09:50 +0000261 SkPdfType1Font(SkNativeParsedPDF* doc, SkPdfType1FontDictionary* dict) {
edisonn@google.com6e49c342013-06-27 20:03:43 +0000262 if (dict->has_FontDescriptor()) {
edisonn@google.com571c70b2013-07-10 17:09:50 +0000263 fBaseFont = SkPdfFont::fontFromFontDescriptor(doc, dict->FontDescriptor(doc));
edisonn@google.com6e49c342013-06-27 20:03:43 +0000264 } else {
edisonn@google.com571c70b2013-07-10 17:09:50 +0000265 fBaseFont = fontFromName(doc, dict, dict->BaseFont(doc).c_str());
edisonn@google.com6e49c342013-06-27 20:03:43 +0000266 }
edisonn@google.com1be794f2013-06-21 21:43:09 +0000267 }
268
edisonn@google.com1be794f2013-06-21 21:43:09 +0000269public:
edisonn@google.com6e49c342013-06-27 20:03:43 +0000270 virtual double drawOneChar(unsigned int ch, SkPaint* paint, PdfContext* pdfContext, SkCanvas* canvas) {
271 return fBaseFont->drawOneChar(ToUnicode(ch), paint, pdfContext, canvas);
edisonn@google.com1be794f2013-06-21 21:43:09 +0000272 }
273
274 virtual void afterWord(SkPaint* paint, SkMatrix* matrix) {
275
276 }
edisonn@google.com6e49c342013-06-27 20:03:43 +0000277};
edisonn@google.com1be794f2013-06-21 21:43:09 +0000278
edisonn@google.com6e49c342013-06-27 20:03:43 +0000279class SkPdfTrueTypeFont : public SkPdfType1Font {
edisonn@google.com1be794f2013-06-21 21:43:09 +0000280public:
edisonn@google.com571c70b2013-07-10 17:09:50 +0000281 SkPdfTrueTypeFont(SkNativeParsedPDF* doc, SkPdfTrueTypeFontDictionary* dict) : SkPdfType1Font(doc, dict) {
edisonn@google.com1be794f2013-06-21 21:43:09 +0000282 }
283};
284
edisonn@google.com6e49c342013-06-27 20:03:43 +0000285class SkPdfMultiMasterFont : public SkPdfType1Font {
edisonn@google.com1be794f2013-06-21 21:43:09 +0000286public:
edisonn@google.com571c70b2013-07-10 17:09:50 +0000287 SkPdfMultiMasterFont(SkNativeParsedPDF* doc, SkPdfMultiMasterFontDictionary* dict) : SkPdfType1Font(doc, dict) {
edisonn@google.com1be794f2013-06-21 21:43:09 +0000288 }
289};
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000290/*
291class CIDToGIDMap {
292 virtual unsigned int map(unsigned int cid) = 0;
293 static CIDToGIDMap* fromName(const char* name);
294};
295
296class CIDToGIDMap_Identity {
297 virtual unsigned int map(unsigned int cid) { return cid; }
298
299 static CIDToGIDMap_Identity* instance() {
300 static CIDToGIDMap_Identity* inst = new CIDToGIDMap_Identity();
301 return inst;
302 }
303};
304
305CIDToGIDMap* CIDToGIDMap::fromName(const char* name) {
306 // The only one supported right now is Identity
307 if (strcmp(name, "Identity") == 0) {
308 return CIDToGIDMap_Identity::instance();
309 }
310
311#ifdef PDF_TRACE
312 // TODO(edisonn): warning/report
313 printf("Unknown CIDToGIDMap: %s\n", name);
314#endif
315 return NULL;
316}
317CIDToGIDMap* fCidToGid;
318*/
edisonn@google.com1be794f2013-06-21 21:43:09 +0000319
320class SkPdfType3Font : public SkPdfFont {
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000321 struct Type3FontChar {
edisonn@google.comb0145ce2013-08-05 16:23:23 +0000322 SkPdfObject* fObj;
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000323 double fWidth;
324 };
325
326 SkPdfDictionary* fCharProcs;
327 SkPdfEncodingDictionary* fEncodingDict;
328 unsigned int fFirstChar;
329 unsigned int fLastChar;
330
331 SkRect fFontBBox;
332 SkMatrix fFonMatrix;
333
334 Type3FontChar* fChars;
335
edisonn@google.com1be794f2013-06-21 21:43:09 +0000336public:
edisonn@google.com571c70b2013-07-10 17:09:50 +0000337 SkPdfType3Font(SkNativeParsedPDF* parsed, SkPdfType3FontDictionary* dict) {
338 fBaseFont = fontFromName(parsed, dict, dict->BaseFont(parsed).c_str());
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000339
340 if (dict->has_Encoding()) {
edisonn@google.com571c70b2013-07-10 17:09:50 +0000341 if (dict->isEncodingAName(parsed)) {
342 fEncoding = SkPdfEncoding::fromName(dict->getEncodingAsName(parsed).c_str());
343 } else if (dict->isEncodingAEncodingdictionary(parsed)) {
edisonn@google.com2fd5d362013-07-23 19:43:48 +0000344 // No encoding.
345 fEncoding = SkPdfDefaultEncoding::instance();
edisonn@google.com571c70b2013-07-10 17:09:50 +0000346 fEncodingDict = dict->getEncodingAsEncodingdictionary(parsed);
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000347 }
348 }
349
350 // null?
edisonn@google.com571c70b2013-07-10 17:09:50 +0000351 fCharProcs = dict->CharProcs(parsed);
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000352
353 fToUnicode = NULL;
354 if (dict->has_ToUnicode()) {
edisonn@google.com571c70b2013-07-10 17:09:50 +0000355 fToUnicode = new SkPdfToUnicode(parsed, dict->ToUnicode(parsed));
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000356 }
357
edisonn@google.coma3356fc2013-07-10 18:20:06 +0000358 fFirstChar = (unsigned int)dict->FirstChar(parsed);
359 fLastChar = (unsigned int)dict->LastChar(parsed);
edisonn@google.com571c70b2013-07-10 17:09:50 +0000360 fFonMatrix = dict->has_FontMatrix() ? dict->FontMatrix(parsed) : SkMatrix::I();
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000361
edisonn@google.com571c70b2013-07-10 17:09:50 +0000362 if (dict->has_FontBBox()) {
363 fFontBBox = dict->FontBBox(parsed);
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000364 }
365
366 fChars = new Type3FontChar[fLastChar - fFirstChar + 1];
367
368 memset(fChars, 0, sizeof(fChars[0]) * (fLastChar - fFirstChar + 1));
369
edisonn@google.com571c70b2013-07-10 17:09:50 +0000370 const SkPdfArray* widths = dict->Widths(parsed);
371 for (unsigned int i = 0 ; i < widths->size(); i++) {
edisonn@google.comac4bedc2013-07-25 21:40:23 +0000372 if ((fFirstChar + i) >= fFirstChar && (fFirstChar + i) <= fLastChar) {
373 fChars[i].fWidth = (*widths)[i]->numberValue();
374 } else {
375 // TODO(edisonn): report pdf corruption
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000376 }
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000377 }
378
edisonn@google.com571c70b2013-07-10 17:09:50 +0000379 const SkPdfArray* diffs = fEncodingDict->Differences(parsed);
380 unsigned int j = fFirstChar;
381 for (unsigned int i = 0 ; i < diffs->size(); i++) {
382 if ((*diffs)[i]->isInteger()) {
edisonn@google.coma3356fc2013-07-10 18:20:06 +0000383 j = (unsigned int)(*diffs)[i]->intValue();
edisonn@google.com571c70b2013-07-10 17:09:50 +0000384 } else if ((*diffs)[i]->isName()) {
edisonn@google.comac4bedc2013-07-25 21:40:23 +0000385 if (j >= fFirstChar && j <= fLastChar) {
386 fChars[j - fFirstChar].fObj = fCharProcs->get((*diffs)[i]);
387 } else {
388 // TODO(edisonn): report pdf corruption
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000389 }
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000390 j++;
391 } else {
edisonn@google.comac4bedc2013-07-25 21:40:23 +0000392 // TODO(edisonn): report bad pdf
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000393 }
394 }
edisonn@google.com1be794f2013-06-21 21:43:09 +0000395 }
396
397public:
edisonn@google.com6e49c342013-06-27 20:03:43 +0000398 virtual double drawOneChar(unsigned int ch, SkPaint* paint, PdfContext* pdfContext, SkCanvas* canvas) {
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000399 if (ch < fFirstChar || ch > fLastChar || !fChars[ch - fFirstChar].fObj) {
edisonn@google.com6e49c342013-06-27 20:03:43 +0000400 return fBaseFont->drawOneChar(ToUnicode(ch), paint, pdfContext, canvas);
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000401 }
edisonn@google.com1be794f2013-06-21 21:43:09 +0000402
edisonn@google.comb857a0c2013-06-25 20:45:40 +0000403#ifdef PDF_TRACE
404 printf("Type 3 char to unicode: %c\n", ToUnicode(ch));
405 if (ToUnicode(ch) == 'A') {
406 printf("break;\n");
407 }
408#endif
409
edisonn@google.com571c70b2013-07-10 17:09:50 +0000410 // TODO(edisonn): is it better to resolve the reference at load time, or now?
411 doType3Char(pdfContext, canvas, pdfContext->fPdfDoc->resolveReference(fChars[ch - fFirstChar].fObj), fFontBBox, fFonMatrix, pdfContext->fGraphicsState.fCurFontSize);
edisonn@google.com1be794f2013-06-21 21:43:09 +0000412
edisonn@google.com6e49c342013-06-27 20:03:43 +0000413 // TODO(edisonn): verify/test translate code, not tested yet
414 pdfContext->fGraphicsState.fMatrixTm.preTranslate(SkDoubleToScalar(pdfContext->fGraphicsState.fCurFontSize * fChars[ch - fFirstChar].fWidth),
415 SkDoubleToScalar(0.0));
edisonn@google.com571c70b2013-07-10 17:09:50 +0000416 return fChars[ch - fFirstChar].fWidth;
edisonn@google.com1be794f2013-06-21 21:43:09 +0000417 }
418
419 virtual void afterWord(SkPaint* paint, SkMatrix* matrix) {
420
421 }
422};
423
424#endif // __DEFINED__SkPdfFont