| // 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; |
| } |