Newer
Older
Import / applications / MakePDF / DocOutput.cpp
#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);
}