/*
* Copyright 2013 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "SkPdfContext.h"
#include "SkPdfNativeDoc.h"
#include "SkPdfReporter.h"
#include "SkPdfTokenLooper.h"
///////////////////////////////////////////////////////////////////////////////
class PdfMainLooper : public SkPdfTokenLooper {
public:
PdfMainLooper(SkPdfNativeTokenizer* tokenizer,
SkPdfContext* pdfContext,
SkCanvas* canvas)
: INHERITED(tokenizer, pdfContext, canvas) {}
SkPdfResult consumeToken(PdfToken& token) override;
void loop() override;
private:
typedef SkPdfTokenLooper INHERITED;
};
///////////////////////////////////////////////////////////////////////////////
SkPdfContext::SkPdfContext(SkPdfNativeDoc* doc)
: fPdfDoc(doc)
{
SkASSERT(fPdfDoc != NULL);
}
void SkPdfContext::parseStream(SkPdfNativeObject* stream, SkCanvas* canvas) {
if (NULL == stream) {
// Nothing to parse.
return;
}
SkPdfNativeTokenizer tokenizer(stream, &fTmpPageAllocator, fPdfDoc);
PdfMainLooper looper(&tokenizer, this, canvas);
looper.loop();
}
///////////////////////////////////////////////////////////////////////////////
// FIXME (scroggo): This probably belongs in a debugging file.
// For reportRenderStats declaration.
#include "SkPdfRenderer.h"
// Temp code to measure what operands fail.
template <typename T> class SkTDictWithDefaultConstructor : public SkTDict<T> {
public:
SkTDictWithDefaultConstructor() : SkTDict<T>(10) {}
};
SkTDictWithDefaultConstructor<int> gRenderStats[kCount_SkPdfResult];
const char* gRenderStatsNames[kCount_SkPdfResult] = {
"Success",
"Partially implemented",
"Not yet implemented",
"Ignore Error",
"Error",
"Unsupported/Unknown"
};
// Declared in SkPdfRenderer.h. Should be moved to a central debugging location.
void reportPdfRenderStats() {
for (int i = 0 ; i < kCount_SkPdfResult; i++) {
SkTDict<int>::Iter iter(gRenderStats[i]);
const char* key;
int value = 0;
while ((key = iter.next(&value)) != NULL) {
SkDebugf("%s: %s -> count %i\n", gRenderStatsNames[i], key, value);
}
}
}
#include "SkPdfOps.h"
SkPdfResult PdfMainLooper::consumeToken(PdfToken& token) {
if (token.fType == kKeyword_TokenType && token.fKeywordLength < 256)
{
PdfOperatorRenderer pdfOperatorRenderer = NULL;
if (gPdfOps.find(token.fKeyword, token.fKeywordLength, &pdfOperatorRenderer) &&
pdfOperatorRenderer) {
// Main work is done by pdfOperatorRenderer(...)
SkPdfResult result = pdfOperatorRenderer(fPdfContext, fCanvas, this);
int cnt = 0;
gRenderStats[result].find(token.fKeyword, token.fKeywordLength, &cnt);
gRenderStats[result].set(token.fKeyword, token.fKeywordLength, cnt + 1);
} else {
int cnt = 0;
gRenderStats[kUnsupported_SkPdfResult].find(token.fKeyword,
token.fKeywordLength,
&cnt);
gRenderStats[kUnsupported_SkPdfResult].set(token.fKeyword,
token.fKeywordLength,
cnt + 1);
}
}
else if (token.fType == kObject_TokenType)
{
fPdfContext->fObjectStack.push( token.fObject );
}
else {
// TODO(edisonn): store the keyword as a object, so we can track the location in file,
// and report where the error was triggered
SkPdfReport(kCodeWarning_SkPdfIssueSeverity, kNYI_SkPdfIssue, token.fKeyword, NULL,
fPdfContext);
return kIgnoreError_SkPdfResult;
}
return kOK_SkPdfResult;
}
void PdfMainLooper::loop() {
PdfToken token;
while (fTokenizer->readToken(&token, true)) {
this->consumeToken(token);
}
}