// Copyright 2017 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. #include "public/fpdf_edit.h" #include <vector> #include "core/fpdfapi/page/cpdf_path.h" #include "core/fpdfapi/page/cpdf_pathobject.h" #include "core/fxcrt/fx_system.h" #include "fpdfsdk/cpdfsdk_helpers.h" #include "third_party/base/ptr_util.h" // These checks are here because core/ and public/ cannot depend on each other. static_assert(CFX_GraphStateData::LineCapButt == FPDF_LINECAP_BUTT, "CFX_GraphStateData::LineCapButt value mismatch"); static_assert(CFX_GraphStateData::LineCapRound == FPDF_LINECAP_ROUND, "CFX_GraphStateData::LineCapRound value mismatch"); static_assert(CFX_GraphStateData::LineCapSquare == FPDF_LINECAP_PROJECTING_SQUARE, "CFX_GraphStateData::LineCapSquare value mismatch"); static_assert(CFX_GraphStateData::LineJoinMiter == FPDF_LINEJOIN_MITER, "CFX_GraphStateData::LineJoinMiter value mismatch"); static_assert(CFX_GraphStateData::LineJoinRound == FPDF_LINEJOIN_ROUND, "CFX_GraphStateData::LineJoinRound value mismatch"); static_assert(CFX_GraphStateData::LineJoinBevel == FPDF_LINEJOIN_BEVEL, "CFX_GraphStateData::LineJoinBevel value mismatch"); static_assert(static_cast<int>(FXPT_TYPE::LineTo) == FPDF_SEGMENT_LINETO, "FXPT_TYPE::LineTo value mismatch"); static_assert(static_cast<int>(FXPT_TYPE::BezierTo) == FPDF_SEGMENT_BEZIERTO, "FXPT_TYPE::BezierTo value mismatch"); static_assert(static_cast<int>(FXPT_TYPE::MoveTo) == FPDF_SEGMENT_MOVETO, "FXPT_TYPE::MoveTo value mismatch"); namespace { CPDF_PathObject* CPDFPathObjectFromFPDFPageObject(FPDF_PAGEOBJECT page_object) { auto* obj = CPDFPageObjectFromFPDFPageObject(page_object); return obj ? obj->AsPath() : nullptr; } } // namespace FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV FPDFPageObj_CreateNewPath(float x, float y) { auto pPathObj = pdfium::MakeUnique<CPDF_PathObject>(); pPathObj->m_Path.AppendPoint(CFX_PointF(x, y), FXPT_TYPE::MoveTo, false); pPathObj->DefaultStates(); // Caller takes ownership. return FPDFPageObjectFromCPDFPageObject(pPathObj.release()); } FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV FPDFPageObj_CreateNewRect(float x, float y, float w, float h) { auto pPathObj = pdfium::MakeUnique<CPDF_PathObject>(); pPathObj->m_Path.AppendRect(x, y, x + w, y + h); pPathObj->DefaultStates(); // Caller takes ownership. return FPDFPageObjectFromCPDFPageObject(pPathObj.release()); } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_SetStrokeColor(FPDF_PAGEOBJECT path, unsigned int R, unsigned int G, unsigned int B, unsigned int A) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; return FPDFPageObj_SetStrokeColor(path, R, G, B, A); } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_GetStrokeColor(FPDF_PAGEOBJECT path, unsigned int* R, unsigned int* G, unsigned int* B, unsigned int* A) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; return FPDFPageObj_GetStrokeColor(path, R, G, B, A); } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_SetStrokeWidth(FPDF_PAGEOBJECT path, float width) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; return FPDFPageObj_SetStrokeWidth(path, width); } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_SetFillColor(FPDF_PAGEOBJECT path, unsigned int R, unsigned int G, unsigned int B, unsigned int A) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; return FPDFPageObj_SetFillColor(path, R, G, B, A); } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_GetFillColor(FPDF_PAGEOBJECT path, unsigned int* R, unsigned int* G, unsigned int* B, unsigned int* A) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; return FPDFPageObj_GetFillColor(path, R, G, B, A); } FPDF_EXPORT int FPDF_CALLCONV FPDFPath_CountSegments(FPDF_PAGEOBJECT path) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return -1; return pdfium::CollectionSize<int>(pPathObj->m_Path.GetPoints()); } FPDF_EXPORT FPDF_PATHSEGMENT FPDF_CALLCONV FPDFPath_GetPathSegment(FPDF_PAGEOBJECT path, int index) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return nullptr; const std::vector<FX_PATHPOINT>& points = pPathObj->m_Path.GetPoints(); if (!pdfium::IndexInBounds(points, index)) return nullptr; return FPDFPathSegmentFromFXPathPoint(&points[index]); } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_MoveTo(FPDF_PAGEOBJECT path, float x, float y) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; pPathObj->m_Path.AppendPoint(CFX_PointF(x, y), FXPT_TYPE::MoveTo, false); pPathObj->SetDirty(true); return true; } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_LineTo(FPDF_PAGEOBJECT path, float x, float y) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; pPathObj->m_Path.AppendPoint(CFX_PointF(x, y), FXPT_TYPE::LineTo, false); pPathObj->SetDirty(true); return true; } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_BezierTo(FPDF_PAGEOBJECT path, float x1, float y1, float x2, float y2, float x3, float y3) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; pPathObj->m_Path.AppendPoint(CFX_PointF(x1, y1), FXPT_TYPE::BezierTo, false); pPathObj->m_Path.AppendPoint(CFX_PointF(x2, y2), FXPT_TYPE::BezierTo, false); pPathObj->m_Path.AppendPoint(CFX_PointF(x3, y3), FXPT_TYPE::BezierTo, false); pPathObj->SetDirty(true); return true; } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_Close(FPDF_PAGEOBJECT path) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; if (pPathObj->m_Path.GetPoints().empty()) return false; pPathObj->m_Path.ClosePath(); pPathObj->SetDirty(true); return true; } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_SetDrawMode(FPDF_PAGEOBJECT path, int fillmode, FPDF_BOOL stroke) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; if (fillmode == FPDF_FILLMODE_ALTERNATE) pPathObj->m_FillType = FXFILL_ALTERNATE; else if (fillmode == FPDF_FILLMODE_WINDING) pPathObj->m_FillType = FXFILL_WINDING; else pPathObj->m_FillType = 0; pPathObj->m_bStroke = stroke != 0; pPathObj->SetDirty(true); return true; } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_GetDrawMode(FPDF_PAGEOBJECT path, int* fillmode, FPDF_BOOL* stroke) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj || !fillmode || !stroke) return false; if (pPathObj->m_FillType == FXFILL_ALTERNATE) *fillmode = FPDF_FILLMODE_ALTERNATE; else if (pPathObj->m_FillType == FXFILL_WINDING) *fillmode = FPDF_FILLMODE_WINDING; else *fillmode = FPDF_FILLMODE_NONE; *stroke = pPathObj->m_bStroke; return true; } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_GetMatrix(FPDF_PAGEOBJECT path, double* a, double* b, double* c, double* d, double* e, double* f) { if (!path || !a || !b || !c || !d || !e || !f) return false; CPDF_PathObject* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; *a = pPathObj->m_Matrix.a; *b = pPathObj->m_Matrix.b; *c = pPathObj->m_Matrix.c; *d = pPathObj->m_Matrix.d; *e = pPathObj->m_Matrix.e; *f = pPathObj->m_Matrix.f; return true; } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_SetMatrix(FPDF_PAGEOBJECT path, double a, double b, double c, double d, double e, double f) { if (!path) return false; CPDF_PathObject* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return false; pPathObj->m_Matrix.a = a; pPathObj->m_Matrix.b = b; pPathObj->m_Matrix.c = c; pPathObj->m_Matrix.d = d; pPathObj->m_Matrix.e = e; pPathObj->m_Matrix.f = f; pPathObj->SetDirty(true); return true; } FPDF_EXPORT void FPDF_CALLCONV FPDFPath_SetLineJoin(FPDF_PAGEOBJECT path, int line_join) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return; FPDFPageObj_SetLineJoin(path, line_join); } FPDF_EXPORT void FPDF_CALLCONV FPDFPath_SetLineCap(FPDF_PAGEOBJECT path, int line_cap) { auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path); if (!pPathObj) return; FPDFPageObj_SetLineCap(path, line_cap); } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPathSegment_GetPoint(FPDF_PATHSEGMENT segment, float* x, float* y) { auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment); if (!pPathPoint || !x || !y) return false; *x = pPathPoint->m_Point.x; *y = pPathPoint->m_Point.y; return true; } FPDF_EXPORT int FPDF_CALLCONV FPDFPathSegment_GetType(FPDF_PATHSEGMENT segment) { auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment); return pPathPoint ? static_cast<int>(pPathPoint->m_Type) : FPDF_SEGMENT_UNKNOWN; } FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPathSegment_GetClose(FPDF_PATHSEGMENT segment) { auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment); return pPathPoint ? pPathPoint->m_CloseFigure : false; }