blob: 30c6cb0698fba03adcb03f6b54cc5b74376f3850 [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
Miklos Vajna36eed872017-09-20 22:52:43 +02005#include <vector>
6
Nicolas Pena55e026b2017-02-07 14:59:23 -05007#include "public/fpdf_edit.h"
8
9#include "core/fpdfapi/page/cpdf_path.h"
10#include "core/fpdfapi/page/cpdf_pathobject.h"
11#include "core/fxcrt/fx_system.h"
wileyrya864e9fb2017-05-26 11:38:14 -050012#include "fpdfsdk/fsdk_define.h"
Tom Sepezfe91c6c2017-05-16 15:33:20 -070013#include "third_party/base/ptr_util.h"
Nicolas Pena55e026b2017-02-07 14:59:23 -050014
wileyrya8a598332017-06-01 15:33:42 -050015// These checks are here because core/ and public/ cannot depend on each other.
16static_assert(CFX_GraphStateData::LineCapButt == FPDF_LINECAP_BUTT,
17 "CFX_GraphStateData::LineCapButt value mismatch");
18static_assert(CFX_GraphStateData::LineCapRound == FPDF_LINECAP_ROUND,
19 "CFX_GraphStateData::LineCapRound value mismatch");
20static_assert(CFX_GraphStateData::LineCapSquare ==
21 FPDF_LINECAP_PROJECTING_SQUARE,
22 "CFX_GraphStateData::LineCapSquare value mismatch");
23
24static_assert(CFX_GraphStateData::LineJoinMiter == FPDF_LINEJOIN_MITER,
25 "CFX_GraphStateData::LineJoinMiter value mismatch");
26static_assert(CFX_GraphStateData::LineJoinRound == FPDF_LINEJOIN_ROUND,
27 "CFX_GraphStateData::LineJoinRound value mismatch");
28static_assert(CFX_GraphStateData::LineJoinBevel == FPDF_LINEJOIN_BEVEL,
29 "CFX_GraphStateData::LineJoinBevel value mismatch");
30
Miklos Vajna36eed872017-09-20 22:52:43 +020031static_assert(static_cast<int>(FXPT_TYPE::LineTo) == FPDF_SEGMENT_LINETO,
32 "FXPT_TYPE::LineTo value mismatch");
33static_assert(static_cast<int>(FXPT_TYPE::BezierTo) == FPDF_SEGMENT_BEZIERTO,
34 "FXPT_TYPE::BezierTo value mismatch");
35static_assert(static_cast<int>(FXPT_TYPE::MoveTo) == FPDF_SEGMENT_MOVETO,
36 "FXPT_TYPE::MoveTo value mismatch");
37
Dan Sinclair00d2ad12017-08-10 14:13:02 -040038FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV FPDFPageObj_CreateNewPath(float x,
39 float y) {
Tom Sepezfe91c6c2017-05-16 15:33:20 -070040 auto pPathObj = pdfium::MakeUnique<CPDF_PathObject>();
dan sinclairb147e072017-02-22 19:56:15 -050041 pPathObj->m_Path.AppendPoint(CFX_PointF(x, y), FXPT_TYPE::MoveTo, false);
Nicolas Pena0fc185e2017-02-08 12:13:20 -050042 pPathObj->DefaultStates();
Tom Sepezfe91c6c2017-05-16 15:33:20 -070043 return pPathObj.release(); // Caller takes ownership.
Nicolas Pena55e026b2017-02-07 14:59:23 -050044}
45
Dan Sinclair00d2ad12017-08-10 14:13:02 -040046FPDF_EXPORT FPDF_PAGEOBJECT FPDF_CALLCONV FPDFPageObj_CreateNewRect(float x,
47 float y,
48 float w,
49 float h) {
Tom Sepezfe91c6c2017-05-16 15:33:20 -070050 auto pPathObj = pdfium::MakeUnique<CPDF_PathObject>();
Nicolas Pena55e026b2017-02-07 14:59:23 -050051 pPathObj->m_Path.AppendRect(x, y, x + w, y + h);
Nicolas Pena0fc185e2017-02-08 12:13:20 -050052 pPathObj->DefaultStates();
Tom Sepezfe91c6c2017-05-16 15:33:20 -070053 return pPathObj.release(); // Caller takes ownership.
Nicolas Pena55e026b2017-02-07 14:59:23 -050054}
55
Dan Sinclair00d2ad12017-08-10 14:13:02 -040056FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
57FPDFPath_SetStrokeColor(FPDF_PAGEOBJECT path,
58 unsigned int R,
59 unsigned int G,
60 unsigned int B,
61 unsigned int A) {
Lei Zhang038740c2017-09-13 13:22:54 -070062 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
63 if (!pPathObj || R > 255 || G > 255 || B > 255 || A > 255)
Nicolas Pena55e026b2017-02-07 14:59:23 -050064 return false;
65
Tom Sepezfe91c6c2017-05-16 15:33:20 -070066 float rgb[3] = {R / 255.f, G / 255.f, B / 255.f};
Nicolas Pena55e026b2017-02-07 14:59:23 -050067 pPathObj->m_GeneralState.SetStrokeAlpha(A / 255.f);
Nicolas Pena55e026b2017-02-07 14:59:23 -050068 pPathObj->m_ColorState.SetStrokeColor(
69 CPDF_ColorSpace::GetStockCS(PDFCS_DEVICERGB), rgb, 3);
wileyryae858aa42017-05-31 14:49:05 -050070 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -050071 return true;
72}
73
Dan Sinclair00d2ad12017-08-10 14:13:02 -040074FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
75FPDFPath_GetStrokeColor(FPDF_PAGEOBJECT path,
76 unsigned int* R,
77 unsigned int* G,
78 unsigned int* B,
79 unsigned int* A) {
Lei Zhang038740c2017-09-13 13:22:54 -070080 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
81 if (!pPathObj || !R || !G || !B || !A)
Jane Liu3b057432017-06-19 10:44:01 -040082 return false;
83
Jane Liu3b057432017-06-19 10:44:01 -040084 uint32_t strokeRGB = pPathObj->m_ColorState.GetStrokeRGB();
85 *R = FXSYS_GetRValue(strokeRGB);
86 *G = FXSYS_GetGValue(strokeRGB);
87 *B = FXSYS_GetBValue(strokeRGB);
88 *A = static_cast<unsigned int>(pPathObj->m_GeneralState.GetStrokeAlpha() *
89 255.f);
90 return true;
91}
92
Dan Sinclair00d2ad12017-08-10 14:13:02 -040093FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
94FPDFPath_SetStrokeWidth(FPDF_PAGEOBJECT path, float width) {
Lei Zhang038740c2017-09-13 13:22:54 -070095 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
96 if (!pPathObj || width < 0.0f)
Nicolas Pena2eb1a702017-02-09 18:17:33 -050097 return false;
98
Nicolas Pena2eb1a702017-02-09 18:17:33 -050099 pPathObj->m_GraphState.SetLineWidth(width);
wileyryae858aa42017-05-31 14:49:05 -0500100 pPathObj->SetDirty(true);
Nicolas Pena2eb1a702017-02-09 18:17:33 -0500101 return true;
102}
103
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400104FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_SetFillColor(FPDF_PAGEOBJECT path,
105 unsigned int R,
106 unsigned int G,
107 unsigned int B,
108 unsigned int A) {
wileyrya864e9fb2017-05-26 11:38:14 -0500109 return FPDFPageObj_SetFillColor(path, R, G, B, A);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500110}
111
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400112FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_GetFillColor(FPDF_PAGEOBJECT path,
113 unsigned int* R,
114 unsigned int* G,
115 unsigned int* B,
116 unsigned int* A) {
Lei Zhang038740c2017-09-13 13:22:54 -0700117 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
118 if (!pPathObj || !R || !G || !B || !A)
Miklos Vajnaed4705b2017-04-05 09:24:50 +0200119 return false;
120
Miklos Vajnaed4705b2017-04-05 09:24:50 +0200121 uint32_t fillRGB = pPathObj->m_ColorState.GetFillRGB();
122 *R = FXSYS_GetRValue(fillRGB);
123 *G = FXSYS_GetGValue(fillRGB);
124 *B = FXSYS_GetBValue(fillRGB);
125 *A = static_cast<unsigned int>(pPathObj->m_GeneralState.GetFillAlpha() *
126 255.f);
127 return true;
128}
129
Miklos Vajna0150a542017-09-21 21:46:56 +0200130FPDF_EXPORT int FPDF_CALLCONV FPDFPath_CountSegments(FPDF_PAGEOBJECT path) {
Miklos Vajna12abfd02017-09-15 07:49:03 +0200131 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
132 if (!pPathObj)
133 return -1;
134 return pdfium::CollectionSize<int>(pPathObj->m_Path.GetPoints());
135}
136
Miklos Vajna36eed872017-09-20 22:52:43 +0200137FPDF_EXPORT FPDF_PATHSEGMENT FPDF_CALLCONV
138FPDFPath_GetPathSegment(FPDF_PAGEOBJECT path, int index) {
139 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
140 if (!pPathObj)
141 return nullptr;
142
143 const std::vector<FX_PATHPOINT>& points = pPathObj->m_Path.GetPoints();
144 return pdfium::IndexInBounds(points, index) ? &points[index] : nullptr;
145}
146
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400147FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_MoveTo(FPDF_PAGEOBJECT path,
148 float x,
149 float y) {
Lei Zhang038740c2017-09-13 13:22:54 -0700150 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
151 if (!pPathObj)
Nicolas Pena55e026b2017-02-07 14:59:23 -0500152 return false;
153
dan sinclairb147e072017-02-22 19:56:15 -0500154 pPathObj->m_Path.AppendPoint(CFX_PointF(x, y), FXPT_TYPE::MoveTo, false);
wileyryae858aa42017-05-31 14:49:05 -0500155 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500156 return true;
157}
158
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400159FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_LineTo(FPDF_PAGEOBJECT path,
160 float x,
161 float y) {
Lei Zhang038740c2017-09-13 13:22:54 -0700162 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
163 if (!pPathObj)
Nicolas Pena55e026b2017-02-07 14:59:23 -0500164 return false;
165
dan sinclairb147e072017-02-22 19:56:15 -0500166 pPathObj->m_Path.AppendPoint(CFX_PointF(x, y), FXPT_TYPE::LineTo, false);
wileyryae858aa42017-05-31 14:49:05 -0500167 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500168 return true;
169}
170
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400171FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_BezierTo(FPDF_PAGEOBJECT path,
172 float x1,
173 float y1,
174 float x2,
175 float y2,
176 float x3,
177 float y3) {
Lei Zhang038740c2017-09-13 13:22:54 -0700178 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
179 if (!pPathObj)
Nicolas Pena55e026b2017-02-07 14:59:23 -0500180 return false;
181
dan sinclairb147e072017-02-22 19:56:15 -0500182 pPathObj->m_Path.AppendPoint(CFX_PointF(x1, y1), FXPT_TYPE::BezierTo, false);
183 pPathObj->m_Path.AppendPoint(CFX_PointF(x2, y2), FXPT_TYPE::BezierTo, false);
184 pPathObj->m_Path.AppendPoint(CFX_PointF(x3, y3), FXPT_TYPE::BezierTo, false);
wileyryae858aa42017-05-31 14:49:05 -0500185 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500186 return true;
187}
188
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400189FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_Close(FPDF_PAGEOBJECT path) {
Lei Zhang038740c2017-09-13 13:22:54 -0700190 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
191 if (!pPathObj)
Nicolas Pena55e026b2017-02-07 14:59:23 -0500192 return false;
193
Dan Sinclaire4602322017-02-15 11:07:32 -0500194 if (pPathObj->m_Path.GetPoints().empty())
Nicolas Pena55e026b2017-02-07 14:59:23 -0500195 return false;
196
Dan Sinclaire4602322017-02-15 11:07:32 -0500197 pPathObj->m_Path.ClosePath();
wileyryae858aa42017-05-31 14:49:05 -0500198 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500199 return true;
200}
201
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400202FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV FPDFPath_SetDrawMode(FPDF_PAGEOBJECT path,
203 int fillmode,
204 FPDF_BOOL stroke) {
Jane Liu3b057432017-06-19 10:44:01 -0400205 auto* pPathObj = CPDFPathObjectFromFPDFPageObject(path);
Lei Zhang038740c2017-09-13 13:22:54 -0700206 if (!pPathObj)
207 return false;
Nicolas Pena55e026b2017-02-07 14:59:23 -0500208
209 if (fillmode == FPDF_FILLMODE_ALTERNATE)
210 pPathObj->m_FillType = FXFILL_ALTERNATE;
211 else if (fillmode == FPDF_FILLMODE_WINDING)
212 pPathObj->m_FillType = FXFILL_WINDING;
213 else
214 pPathObj->m_FillType = 0;
215 pPathObj->m_bStroke = stroke != 0;
wileyryae858aa42017-05-31 14:49:05 -0500216 pPathObj->SetDirty(true);
Nicolas Pena55e026b2017-02-07 14:59:23 -0500217 return true;
218}
wileyrya22a237f2017-05-26 09:26:27 -0500219
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400220FPDF_EXPORT void FPDF_CALLCONV FPDFPath_SetLineJoin(FPDF_PAGEOBJECT path,
221 int line_join) {
wileyrya22a237f2017-05-26 09:26:27 -0500222 if (!path)
223 return;
224 if (line_join <
225 static_cast<int>(CFX_GraphStateData::LineJoin::LineJoinMiter) ||
226 line_join >
227 static_cast<int>(CFX_GraphStateData::LineJoin::LineJoinBevel)) {
228 return;
229 }
Jane Liu1a084022017-06-29 19:47:12 -0400230 auto* pPathObj = CPDFPageObjectFromFPDFPageObject(path);
wileyrya22a237f2017-05-26 09:26:27 -0500231 CFX_GraphStateData::LineJoin lineJoin =
232 static_cast<CFX_GraphStateData::LineJoin>(line_join);
233 pPathObj->m_GraphState.SetLineJoin(lineJoin);
wileyryae858aa42017-05-31 14:49:05 -0500234 pPathObj->SetDirty(true);
wileyrya22a237f2017-05-26 09:26:27 -0500235}
236
Dan Sinclair00d2ad12017-08-10 14:13:02 -0400237FPDF_EXPORT void FPDF_CALLCONV FPDFPath_SetLineCap(FPDF_PAGEOBJECT path,
238 int line_cap) {
wileyrya22a237f2017-05-26 09:26:27 -0500239 if (!path)
240 return;
241 if (line_cap < static_cast<int>(CFX_GraphStateData::LineCap::LineCapButt) ||
242 line_cap > static_cast<int>(CFX_GraphStateData::LineCap::LineCapSquare)) {
243 return;
244 }
Jane Liu1a084022017-06-29 19:47:12 -0400245 auto* pPathObj = CPDFPageObjectFromFPDFPageObject(path);
wileyrya22a237f2017-05-26 09:26:27 -0500246 CFX_GraphStateData::LineCap lineCap =
247 static_cast<CFX_GraphStateData::LineCap>(line_cap);
248 pPathObj->m_GraphState.SetLineCap(lineCap);
wileyryae858aa42017-05-31 14:49:05 -0500249 pPathObj->SetDirty(true);
wileyrya22a237f2017-05-26 09:26:27 -0500250}
Miklos Vajna36eed872017-09-20 22:52:43 +0200251
252FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
253FPDFPathSegment_GetPoint(FPDF_PATHSEGMENT segment, float* x, float* y) {
254 auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment);
255 if (!pPathPoint || !x || !y)
256 return false;
257
258 *x = pPathPoint->m_Point.x;
259 *y = pPathPoint->m_Point.y;
260
261 return true;
262}
263
264FPDF_EXPORT int FPDF_CALLCONV
265FPDFPathSegment_GetType(FPDF_PATHSEGMENT segment) {
266 auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment);
267
268 return pPathPoint ? static_cast<int>(pPathPoint->m_Type)
269 : FPDF_SEGMENT_UNKNOWN;
270}
271
272FPDF_EXPORT FPDF_BOOL FPDF_CALLCONV
273FPDFPathSegment_GetClose(FPDF_PATHSEGMENT segment) {
274 auto* pPathPoint = FXPathPointFromFPDFPathSegment(segment);
275
276 return pPathPoint ? pPathPoint->m_CloseFigure : false;
277}