|  | // Copyright 2014 PDFium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com | 
|  |  | 
|  | #include "core/fxge/win32/cfx_psrenderer.h" | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <memory> | 
|  | #include <sstream> | 
|  | #include <utility> | 
|  |  | 
|  | #include "core/fxcrt/maybe_owned.h" | 
|  | #include "core/fxge/cfx_fontcache.h" | 
|  | #include "core/fxge/cfx_gemodule.h" | 
|  | #include "core/fxge/cfx_glyphcache.h" | 
|  | #include "core/fxge/cfx_pathdata.h" | 
|  | #include "core/fxge/cfx_renderdevice.h" | 
|  | #include "core/fxge/dib/cfx_dibextractor.h" | 
|  | #include "core/fxge/dib/cfx_dibitmap.h" | 
|  | #include "core/fxge/fx_dib.h" | 
|  | #include "core/fxge/text_char_pos.h" | 
|  | #include "core/fxge/win32/cpsoutput.h" | 
|  | #include "third_party/base/ptr_util.h" | 
|  |  | 
|  | struct PSGlyph { | 
|  | UnownedPtr<CFX_Font> m_pFont; | 
|  | uint32_t m_GlyphIndex; | 
|  | bool m_bGlyphAdjust; | 
|  | float m_AdjustMatrix[4]; | 
|  | }; | 
|  |  | 
|  | class CPSFont { | 
|  | public: | 
|  | int m_nGlyphs; | 
|  | PSGlyph m_Glyphs[256]; | 
|  | }; | 
|  |  | 
|  | CFX_PSRenderer::CFX_PSRenderer(const EncoderIface* pEncoderIface) | 
|  | : m_pEncoderIface(pEncoderIface) {} | 
|  |  | 
|  | CFX_PSRenderer::~CFX_PSRenderer() = default; | 
|  |  | 
|  | void CFX_PSRenderer::Init(const RetainPtr<IFX_RetainableWriteStream>& pStream, | 
|  | int pslevel, | 
|  | int width, | 
|  | int height, | 
|  | bool bCmykOutput) { | 
|  | m_PSLevel = pslevel; | 
|  | m_pStream = pStream; | 
|  | m_ClipBox.left = 0; | 
|  | m_ClipBox.top = 0; | 
|  | m_ClipBox.right = width; | 
|  | m_ClipBox.bottom = height; | 
|  | m_bCmykOutput = bCmykOutput; | 
|  | } | 
|  |  | 
|  | bool CFX_PSRenderer::StartRendering() { | 
|  | if (m_bInited) | 
|  | return true; | 
|  |  | 
|  | static const char init_str[] = | 
|  | "\nsave\n/im/initmatrix load def\n" | 
|  | "/n/newpath load def/m/moveto load def/l/lineto load def/c/curveto load " | 
|  | "def/h/closepath load def\n" | 
|  | "/f/fill load def/F/eofill load def/s/stroke load def/W/clip load " | 
|  | "def/W*/eoclip load def\n" | 
|  | "/rg/setrgbcolor load def/k/setcmykcolor load def\n" | 
|  | "/J/setlinecap load def/j/setlinejoin load def/w/setlinewidth load " | 
|  | "def/M/setmiterlimit load def/d/setdash load def\n" | 
|  | "/q/gsave load def/Q/grestore load def/iM/imagemask load def\n" | 
|  | "/Tj/show load def/Ff/findfont load def/Fs/scalefont load def/Sf/setfont " | 
|  | "load def\n" | 
|  | "/cm/concat load def/Cm/currentmatrix load def/mx/matrix load " | 
|  | "def/sm/setmatrix load def\n"; | 
|  | m_pStream->WriteString(init_str); | 
|  | m_bInited = true; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::EndRendering() { | 
|  | if (!m_bInited) | 
|  | return; | 
|  |  | 
|  | m_pStream->WriteString("\nrestore\n"); | 
|  | m_bInited = false; | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::SaveState() { | 
|  | StartRendering(); | 
|  | m_pStream->WriteString("q\n"); | 
|  | m_ClipBoxStack.push_back(m_ClipBox); | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::RestoreState(bool bKeepSaved) { | 
|  | StartRendering(); | 
|  | m_pStream->WriteString("Q\n"); | 
|  | if (bKeepSaved) | 
|  | m_pStream->WriteString("q\n"); | 
|  |  | 
|  | m_bColorSet = false; | 
|  | m_bGraphStateSet = false; | 
|  | if (m_ClipBoxStack.empty()) | 
|  | return; | 
|  |  | 
|  | m_ClipBox = m_ClipBoxStack.back(); | 
|  | if (!bKeepSaved) | 
|  | m_ClipBoxStack.pop_back(); | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::OutputPath(const CFX_PathData* pPathData, | 
|  | const CFX_Matrix* pObject2Device) { | 
|  | std::ostringstream buf; | 
|  | size_t size = pPathData->GetPoints().size(); | 
|  |  | 
|  | for (size_t i = 0; i < size; i++) { | 
|  | FXPT_TYPE type = pPathData->GetType(i); | 
|  | bool closing = pPathData->IsClosingFigure(i); | 
|  | CFX_PointF pos = pPathData->GetPoint(i); | 
|  | if (pObject2Device) | 
|  | pos = pObject2Device->Transform(pos); | 
|  |  | 
|  | buf << pos.x << " " << pos.y; | 
|  | switch (type) { | 
|  | case FXPT_TYPE::MoveTo: | 
|  | buf << " m "; | 
|  | break; | 
|  | case FXPT_TYPE::LineTo: | 
|  | buf << " l "; | 
|  | if (closing) | 
|  | buf << "h "; | 
|  | break; | 
|  | case FXPT_TYPE::BezierTo: { | 
|  | CFX_PointF pos1 = pPathData->GetPoint(i + 1); | 
|  | CFX_PointF pos2 = pPathData->GetPoint(i + 2); | 
|  | if (pObject2Device) { | 
|  | pos1 = pObject2Device->Transform(pos1); | 
|  | pos2 = pObject2Device->Transform(pos2); | 
|  | } | 
|  | buf << " " << pos1.x << " " << pos1.y << " " << pos2.x << " " << pos2.y | 
|  | << " c"; | 
|  | if (closing) | 
|  | buf << " h"; | 
|  | buf << "\n"; | 
|  | i += 2; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | WriteToStream(&buf); | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::SetClip_PathFill(const CFX_PathData* pPathData, | 
|  | const CFX_Matrix* pObject2Device, | 
|  | int fill_mode) { | 
|  | StartRendering(); | 
|  | OutputPath(pPathData, pObject2Device); | 
|  | CFX_FloatRect rect = pPathData->GetBoundingBox(); | 
|  | if (pObject2Device) | 
|  | rect = pObject2Device->TransformRect(rect); | 
|  |  | 
|  | m_ClipBox.left = static_cast<int>(rect.left); | 
|  | m_ClipBox.right = static_cast<int>(rect.left + rect.right); | 
|  | m_ClipBox.top = static_cast<int>(rect.top + rect.bottom); | 
|  | m_ClipBox.bottom = static_cast<int>(rect.bottom); | 
|  |  | 
|  | m_pStream->WriteString("W"); | 
|  | if ((fill_mode & 3) != FXFILL_WINDING) | 
|  | m_pStream->WriteString("*"); | 
|  | m_pStream->WriteString(" n\n"); | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::SetClip_PathStroke(const CFX_PathData* pPathData, | 
|  | const CFX_Matrix* pObject2Device, | 
|  | const CFX_GraphStateData* pGraphState) { | 
|  | StartRendering(); | 
|  | SetGraphState(pGraphState); | 
|  |  | 
|  | std::ostringstream buf; | 
|  | buf << "mx Cm [" << pObject2Device->a << " " << pObject2Device->b << " " | 
|  | << pObject2Device->c << " " << pObject2Device->d << " " | 
|  | << pObject2Device->e << " " << pObject2Device->f << "]cm "; | 
|  | WriteToStream(&buf); | 
|  |  | 
|  | OutputPath(pPathData, nullptr); | 
|  | CFX_FloatRect rect = pPathData->GetBoundingBox(pGraphState->m_LineWidth, | 
|  | pGraphState->m_MiterLimit); | 
|  | m_ClipBox.Intersect(pObject2Device->TransformRect(rect).GetOuterRect()); | 
|  |  | 
|  | m_pStream->WriteString("strokepath W n sm\n"); | 
|  | } | 
|  |  | 
|  | bool CFX_PSRenderer::DrawPath(const CFX_PathData* pPathData, | 
|  | const CFX_Matrix* pObject2Device, | 
|  | const CFX_GraphStateData* pGraphState, | 
|  | uint32_t fill_color, | 
|  | uint32_t stroke_color, | 
|  | int fill_mode) { | 
|  | StartRendering(); | 
|  | int fill_alpha = FXARGB_A(fill_color); | 
|  | int stroke_alpha = FXARGB_A(stroke_color); | 
|  | if (fill_alpha && fill_alpha < 255) | 
|  | return false; | 
|  | if (stroke_alpha && stroke_alpha < 255) | 
|  | return false; | 
|  | if (fill_alpha == 0 && stroke_alpha == 0) | 
|  | return false; | 
|  |  | 
|  | if (stroke_alpha) { | 
|  | SetGraphState(pGraphState); | 
|  | if (pObject2Device) { | 
|  | std::ostringstream buf; | 
|  | buf << "mx Cm [" << pObject2Device->a << " " << pObject2Device->b << " " | 
|  | << pObject2Device->c << " " << pObject2Device->d << " " | 
|  | << pObject2Device->e << " " << pObject2Device->f << "]cm "; | 
|  | WriteToStream(&buf); | 
|  | } | 
|  | } | 
|  |  | 
|  | OutputPath(pPathData, stroke_alpha ? nullptr : pObject2Device); | 
|  | if (fill_mode && fill_alpha) { | 
|  | SetColor(fill_color); | 
|  | if ((fill_mode & 3) == FXFILL_WINDING) { | 
|  | if (stroke_alpha) | 
|  | m_pStream->WriteString("q f Q "); | 
|  | else | 
|  | m_pStream->WriteString("f"); | 
|  | } else if ((fill_mode & 3) == FXFILL_ALTERNATE) { | 
|  | if (stroke_alpha) | 
|  | m_pStream->WriteString("q F Q "); | 
|  | else | 
|  | m_pStream->WriteString("F"); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (stroke_alpha) { | 
|  | SetColor(stroke_color); | 
|  | m_pStream->WriteString("s"); | 
|  | if (pObject2Device) | 
|  | m_pStream->WriteString(" sm"); | 
|  | } | 
|  |  | 
|  | m_pStream->WriteString("\n"); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::SetGraphState(const CFX_GraphStateData* pGraphState) { | 
|  | std::ostringstream buf; | 
|  | if (!m_bGraphStateSet || | 
|  | m_CurGraphState.m_LineCap != pGraphState->m_LineCap) { | 
|  | buf << static_cast<int>(pGraphState->m_LineCap) << " J\n"; | 
|  | } | 
|  | if (!m_bGraphStateSet || | 
|  | m_CurGraphState.m_DashArray != pGraphState->m_DashArray) { | 
|  | buf << "["; | 
|  | for (const auto& dash : pGraphState->m_DashArray) | 
|  | buf << dash << " "; | 
|  | buf << "]" << pGraphState->m_DashPhase << " d\n"; | 
|  | } | 
|  | if (!m_bGraphStateSet || | 
|  | m_CurGraphState.m_LineJoin != pGraphState->m_LineJoin) { | 
|  | buf << static_cast<int>(pGraphState->m_LineJoin) << " j\n"; | 
|  | } | 
|  | if (!m_bGraphStateSet || | 
|  | m_CurGraphState.m_LineWidth != pGraphState->m_LineWidth) { | 
|  | buf << pGraphState->m_LineWidth << " w\n"; | 
|  | } | 
|  | if (!m_bGraphStateSet || | 
|  | m_CurGraphState.m_MiterLimit != pGraphState->m_MiterLimit) { | 
|  | buf << pGraphState->m_MiterLimit << " M\n"; | 
|  | } | 
|  | m_CurGraphState = *pGraphState; | 
|  | m_bGraphStateSet = true; | 
|  | WriteToStream(&buf); | 
|  | } | 
|  |  | 
|  | bool CFX_PSRenderer::SetDIBits(const RetainPtr<CFX_DIBBase>& pSource, | 
|  | uint32_t color, | 
|  | int left, | 
|  | int top) { | 
|  | StartRendering(); | 
|  | CFX_Matrix matrix = CFX_RenderDevice::GetFlipMatrix( | 
|  | pSource->GetWidth(), pSource->GetHeight(), left, top); | 
|  | return DrawDIBits(pSource, color, matrix, FXDIB_ResampleOptions()); | 
|  | } | 
|  |  | 
|  | bool CFX_PSRenderer::StretchDIBits(const RetainPtr<CFX_DIBBase>& pSource, | 
|  | uint32_t color, | 
|  | int dest_left, | 
|  | int dest_top, | 
|  | int dest_width, | 
|  | int dest_height, | 
|  | const FXDIB_ResampleOptions& options) { | 
|  | StartRendering(); | 
|  | CFX_Matrix matrix = CFX_RenderDevice::GetFlipMatrix(dest_width, dest_height, | 
|  | dest_left, dest_top); | 
|  | return DrawDIBits(pSource, color, matrix, options); | 
|  | } | 
|  |  | 
|  | bool CFX_PSRenderer::DrawDIBits(const RetainPtr<CFX_DIBBase>& pSource, | 
|  | uint32_t color, | 
|  | const CFX_Matrix& matrix, | 
|  | const FXDIB_ResampleOptions& options) { | 
|  | StartRendering(); | 
|  | if ((matrix.a == 0 && matrix.b == 0) || (matrix.c == 0 && matrix.d == 0)) | 
|  | return true; | 
|  |  | 
|  | if (pSource->HasAlpha()) | 
|  | return false; | 
|  |  | 
|  | int alpha = FXARGB_A(color); | 
|  | if (pSource->IsAlphaMask() && (alpha < 255 || pSource->GetBPP() != 1)) | 
|  | return false; | 
|  |  | 
|  | m_pStream->WriteString("q\n"); | 
|  |  | 
|  | std::ostringstream buf; | 
|  | buf << "[" << matrix.a << " " << matrix.b << " " << matrix.c << " " | 
|  | << matrix.d << " " << matrix.e << " " << matrix.f << "]cm "; | 
|  |  | 
|  | int width = pSource->GetWidth(); | 
|  | int height = pSource->GetHeight(); | 
|  | buf << width << " " << height; | 
|  |  | 
|  | if (pSource->GetBPP() == 1 && !pSource->GetPalette()) { | 
|  | int pitch = (width + 7) / 8; | 
|  | uint32_t src_size = height * pitch; | 
|  | std::unique_ptr<uint8_t, FxFreeDeleter> src_buf( | 
|  | FX_Alloc(uint8_t, src_size)); | 
|  | for (int row = 0; row < height; row++) { | 
|  | const uint8_t* src_scan = pSource->GetScanline(row); | 
|  | memcpy(src_buf.get() + row * pitch, src_scan, pitch); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<uint8_t, FxFreeDeleter> output_buf; | 
|  | uint32_t output_size; | 
|  | bool compressed = FaxCompressData(std::move(src_buf), width, height, | 
|  | &output_buf, &output_size); | 
|  | if (pSource->IsAlphaMask()) { | 
|  | SetColor(color); | 
|  | m_bColorSet = false; | 
|  | buf << " true["; | 
|  | } else { | 
|  | buf << " 1["; | 
|  | } | 
|  | buf << width << " 0 0 -" << height << " 0 " << height | 
|  | << "]currentfile/ASCII85Decode filter "; | 
|  |  | 
|  | if (compressed) { | 
|  | buf << "<</K -1/EndOfBlock false/Columns " << width << "/Rows " << height | 
|  | << ">>/CCITTFaxDecode filter "; | 
|  | } | 
|  | if (pSource->IsAlphaMask()) | 
|  | buf << "iM\n"; | 
|  | else | 
|  | buf << "false 1 colorimage\n"; | 
|  |  | 
|  | WriteToStream(&buf); | 
|  | WritePSBinary(output_buf.get(), output_size); | 
|  | } else { | 
|  | CFX_DIBExtractor source_extractor(pSource); | 
|  | RetainPtr<CFX_DIBBase> pConverted = source_extractor.GetBitmap(); | 
|  | if (!pConverted) | 
|  | return false; | 
|  | switch (pSource->GetFormat()) { | 
|  | case FXDIB_1bppRgb: | 
|  | case FXDIB_Rgb32: | 
|  | pConverted = pConverted->CloneConvert(FXDIB_Rgb); | 
|  | break; | 
|  | case FXDIB_8bppRgb: | 
|  | if (pSource->GetPalette()) { | 
|  | pConverted = pConverted->CloneConvert(FXDIB_Rgb); | 
|  | } | 
|  | break; | 
|  | case FXDIB_1bppCmyk: | 
|  | pConverted = pConverted->CloneConvert(FXDIB_Cmyk); | 
|  | break; | 
|  | case FXDIB_8bppCmyk: | 
|  | if (pSource->GetPalette()) { | 
|  | pConverted = pConverted->CloneConvert(FXDIB_Cmyk); | 
|  | } | 
|  | break; | 
|  | default: | 
|  | break; | 
|  | } | 
|  | if (!pConverted) { | 
|  | m_pStream->WriteString("\nQ\n"); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | int bpp = pConverted->GetBPP() / 8; | 
|  | uint8_t* output_buf = nullptr; | 
|  | size_t output_size = 0; | 
|  | const char* filter = nullptr; | 
|  | if ((m_PSLevel == 2 || options.bLossy) && | 
|  | m_pEncoderIface->pJpegEncodeFunc(pConverted, &output_buf, | 
|  | &output_size)) { | 
|  | filter = "/DCTDecode filter "; | 
|  | } | 
|  | if (!filter) { | 
|  | int src_pitch = width * bpp; | 
|  | output_size = height * src_pitch; | 
|  | output_buf = FX_Alloc(uint8_t, output_size); | 
|  | for (int row = 0; row < height; row++) { | 
|  | const uint8_t* src_scan = pConverted->GetScanline(row); | 
|  | uint8_t* dest_scan = output_buf + row * src_pitch; | 
|  | if (bpp == 3) { | 
|  | for (int col = 0; col < width; col++) { | 
|  | *dest_scan++ = src_scan[2]; | 
|  | *dest_scan++ = src_scan[1]; | 
|  | *dest_scan++ = *src_scan; | 
|  | src_scan += 3; | 
|  | } | 
|  | } else { | 
|  | memcpy(dest_scan, src_scan, src_pitch); | 
|  | } | 
|  | } | 
|  | uint8_t* compressed_buf; | 
|  | uint32_t compressed_size; | 
|  | PSCompressData(output_buf, output_size, &compressed_buf, &compressed_size, | 
|  | &filter); | 
|  | if (output_buf != compressed_buf) | 
|  | FX_Free(output_buf); | 
|  |  | 
|  | output_buf = compressed_buf; | 
|  | output_size = compressed_size; | 
|  | } | 
|  | buf << " 8["; | 
|  | buf << width << " 0 0 -" << height << " 0 " << height << "]"; | 
|  | buf << "currentfile/ASCII85Decode filter "; | 
|  | if (filter) | 
|  | buf << filter; | 
|  |  | 
|  | buf << "false " << bpp; | 
|  | buf << " colorimage\n"; | 
|  | WriteToStream(&buf); | 
|  |  | 
|  | WritePSBinary(output_buf, output_size); | 
|  | FX_Free(output_buf); | 
|  | } | 
|  | m_pStream->WriteString("\nQ\n"); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::SetColor(uint32_t color) { | 
|  | bool bCMYK = false; | 
|  | if (bCMYK != m_bCmykOutput || !m_bColorSet || m_LastColor != color) { | 
|  | std::ostringstream buf; | 
|  | if (bCMYK) { | 
|  | buf << FXSYS_GetCValue(color) / 255.0 << " " | 
|  | << FXSYS_GetMValue(color) / 255.0 << " " | 
|  | << FXSYS_GetYValue(color) / 255.0 << " " | 
|  | << FXSYS_GetKValue(color) / 255.0 << " k\n"; | 
|  | } else { | 
|  | buf << FXARGB_R(color) / 255.0 << " " << FXARGB_G(color) / 255.0 << " " | 
|  | << FXARGB_B(color) / 255.0 << " rg\n"; | 
|  | } | 
|  | if (bCMYK == m_bCmykOutput) { | 
|  | m_bColorSet = true; | 
|  | m_LastColor = color; | 
|  | } | 
|  | WriteToStream(&buf); | 
|  | } | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::FindPSFontGlyph(CFX_GlyphCache* pGlyphCache, | 
|  | CFX_Font* pFont, | 
|  | const TextCharPos& charpos, | 
|  | int* ps_fontnum, | 
|  | int* ps_glyphindex) { | 
|  | int i = 0; | 
|  | for (const auto& pPSFont : m_PSFontList) { | 
|  | for (int j = 0; j < pPSFont->m_nGlyphs; j++) { | 
|  | if (pPSFont->m_Glyphs[j].m_pFont == pFont && | 
|  | pPSFont->m_Glyphs[j].m_GlyphIndex == charpos.m_GlyphIndex && | 
|  | ((!pPSFont->m_Glyphs[j].m_bGlyphAdjust && !charpos.m_bGlyphAdjust) || | 
|  | (pPSFont->m_Glyphs[j].m_bGlyphAdjust && charpos.m_bGlyphAdjust && | 
|  | (fabs(pPSFont->m_Glyphs[j].m_AdjustMatrix[0] - | 
|  | charpos.m_AdjustMatrix[0]) < 0.01 && | 
|  | fabs(pPSFont->m_Glyphs[j].m_AdjustMatrix[1] - | 
|  | charpos.m_AdjustMatrix[1]) < 0.01 && | 
|  | fabs(pPSFont->m_Glyphs[j].m_AdjustMatrix[2] - | 
|  | charpos.m_AdjustMatrix[2]) < 0.01 && | 
|  | fabs(pPSFont->m_Glyphs[j].m_AdjustMatrix[3] - | 
|  | charpos.m_AdjustMatrix[3]) < 0.01)))) { | 
|  | *ps_fontnum = i; | 
|  | *ps_glyphindex = j; | 
|  | return; | 
|  | } | 
|  | } | 
|  | ++i; | 
|  | } | 
|  |  | 
|  | if (m_PSFontList.empty() || m_PSFontList.back()->m_nGlyphs == 256) { | 
|  | m_PSFontList.push_back(pdfium::MakeUnique<CPSFont>()); | 
|  | m_PSFontList.back()->m_nGlyphs = 0; | 
|  | std::ostringstream buf; | 
|  | buf << "8 dict begin/FontType 3 def/FontMatrix[1 0 0 1 0 0]def\n" | 
|  | "/FontBBox[0 0 0 0]def/Encoding 256 array def 0 1 255{Encoding " | 
|  | "exch/.notdef put}for\n" | 
|  | "/CharProcs 1 dict def CharProcs begin/.notdef {} def end\n" | 
|  | "/BuildGlyph{1 0 -10 -10 10 10 setcachedevice exch/CharProcs get " | 
|  | "exch 2 copy known not{pop/.notdef}if get exec}bind def\n" | 
|  | "/BuildChar{1 index/Encoding get exch get 1 index/BuildGlyph get " | 
|  | "exec}bind def\n" | 
|  | "currentdict end\n"; | 
|  | buf << "/X" << static_cast<uint32_t>(m_PSFontList.size() - 1) | 
|  | << " exch definefont pop\n"; | 
|  | WriteToStream(&buf); | 
|  | buf.str(""); | 
|  | } | 
|  |  | 
|  | *ps_fontnum = m_PSFontList.size() - 1; | 
|  | CPSFont* pPSFont = m_PSFontList[*ps_fontnum].get(); | 
|  | int glyphindex = pPSFont->m_nGlyphs; | 
|  | *ps_glyphindex = glyphindex; | 
|  | pPSFont->m_Glyphs[glyphindex].m_GlyphIndex = charpos.m_GlyphIndex; | 
|  | pPSFont->m_Glyphs[glyphindex].m_pFont = pFont; | 
|  | pPSFont->m_Glyphs[glyphindex].m_bGlyphAdjust = charpos.m_bGlyphAdjust; | 
|  | if (charpos.m_bGlyphAdjust) { | 
|  | pPSFont->m_Glyphs[glyphindex].m_AdjustMatrix[0] = charpos.m_AdjustMatrix[0]; | 
|  | pPSFont->m_Glyphs[glyphindex].m_AdjustMatrix[1] = charpos.m_AdjustMatrix[1]; | 
|  | pPSFont->m_Glyphs[glyphindex].m_AdjustMatrix[2] = charpos.m_AdjustMatrix[2]; | 
|  | pPSFont->m_Glyphs[glyphindex].m_AdjustMatrix[3] = charpos.m_AdjustMatrix[3]; | 
|  | } | 
|  | pPSFont->m_nGlyphs++; | 
|  |  | 
|  | CFX_Matrix matrix; | 
|  | if (charpos.m_bGlyphAdjust) { | 
|  | matrix = | 
|  | CFX_Matrix(charpos.m_AdjustMatrix[0], charpos.m_AdjustMatrix[1], | 
|  | charpos.m_AdjustMatrix[2], charpos.m_AdjustMatrix[3], 0, 0); | 
|  | } | 
|  | const CFX_PathData* pPathData = pGlyphCache->LoadGlyphPath( | 
|  | pFont, charpos.m_GlyphIndex, charpos.m_FontCharWidth); | 
|  | if (!pPathData) | 
|  | return; | 
|  |  | 
|  | CFX_PathData TransformedPath(*pPathData); | 
|  | if (charpos.m_bGlyphAdjust) | 
|  | TransformedPath.Transform(matrix); | 
|  |  | 
|  | std::ostringstream buf; | 
|  | buf << "/X" << *ps_fontnum << " Ff/CharProcs get begin/" << glyphindex | 
|  | << "{n "; | 
|  | for (size_t p = 0; p < TransformedPath.GetPoints().size(); p++) { | 
|  | CFX_PointF point = TransformedPath.GetPoint(p); | 
|  | switch (TransformedPath.GetType(p)) { | 
|  | case FXPT_TYPE::MoveTo: { | 
|  | buf << point.x << " " << point.y << " m\n"; | 
|  | break; | 
|  | } | 
|  | case FXPT_TYPE::LineTo: { | 
|  | buf << point.x << " " << point.y << " l\n"; | 
|  | break; | 
|  | } | 
|  | case FXPT_TYPE::BezierTo: { | 
|  | CFX_PointF point1 = TransformedPath.GetPoint(p + 1); | 
|  | CFX_PointF point2 = TransformedPath.GetPoint(p + 2); | 
|  | buf << point.x << " " << point.y << " " << point1.x << " " << point1.y | 
|  | << " " << point2.x << " " << point2.y << " c\n"; | 
|  | p += 2; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | buf << "f}bind def end\n"; | 
|  | buf << "/X" << *ps_fontnum << " Ff/Encoding get " << glyphindex << "/" | 
|  | << glyphindex << " put\n"; | 
|  | WriteToStream(&buf); | 
|  | } | 
|  |  | 
|  | bool CFX_PSRenderer::DrawText(int nChars, | 
|  | const TextCharPos* pCharPos, | 
|  | CFX_Font* pFont, | 
|  | const CFX_Matrix& mtObject2Device, | 
|  | float font_size, | 
|  | uint32_t color) { | 
|  | // Check object to device matrix first, since it can scale the font size. | 
|  | if ((mtObject2Device.a == 0 && mtObject2Device.b == 0) || | 
|  | (mtObject2Device.c == 0 && mtObject2Device.d == 0)) { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // Do not send near zero font sizes to printers. See crbug.com/767343. | 
|  | float scale = | 
|  | std::min(mtObject2Device.GetXUnit(), mtObject2Device.GetYUnit()); | 
|  | static constexpr float kEpsilon = 0.01f; | 
|  | if (std::fabs(font_size * scale) < kEpsilon) | 
|  | return true; | 
|  |  | 
|  | StartRendering(); | 
|  | int alpha = FXARGB_A(color); | 
|  | if (alpha < 255) | 
|  | return false; | 
|  |  | 
|  | SetColor(color); | 
|  | std::ostringstream buf; | 
|  | buf << "q[" << mtObject2Device.a << " " << mtObject2Device.b << " " | 
|  | << mtObject2Device.c << " " << mtObject2Device.d << " " | 
|  | << mtObject2Device.e << " " << mtObject2Device.f << "]cm\n"; | 
|  |  | 
|  | CFX_FontCache* pCache = CFX_GEModule::Get()->GetFontCache(); | 
|  | RetainPtr<CFX_GlyphCache> pGlyphCache = pCache->GetGlyphCache(pFont); | 
|  | int last_fontnum = -1; | 
|  | for (int i = 0; i < nChars; i++) { | 
|  | int ps_fontnum, ps_glyphindex; | 
|  | FindPSFontGlyph(pGlyphCache.Get(), pFont, pCharPos[i], &ps_fontnum, | 
|  | &ps_glyphindex); | 
|  | if (last_fontnum != ps_fontnum) { | 
|  | buf << "/X" << ps_fontnum << " Ff " << font_size << " Fs Sf "; | 
|  | last_fontnum = ps_fontnum; | 
|  | } | 
|  | buf << pCharPos[i].m_Origin.x << " " << pCharPos[i].m_Origin.y << " m"; | 
|  | ByteString hex = ByteString::Format("<%02X>", ps_glyphindex); | 
|  | buf << hex.AsStringView() << "Tj\n"; | 
|  | } | 
|  | buf << "Q\n"; | 
|  | WriteToStream(&buf); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool CFX_PSRenderer::FaxCompressData( | 
|  | std::unique_ptr<uint8_t, FxFreeDeleter> src_buf, | 
|  | int width, | 
|  | int height, | 
|  | std::unique_ptr<uint8_t, FxFreeDeleter>* dest_buf, | 
|  | uint32_t* dest_size) const { | 
|  | if (width * height <= 128) { | 
|  | *dest_buf = std::move(src_buf); | 
|  | *dest_size = (width + 7) / 8 * height; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | m_pEncoderIface->pFaxEncodeFunc(src_buf.get(), width, height, (width + 7) / 8, | 
|  | dest_buf, dest_size); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::PSCompressData(uint8_t* src_buf, | 
|  | uint32_t src_size, | 
|  | uint8_t** output_buf, | 
|  | uint32_t* output_size, | 
|  | const char** filter) const { | 
|  | *output_buf = src_buf; | 
|  | *output_size = src_size; | 
|  | *filter = ""; | 
|  | if (src_size < 1024) | 
|  | return; | 
|  |  | 
|  | uint8_t* dest_buf = nullptr; | 
|  | uint32_t dest_size = src_size; | 
|  | if (m_PSLevel >= 3) { | 
|  | std::unique_ptr<uint8_t, FxFreeDeleter> dest_buf_unique; | 
|  | if (m_pEncoderIface->pFlateEncodeFunc(src_buf, src_size, &dest_buf_unique, | 
|  | &dest_size)) { | 
|  | dest_buf = dest_buf_unique.release(); | 
|  | *filter = "/FlateDecode filter "; | 
|  | } | 
|  | } else { | 
|  | std::unique_ptr<uint8_t, FxFreeDeleter> dest_buf_unique; | 
|  | if (m_pEncoderIface->pRunLengthEncodeFunc({src_buf, src_size}, | 
|  | &dest_buf_unique, &dest_size)) { | 
|  | dest_buf = dest_buf_unique.release(); | 
|  | *filter = "/RunLengthDecode filter "; | 
|  | } | 
|  | } | 
|  | if (dest_size < src_size) { | 
|  | *output_buf = dest_buf; | 
|  | *output_size = dest_size; | 
|  | } else { | 
|  | *filter = nullptr; | 
|  | FX_Free(dest_buf); | 
|  | } | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::WritePSBinary(const uint8_t* data, int len) { | 
|  | std::unique_ptr<uint8_t, FxFreeDeleter> dest_buf; | 
|  | uint32_t dest_size; | 
|  | if (m_pEncoderIface->pA85EncodeFunc({data, static_cast<size_t>(len)}, | 
|  | &dest_buf, &dest_size)) { | 
|  | m_pStream->WriteBlock(dest_buf.get(), dest_size); | 
|  | } else { | 
|  | m_pStream->WriteBlock(data, len); | 
|  | } | 
|  | } | 
|  |  | 
|  | void CFX_PSRenderer::WriteToStream(std::ostringstream* stringStream) { | 
|  | if (stringStream->tellp() > 0) | 
|  | m_pStream->WriteBlock(stringStream->str().c_str(), stringStream->tellp()); | 
|  | } |