blob: aaa4b72fe33b6706e3f7446492cfe5e657df6fa6 [file] [log] [blame]
Nicolas Pena55e026b2017-02-07 14:59:23 -05001// Copyright 2017 PDFium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "public/fpdf_edit.h"
6
Lei Zhang02d42b62018-10-24 17:56:26 +00007#include <utility>
Dan Sinclair971a6742018-03-28 19:23:25 +00008#include <vector>
9
Nicolas Pena55e026b2017-02-07 14:59:23 -050010#include "core/fpdfapi/page/cpdf_path.h"
11#include "core/fpdfapi/page/cpdf_pathobject.h"
12#include "core/fxcrt/fx_system.h"
Dan Sinclair00d47a62018-03-28 18:39:04 +000013#include "fpdfsdk/cpdfsdk_helpers.h"
Tom Sepezfe91c6c2017-05-16 15:33:20 -070014#include "third_party/base/ptr_util.h"
Lei Zhang35d45872018-10-12 19:07:03 +000015#include "third_party/base/stl_util.h"
Nicolas Pena55e026b2017-02-07 14:59:23 -050016
wileyrya8a598332017-06-01 15:33:42 -050017// These checks are here because core/ and public/ cannot depend on each other.
18static_assert(CFX_GraphStateData::LineCapButt == FPDF_LINECAP_BUTT,
19 "CFX_GraphStateData::LineCapButt value mismatch");
20static_assert(CFX_GraphStateData::LineCapRound == FPDF_LINECAP_ROUND,
21 "CFX_GraphStateData::LineCapRound value mismatch");
22static_assert(CFX_GraphStateData::LineCapSquare ==
23 FPDF_LINECAP_PROJECTING_SQUARE,
24 "CFX_GraphStateData::LineCapSquare value mismatch");
25
26static_assert(CFX_GraphStateData::LineJoinMiter == FPDF_LINEJOIN_MITER,
27 "CFX_GraphStateData::LineJoinMiter value mismatch");
28static_assert(CFX_GraphStateData::LineJoinRound == FPDF_LINEJOIN_ROUND,
29 "CFX_GraphStateData::LineJoinRound value mismatch");
30static_assert(CFX_GraphStateData::LineJoinBevel == FPDF_LINEJOIN_BEVEL,
31 "CFX_GraphStateData::LineJoinBevel value mismatch");
32
Miklos Vajna36eed872017-09-20 22:52:43 +020033static_assert(static_cast<int>(FXPT_TYPE::LineTo) == FPDF_SEGMENT_LINETO,
34 "FXPT_TYPE::LineTo value mismatch");
35static_assert(static_cast<int>(FXPT_TYPE::BezierTo) == FPDF_SEGMENT_BEZIERTO,
36 "FXPT_TYPE::BezierTo value mismatch");
37static_assert(static_cast<int>(FXPT_TYPE::MoveTo) == FPDF_SEGMENT_MOVETO,
38 "FXPT_TYPE::MoveTo value mismatch");
39
Lei Zhang40d37b12018-03-20 21:06:35 +000040namespace {
41
Dan Sinclair7aba4722018-03-28 17:04:16 +000042CPDF_PathObject* CPDFPathObjectFromFPDFPageObject(FPDF_PAGEOBJECT page_object) {
43 auto* obj = CPDFPageObjectFromFPDFPageObject(page_object);
44 return obj ? obj->AsPath() : nullptr;
45}
46
Lei Zhang40d37b12018-03-20 21:06:35 +000047} // namespace
48
Dan Sinclair00d2ad12017-08-10 14:13:02 -040049FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV FPDFPageObj_CreateNewPath(float x,
50 float y) {
Henrique Nakashima354ed202018-06-05 18:56:52 +000051 auto pPathObj = pdfium::MakeUnique<CPDF_PathObject>();
Lei Zhange8355742018-10-24 18:46:26 +000052 pPathObj->path().AppendPoint(CFX_PointF(x, y), FXPT_TYPE::MoveTo, false);
Nicolas Pena0fc185e2017-02-08 12:13:20 -050053 pPathObj->DefaultStates();
Tom Sepez525147a2018-05-03 17:19:53 +000054
55 // Caller takes ownership.
56 return FPDFPageObjectFromCPDFPageObject(pPathObj.release());
Nicolas Pena55e026b2017-02-07 14:59:23 -050057}
58
Dan Sinclair00d2ad12017-08-10 14:13:02 -040059FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV FPDFPageObj_CreateNewRect(float x,
60 float y,
61 float w,
62 float h) {
Henrique Nakashima354ed202018-06-05 18:56:52 +000063 auto pPathObj = pdfium::MakeUnique<CPDF_PathObject>();
Lei Zhange8355742018-10-24 18:46:26 +000064 pPathObj->path().AppendRect(x, y, x + w, y + h);
Nicolas Pena0fc185e2017-02-08 12:13:20 -050065 pPathObj->DefaultStates();
Tom Sepez525147a2018-05-03 17:19:53 +000066
67 // Caller takes ownership.
68 return FPDFPageObjectFromCPDFPageObject(pPathObj.release());
Nicolas Pena55e026b2017-02-07 14:59:23 -050069}
70
Miklos Vajna0150a542017-09-21 21:46:56 +020071FPDF_EXPORT int FPDF_CALLCONV FPDFPath_CountSegments(FPDF_PAGEOBJECT path) {
Miklos Vajna12abfd02017-09-15 07:49:03 +020072 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
73 if (!pPathObj)
74 return -1;
Lei Zhange8355742018-10-24 18:46:26 +000075 return pdfium::CollectionSize<int>(pPathObj->path().GetPoints());
Miklos Vajna12abfd02017-09-15 07:49:03 +020076}
77
Miklos Vajna36eed872017-09-20 22:52:43 +020078FPDF_EXPORT FPDF_PATHSEGMENT FPDF_CALLCONV
79FPDFPath_GetPathSegment(FPDF_PAGEOBJECT path, int index) {
80 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
81 if (!pPathObj)
82 return nullptr;
83
Lei Zhange8355742018-10-24 18:46:26 +000084 const std::vector<FX_PATHPOINT>& points = pPathObj->path().GetPoints();
Tom Sepez525147a2018-05-03 17:19:53 +000085 if (!pdfium::IndexInBounds(points, index))
86 return nullptr;
87
88 return FPDFPathSegmentFromFXPathPoint(&points[index]);
Miklos Vajna36eed872017-09-20 22:52:43 +020089}
90
Dan Sinclair00d2ad12017-08-10 14:13:02 -040091FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_MoveTo(FPDF_PAGEOBJECT path,
92 float x,
93 float y) {
Lei Zhang038740c2017-09-13 13:22:54 -070094 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
95 if (!pPathObj)
Nicolas Pena55e026b2017-02-07 14:59:23 -050096 return false;
97
Lei Zhange8355742018-10-24 18:46:26 +000098 pPathObj->path().AppendPoint(CFX_PointF(x, y), FXPT_TYPE::MoveTo, false);
wileyryae858aa42017-05-31 14:49:05 -050099 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500100 return true;
101}
102
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400103FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_LineTo(FPDF_PAGEOBJECT path,
104 float x,
105 float y) {
Lei Zhang038740c2017-09-13 13:22:54 -0700106 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
107 if (!pPathObj)
Nicolas Pena55e026b2017-02-07 14:59:23 -0500108 return false;
109
Lei Zhange8355742018-10-24 18:46:26 +0000110 pPathObj->path().AppendPoint(CFX_PointF(x, y), FXPT_TYPE::LineTo, false);
wileyryae858aa42017-05-31 14:49:05 -0500111 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500112 return true;
113}
114
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400115FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_BezierTo(FPDF_PAGEOBJECT path,
116 float x1,
117 float y1,
118 float x2,
119 float y2,
120 float x3,
121 float y3) {
Lei Zhang038740c2017-09-13 13:22:54 -0700122 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
123 if (!pPathObj)
Nicolas Pena55e026b2017-02-07 14:59:23 -0500124 return false;
125
Lei Zhange8355742018-10-24 18:46:26 +0000126 CPDF_Path& cpath = pPathObj->path();
127 cpath.AppendPoint(CFX_PointF(x1, y1), FXPT_TYPE::BezierTo, false);
128 cpath.AppendPoint(CFX_PointF(x2, y2), FXPT_TYPE::BezierTo, false);
129 cpath.AppendPoint(CFX_PointF(x3, y3), FXPT_TYPE::BezierTo, false);
wileyryae858aa42017-05-31 14:49:05 -0500130 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500131 return true;
132}
133
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400134FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_Close(FPDF_PAGEOBJECT path) {
Lei Zhang038740c2017-09-13 13:22:54 -0700135 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
136 if (!pPathObj)
Nicolas Pena55e026b2017-02-07 14:59:23 -0500137 return false;
138
Lei Zhange8355742018-10-24 18:46:26 +0000139 CPDF_Path& cpath = pPathObj->path();
140 if (cpath.GetPoints().empty())
Nicolas Pena55e026b2017-02-07 14:59:23 -0500141 return false;
142
Lei Zhange8355742018-10-24 18:46:26 +0000143 cpath.ClosePath();
wileyryae858aa42017-05-31 14:49:05 -0500144 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500145 return true;
146}
147
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400148FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_SetDrawMode(FPDF_PAGEOBJECT path,
149 int fillmode,
150 FPDF_BOOL stroke) {
Jane Liu3b057432017-06-19 10:44:01 -0400151 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
Lei Zhang038740c2017-09-13 13:22:54 -0700152 if (!pPathObj)
153 return false;
Nicolas Pena55e026b2017-02-07 14:59:23 -0500154
Lei Zhange8355742018-10-24 18:46:26 +0000155 pPathObj->set_stroke(!!stroke);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500156 if (fillmode == FPDF_FILLMODE_ALTERNATE)
Tom Sepez4f5d2cf2019-12-16 23:38:05 +0000157 pPathObj->set_alternate_filltype();
Nicolas Pena55e026b2017-02-07 14:59:23 -0500158 else if (fillmode == FPDF_FILLMODE_WINDING)
Tom Sepez4f5d2cf2019-12-16 23:38:05 +0000159 pPathObj->set_winding_filltype();
Nicolas Pena55e026b2017-02-07 14:59:23 -0500160 else
Tom Sepez4f5d2cf2019-12-16 23:38:05 +0000161 pPathObj->set_no_filltype();
wileyryae858aa42017-05-31 14:49:05 -0500162 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500163 return true;
164}
wileyrya22a237f2017-05-26 09:26:27 -0500165
Miklos Vajna491112b2018-05-30 13:30:10 +0000166FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_GetDrawMode(FPDF_PAGEOBJECT path,
167 int* fillmode,
168 FPDF_BOOL* stroke) {
169 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
170 if (!pPathObj || !fillmode || !stroke)
171 return false;
172
Tom Sepez4f5d2cf2019-12-16 23:38:05 +0000173 if (pPathObj->has_alternate_filltype())
Miklos Vajna491112b2018-05-30 13:30:10 +0000174 *fillmode = FPDF_FILLMODE_ALTERNATE;
Tom Sepez4f5d2cf2019-12-16 23:38:05 +0000175 else if (pPathObj->has_winding_filltype())
Miklos Vajna491112b2018-05-30 13:30:10 +0000176 *fillmode = FPDF_FILLMODE_WINDING;
177 else
178 *fillmode = FPDF_FILLMODE_NONE;
179
Lei Zhang1313ca02018-10-24 18:38:57 +0000180 *stroke = pPathObj->stroke();
Miklos Vajna491112b2018-05-30 13:30:10 +0000181 return true;
182}
183
Miklos Vajna97f4d672018-06-04 14:47:17 +0000184FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_GetMatrix(FPDF_PAGEOBJECT path,
Lei Zhang8da98232019-12-11 23:29:33 +0000185 FS_MATRIX* matrix) {
186 if (!path || !matrix)
Miklos Vajna97f4d672018-06-04 14:47:17 +0000187 return false;
188
189 CPDF_PathObject* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
190 if (!pPathObj)
191 return false;
192
Lei Zhangc89c5822020-01-21 20:23:56 +0000193 *matrix = FSMatrixFromCFXMatrix(pPathObj->matrix());
Miklos Vajna97f4d672018-06-04 14:47:17 +0000194 return true;
195}
196
Lei Zhang8da98232019-12-11 23:29:33 +0000197FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
198FPDFPath_SetMatrix(FPDF_PAGEOBJECT path, const FS_MATRIX* matrix) {
199 if (!matrix)
200 return false;
201
Miklos Vajna97f4d672018-06-04 14:47:17 +0000202 CPDF_PathObject* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
203 if (!pPathObj)
204 return false;
205
Lei Zhang8da98232019-12-11 23:29:33 +0000206 pPathObj->set_matrix(CFXMatrixFromFSMatrix(*matrix));
Miklos Vajna97f4d672018-06-04 14:47:17 +0000207 pPathObj->SetDirty(true);
Miklos Vajna97f4d672018-06-04 14:47:17 +0000208 return true;
209}
210
Miklos Vajna36eed872017-09-20 22:52:43 +0200211FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
212FPDFPathSegment_GetPoint(FPDF_PATHSEGMENT segment, float* x, float* y) {
213 auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment);
214 if (!pPathPoint || !x || !y)
215 return false;
216
217 *x = pPathPoint->m_Point.x;
218 *y = pPathPoint->m_Point.y;
Miklos Vajna36eed872017-09-20 22:52:43 +0200219 return true;
220}
221
222FPDF_EXPORT int FPDF_CALLCONV
223FPDFPathSegment_GetType(FPDF_PATHSEGMENT segment) {
224 auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment);
Miklos Vajna36eed872017-09-20 22:52:43 +0200225 return pPathPoint ? static_cast<int>(pPathPoint->m_Type)
226 : FPDF_SEGMENT_UNKNOWN;
227}
228
229FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
230FPDFPathSegment_GetClose(FPDF_PATHSEGMENT segment) {
231 auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment);
Tom Sepez6a4c8f72019-12-12 20:56:43 +0000232 return pPathPoint && pPathPoint->m_CloseFigure;
Miklos Vajna36eed872017-09-20 22:52:43 +0200233}