#include "hpdf.h"
#include "DocOutput.h"
#include "Util.h"
#include <fstream>
#include <map>
#include <ctime>
#include <cassert>
HPDF_Font fontStyles[5];
extern const float fontSizes[6];
static bool fileExists(const char *fileName)
{
if (!fileName)
return false;
return std::ifstream(fileName).good();
}
struct DocImage
{
HPDF_Image image;
};
struct DocOutputPage::Pimpl
{
DocOutputDevice* parent;
HPDF_Doc pdf;
HPDF_Page page;
int activeFontType;
float activeFontSize;
int fontType;
float fontSize;
// URI state
float startLinkX;
float startLinkY;
std::string linkUri;
void UpdateFont();
};
DocOutputPage::DocOutputPage(DocOutputDevice* a_parent) : m_pimpl(std::make_unique<Pimpl>())
{
m_pimpl->parent = a_parent;
m_pimpl->pdf = a_parent->doc();
m_pimpl->page = HPDF_AddPage(m_pimpl->pdf);
m_pimpl->activeFontType = (FontType)-1;
m_pimpl->activeFontSize = -1.0;
m_pimpl->fontType = FT_Normal;
m_pimpl->fontSize = 12.0;
HPDF_Page_BeginText(m_pimpl->page);
}
DocOutputPage::~DocOutputPage()
{
HPDF_Page_EndText(m_pimpl->page);
}
HPDF_Page DocOutputPage::page()
{
return m_pimpl->page;
}
float DocOutputPage::width()
{
return HPDF_Page_GetWidth(m_pimpl->page);
}
float DocOutputPage::height()
{
return HPDF_Page_GetHeight(m_pimpl->page);
}
float DocOutputPage::textWidth(const char* str)
{
if (!str)
return 0.0;
m_pimpl->UpdateFont();
return HPDF_Page_TextWidth(m_pimpl->page, str);
}
bool DocOutputPage::imageSize(const char* fileName, float& width, float& height)
{
width = 0.0;
height = 0.0;
if (!fileExists(fileName))
{
printf("image file does not exist: %s\n", fileName ? fileName : "null");
return false;
}
// file exists, we can assume from here that fileName is not null
bool res;
HPDF_Image img = m_pimpl->parent->loadImage(fileName, res);
if (!res)
{
printf("image file couldn't be loaded: %s\n", fileName);
return false;
}
HPDF_Point pnt;
HPDF_STATUS status = HPDF_Image_GetSize2(img, &pnt);
if (status != HPDF_OK)
{
printf("image size couldn't be determined: %s\n", fileName);
return false;
}
width = pnt.x;
height = pnt.y;
return true;
}
void DocOutputPage::Pimpl::UpdateFont()
{
if (fontType != activeFontType || fontSize != activeFontSize)
{
HPDF_Page_SetFontAndSize(page, fontStyles[fontType], fontSize);
activeFontType = fontType;
activeFontSize = fontSize;
}
}
void DocOutputPage::setFontType(FontType font)
{
m_pimpl->fontType = (int)font;
}
void DocOutputPage::setFontSize(float size)
{
m_pimpl->fontSize = size;
}
void DocOutputPage::setAlpha(float a)
{
HPDF_Page_EndText(m_pimpl->page);
setAlphaPenFill(a, a);
HPDF_Page_BeginText(m_pimpl->page);
}
void DocOutputPage::setAlphaPenFill(float penAlpha, float fillAlpha)
{
HPDF_ExtGState extState = HPDF_CreateExtGState(m_pimpl->pdf);
HPDF_ExtGState_SetAlphaStroke(extState, penAlpha);
HPDF_ExtGState_SetAlphaFill(extState, fillAlpha);
HPDF_ExtGState_SetBlendMode(extState, (HPDF_BlendMode)0);
HPDF_Page_SetExtGState(m_pimpl->page, extState);
}
void DocOutputPage::setGradient(const std::vector<HPDF_GradientStop>& stops, float p1x, float p1y, float p2x, float p2y)
{
HPDF_Shading shading = HPDF_LinearGradient_New(m_pimpl->pdf, stops.data(), stops.size(), p1x, p1y, p2x, p2y);
HPDF_Page_SetShading(m_pimpl->page, shading);
}
void DocOutputPage::drawTriangles(float* points, int numPoints, unsigned char* colors, int nc_comps)
{
assert(numPoints >= 3);
const float radius = 0.f;
HPDF_REAL bbox[4];
HPDF_Page_GSave(m_pimpl->page);
bbox[0] = static_cast<HPDF_REAL>(points[0]);
bbox[1] = static_cast<HPDF_REAL>(points[0]);
bbox[2] = static_cast<HPDF_REAL>(points[1]);
bbox[3] = static_cast<HPDF_REAL>(points[1]);
for (int i = 1; i < numPoints; ++i)
{
bbox[0] = std::min(bbox[0], static_cast<HPDF_REAL>(points[i * 2]));
bbox[1] = std::max(bbox[1], static_cast<HPDF_REAL>(points[i * 2]));
bbox[2] = std::min(bbox[2], static_cast<HPDF_REAL>(points[i * 2 + 1]));
bbox[3] = std::max(bbox[3], static_cast<HPDF_REAL>(points[i * 2 + 1]));
}
bbox[0] -= radius;
bbox[1] += radius;
bbox[2] -= radius;
bbox[3] += radius;
HPDF_Shading shading = HPDF_Shading_New(m_pimpl->pdf,
HPDF_SHADING_FREE_FORM_TRIANGLE_MESH, HPDF_CS_DEVICE_RGB, bbox[0], bbox[1], bbox[2], bbox[3]);
// First triangle
for (int ptIdx = 0; ptIdx < 3; ++ptIdx)
{
const float* pt = points + ptIdx * 2;
const unsigned char* color = colors + ptIdx * nc_comps;
HPDF_Shading_AddVertexRGB(shading, HPDF_FREE_FORM_TRI_MESH_EDGEFLAG_NO_CONNECTION, pt[0], pt[1],
color[0], color[1], color[2]);
}
// Fan-out additional verts
for (int ptIdx = 3; ptIdx < numPoints; ++ptIdx)
{
const float* pt = points + ptIdx * 2;
const unsigned char* color = colors + ptIdx * nc_comps;
HPDF_Shading_AddVertexRGB(
shading, HPDF_FREE_FORM_TRI_MESH_EDGEFLAG_AC, pt[0], pt[1], color[0], color[1], color[2]);
}
HPDF_Page_SetShading(m_pimpl->page, shading);
HPDF_Page_GRestore(m_pimpl->page);
}
/*
void DocOutputPage::setColor(float r, float g, float b)
{
HPDF_Page_SetRGBFill(m_pimpl->page, r, g, b);
}
*/
void DocOutputPage::setFillColor(unsigned int color)
{
float r = ((color >> 16) & 0xff) / 255.0f;
float g = ((color >> 8) & 0xff) / 255.0f;
float b = ((color >> 0) & 0xff) / 255.0f;
HPDF_Page_SetRGBFill(m_pimpl->page, r, g, b);
}
void DocOutputPage::setPenColor(unsigned int color)
{
float r = ((color >> 16) & 0xff) / 255.0f;
float g = ((color >> 8) & 0xff) / 255.0f;
float b = ((color >> 0) & 0xff) / 255.0f;
HPDF_Page_SetRGBStroke(m_pimpl->page, r, g, b);
}
void DocOutputPage::setPenWidth(float width)
{
HPDF_Page_SetLineWidth(m_pimpl->page, width);
}
void DocOutputPage::setPenStyle(unsigned int style)
{
// TODO: not implemented
}
void DocOutputPage::drawText(float x, float y, const char* str)
{
if (!str)
return;
m_pimpl->UpdateFont();
HPDF_Page_TextOut(m_pimpl->page, x, height() - y, str);
}
void DocOutputPage::drawLine(float x1, float y1, float x2, float y2)
{
HPDF_Page_EndText(m_pimpl->page);
//HPDF_Page_SetLineWidth(m_pimpl->page, 2.0);
HPDF_Page_MoveTo(m_pimpl->page, x1, height() - y1);
HPDF_Page_LineTo(m_pimpl->page, x2, height() - y2);
HPDF_Page_Stroke(m_pimpl->page);
HPDF_Page_BeginText(m_pimpl->page);
}
void DocOutputPage::drawRect(float x1, float y1, float w, float h)
{
HPDF_Page_EndText(m_pimpl->page);
HPDF_Page_Rectangle(m_pimpl->page, x1, height() - y1 - h, w, h);
HPDF_Page_FillStroke(m_pimpl->page);
HPDF_Page_BeginText(m_pimpl->page);
}
void DocOutputPage::drawPolygon(float* pnts, int count)
{
if (!pnts)
return;
HPDF_Page_EndText(m_pimpl->page);
HPDF_Page_MoveTo(m_pimpl->page, pnts[0], height() - pnts[1]);
for (int i = 1; i < count; i++)
HPDF_Page_LineTo(m_pimpl->page, pnts[i*2+0], height() - pnts[i*2+1]);
HPDF_Page_ClosePathFillStroke(m_pimpl->page);
HPDF_Page_BeginText(m_pimpl->page);
}
void DocOutputPage::drawImage(const char* fileName, float x1, float y1, float w, float h)
{
if (!fileExists(fileName))
{
printf("image file does not exist: %s\n", fileName ? fileName : "null");
return;
}
// file exists, we can assume from here that fileName is not null
bool res;
HPDF_Image img = m_pimpl->parent->loadImage(fileName, res);
if (!res)
{
printf("image file couldn't be loaded: %s\n", fileName);
return;
}
HPDF_Page_EndText(m_pimpl->page);
HPDF_Page_DrawImage(m_pimpl->page, img, x1, height() - y1 - h, w, h);
HPDF_Page_BeginText(m_pimpl->page);
}
void DocOutputPage::drawCircle(float x, float y, float radius)
{
HPDF_Page_EndText(m_pimpl->page);
HPDF_Page_Circle(m_pimpl->page, x, height() - y, radius);
HPDF_Page_FillStroke(m_pimpl->page);
HPDF_Page_BeginText(m_pimpl->page);
}
void DocOutputPage::startLink(const char* url, float x, float y)
{
m_pimpl->linkUri = url;
m_pimpl->startLinkX = x;
m_pimpl->startLinkY = y;
}
void DocOutputPage::endLink(float x, float y)
{
HPDF_Page_EndText(m_pimpl->page);
HPDF_Rect rect = { m_pimpl->startLinkX, height() - m_pimpl->startLinkY, x, height() - y };
HPDF_Page_CreateURILinkAnnot(m_pimpl->page, rect, m_pimpl->linkUri.c_str());
HPDF_Page_BeginText(m_pimpl->page);
}
void DocOutputPage::drawCheckBox(bool checked, float x, float y, float w, float h)
{
HPDF_Page_EndText(m_pimpl->page);
HPDF_Rect rect = { x, height() - y, x + w, height() - (y + h) };
HPDF_Page_CreateCheckBoxAnnot(m_pimpl->page, rect, checked);
HPDF_Page_BeginText(m_pimpl->page);
}
struct DocOutputDevice::Pimpl
{
HPDF_Doc pdf;
HPDF_Error_Handler errorFn;
std::map<std::string, HPDF_Image> imageCache;
};
#ifdef _WIN32
#include <windows.h> // for OutputDebugStringA
static void error_handler(HPDF_STATUS error_no, HPDF_STATUS detail_no, void * /*user_data*/, int line, const char* file)
{
// DocOutputDevice* context = (DocOutputDevice*)user_data;
auto sizeRequired = size_t(_scprintf("%s(%i): error %04X: detail_no=%u\n", file, line, (HPDF_UINT)error_no, (HPDF_UINT)detail_no) + 1);
auto a = (char*)_alloca(sizeRequired);
sprintf_s(a, sizeRequired, "%s(%i): error %04X: detail_no=%u\n", file, line, (HPDF_UINT)error_no, (HPDF_UINT)detail_no);
OutputDebugStringA(a);
_freea(a);
// delete context; // Can't do this, it could be on the stack as is the case currently
exit(-1);
}
#else
static void error_handler(HPDF_STATUS error_no, HPDF_STATUS detail_no, void * /*user_data*/, int line, const char* file)
{
// DocOutputDevice* context = (DocOutputDevice*)user_data;
fprintf(stderr, "%s(%i): error %04X: detail_no=%u\n", file, line, (HPDF_UINT)error_no, (HPDF_UINT)detail_no);
// delete context; // Can't do this, it could be on the stack as is the case currently
exit(-1);
}
#endif
DocOutputDevice::DocOutputDevice(HPDF_Error_Handler errorFn) : m_pimpl(std::make_unique<Pimpl>())
{
m_pimpl->pdf = 0;
m_pimpl->errorFn = (errorFn) ? errorFn : error_handler;
reset();
}
DocOutputDevice::~DocOutputDevice()
{
HPDF_Free(m_pimpl->pdf);
}
void DocOutputDevice::reset()
{
m_pimpl->imageCache.clear();
if (m_pimpl->pdf)
{
HPDF_Free(m_pimpl->pdf);
}
m_pimpl->pdf = HPDF_New(m_pimpl->errorFn, this);
if (!m_pimpl->pdf)
{
printf("error: cannot create PdfDoc object\n");
return;
}
/*
// ../3rdParty/libharu/include/hpdf_types.h: HPDF_INFO_SUBJECT,
// ../3rdParty/libharu/include/hpdf_types.h: HPDF_INFO_KEYWORDS,
*/
time_t tm = time(&tm);
struct tm* t = gmtime(&tm);
int offsetMinutes = static_cast<int>(t->tm_gmtoff / 60);
HPDF_Date date = { 1900 + t->tm_year, 1 + t->tm_mon, t->tm_mday, t->tm_hour, t->tm_min,
t->tm_sec, 'Z', offsetMinutes / 60, offsetMinutes % 60 };
if (HPDF_SetInfoDateAttr(m_pimpl->pdf, HPDF_INFO_CREATION_DATE, date) != HPDF_OK)
{
printf("error: cannot set date on PdfDoc object\n");
return;
}
if (HPDF_SetInfoAttr(m_pimpl->pdf, HPDF_INFO_PRODUCER, "WickedDocs PDF Generator") != HPDF_OK)
{
printf("error: cannot set producer on PdfDoc object\n");
return;
}
if (HPDF_SetInfoAttr(m_pimpl->pdf, HPDF_INFO_CREATOR, "WickedDocs") != HPDF_OK)
{
printf("error: cannot set creator on PdfDoc object\n");
return;
}
HPDF_SetCompressionMode(m_pimpl->pdf, HPDF_COMP_ALL);
// Makes file larger
//HPDF_SetPageMode(m_pimpl->pdf, HPDF_PAGE_MODE_USE_OUTLINE);
/*
// TODO: password
*/
//fontStyles[0] = HPDF_GetFont(context->pdf, "David", NULL);
fontStyles[0] = HPDF_GetFont(m_pimpl->pdf, "Helvetica", NULL);
fontStyles[1] = HPDF_GetFont(m_pimpl->pdf, "Helvetica-Oblique", NULL);
fontStyles[2] = HPDF_GetFont(m_pimpl->pdf, "Helvetica-Bold", NULL);
fontStyles[3] = HPDF_GetFont(m_pimpl->pdf, "Helvetica-BoldOblique", NULL);
fontStyles[4] = HPDF_GetFont(m_pimpl->pdf, "Courier", NULL);
}
DocOutputPage* DocOutputDevice::newPage()
{
return new DocOutputPage(this);
}
HPDF_Doc DocOutputDevice::doc()
{
return m_pimpl->pdf;
}
HPDF_Image DocOutputDevice::loadImage(const char* fileName, bool& result)
{
if (m_pimpl->imageCache.count(fileName) == 0)
{
// TODO: detect and handle different image file types
// HPDF_LoadJpegImageFromFile
HPDF_Image img = HPDF_LoadPngImageFromFile2(m_pimpl->pdf, fileName);
HPDF_STATUS errorNo = HPDF_Error_GetCode(&m_pimpl->pdf->error);
if (errorNo != HPDF_OK)
{
HPDF_Error_Reset(&m_pimpl->pdf->error);
result = false;
return 0;
}
m_pimpl->imageCache.emplace(fileName, img);
}
result = true;
return m_pimpl->imageCache.at(fileName);
}
void DocOutputDevice::setAuthor(std::string author)
{
if (!author.empty() && HPDF_SetInfoAttr(m_pimpl->pdf, HPDF_INFO_AUTHOR, author.c_str()) != HPDF_OK)
{
printf("error: cannot set author on PdfDoc object\n");
}
}
void DocOutputDevice::setTitle(std::string title, std::string subTitle)
{
if (!title.empty() && HPDF_SetInfoAttr(m_pimpl->pdf, HPDF_INFO_TITLE, title.c_str()) != HPDF_OK)
{
printf("error: cannot set title on PdfDoc object\n");
}
if (!subTitle.empty() && HPDF_SetInfoAttr(m_pimpl->pdf, HPDF_INFO_SUBJECT, subTitle.c_str()) != HPDF_OK)
{
printf("error: cannot set subtitle/subject on PdfDoc object\n");
}
}
void DocOutputDevice::setPassword(std::string password)
{
if (!password.empty() && HPDF_SetPassword(m_pimpl->pdf, password.c_str(), "user") != HPDF_OK)
{
printf("error: cannot set password on PdfDoc object\n");
}
}
void DocOutputDevice::finalize(char* outputBuffer, size_t* size)
{
if (!outputBuffer || !size)
return; // TODO: raise error
HPDF_UINT32 siz = *size;
HPDF_GetContents(m_pimpl->pdf, (HPDF_BYTE*)outputBuffer, &siz);
*size = siz;
}
void DocOutputDevice::finalize(const char* outputFileName)
{
if (!outputFileName)
return; // TODO: raise error
//printf("saving to %s\n", outputFileName);
HPDF_SaveToFile(m_pimpl->pdf, outputFileName);
}