1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
11 #include "opengl/RenderList.hxx"
12 #include "opengl/VertexUtils.hxx"
13 #include "opengl/LineRenderUtils.hxx"
15 #include <basegfx/polygon/b2dpolygontools.hxx>
16 #include <basegfx/polygon/b2dpolygontriangulator.hxx>
17 #include <basegfx/polygon/b2dpolypolygoncutter.hxx>
18 #include <basegfx/polygon/b2dtrapezoid.hxx>
23 /** Append vertices for the polyline
25 * OpenGL polyline drawing algorithm inspired by:
26 * - http://mattdesl.svbtle.com/drawing-lines-is-hard
27 * - https://www.mapbox.com/blog/drawing-antialiased-lines/
28 * - https://cesiumjs.org/2013/04/22/Robust-Polyline-Rendering-with-WebGL/
29 * - http://artgrammer.blogspot.si/2011/05/drawing-nearly-perfect-2d-line-segments.html
30 * - http://artgrammer.blogspot.si/2011/07/drawing-polylines-by-tessellation.html
33 void appendPolyLine(vcl::LineBuilder
& rBuilder
, const basegfx::B2DPolygon
& rPolygon
,
34 basegfx::B2DLineJoin eLineJoin
, css::drawing::LineCap eLineCap
,
35 double fMiterMinimumAngle
)
37 sal_uInt32 nPoints
= rPolygon
.count();
38 bool bClosed
= rPolygon
.isClosed();
40 if (nPoints
== 2 || eLineJoin
== basegfx::B2DLineJoin::NONE
)
42 // If line joint is NONE or a simple line with 2 points, draw the polyline
43 // each line segment separately.
45 for (sal_uInt32 i
= 0; i
< (bClosed
? nPoints
: nPoints
- 1); ++i
)
47 sal_uInt32 index1
= (i
+ 0) % nPoints
; // loop indices - important when polyline is closed
48 sal_uInt32 index2
= (i
+ 1) % nPoints
;
50 glm::vec2
aPoint1(rPolygon
.getB2DPoint(index1
).getX(), rPolygon
.getB2DPoint(index1
).getY());
51 glm::vec2
aPoint2(rPolygon
.getB2DPoint(index2
).getX(), rPolygon
.getB2DPoint(index2
).getY());
53 rBuilder
.appendLine(aPoint1
, aPoint2
);
59 int lastPoint
= int(nPoints
);
61 glm::vec2
p0(rPolygon
.getB2DPoint(nPoints
- 1).getX(), rPolygon
.getB2DPoint(nPoints
- 1).getY());
62 glm::vec2
p1(rPolygon
.getB2DPoint(0).getX(), rPolygon
.getB2DPoint(0).getY());
63 glm::vec2
p2(rPolygon
.getB2DPoint(1).getX(), rPolygon
.getB2DPoint(1).getY());
65 glm::vec2 nextLineVector
;
66 glm::vec2 previousLineVector
;
67 glm::vec2 normal
; // perpendicular to the line vector
69 nextLineVector
= vcl::vertex::normalize(p2
- p1
);
73 normal
= glm::vec2(-nextLineVector
.y
, nextLineVector
.x
); // make perpendicular
74 rBuilder
.appendAndConnectLinePoint(p1
, normal
, 1.0f
);
76 i
++; // first point done already
77 lastPoint
--; // last point will be calculated separately from the loop
80 previousLineVector
= nextLineVector
;
84 lastPoint
++; // we need to connect last point to first point so one more line segment to calculate
85 previousLineVector
= vcl::vertex::normalize(p1
- p0
);
88 for (; i
< lastPoint
; ++i
)
90 int index1
= (i
+ 0) % nPoints
; // loop indices - important when polyline is closed
91 int index2
= (i
+ 1) % nPoints
;
93 p1
= glm::vec2(rPolygon
.getB2DPoint(index1
).getX(), rPolygon
.getB2DPoint(index1
).getY());
94 p2
= glm::vec2(rPolygon
.getB2DPoint(index2
).getX(), rPolygon
.getB2DPoint(index2
).getY());
96 if (p1
== p2
) // skip equal points, normals could div-by-0
99 nextLineVector
= vcl::vertex::normalize(p2
- p1
);
101 if (eLineJoin
== basegfx::B2DLineJoin::Miter
)
103 if (vcl::vertex::lineVectorAngle(previousLineVector
, nextLineVector
) < fMiterMinimumAngle
)
104 rBuilder
.appendBevelJoint(p1
, previousLineVector
, nextLineVector
);
106 rBuilder
.appendMiterJoint(p1
, previousLineVector
, nextLineVector
);
108 else if (eLineJoin
== basegfx::B2DLineJoin::Bevel
)
110 rBuilder
.appendBevelJoint(p1
, previousLineVector
, nextLineVector
);
112 else if (eLineJoin
== basegfx::B2DLineJoin::Round
)
114 rBuilder
.appendRoundJoint(p1
, previousLineVector
, nextLineVector
);
117 previousLineVector
= nextLineVector
;
122 // Create vertices for the last point. There is no line join so just
123 // use the last line segment normal as the extrusion vector.
124 p1
= glm::vec2(rPolygon
.getB2DPoint(nPoints
- 1).getX(), rPolygon
.getB2DPoint(nPoints
- 1).getY());
125 normal
= glm::vec2(-previousLineVector
.y
, previousLineVector
.x
);
126 rBuilder
.appendAndConnectLinePoint(p1
, normal
, 1.0f
);
130 if (!bClosed
&& nPoints
>= 2 && (eLineCap
== css::drawing::LineCap_ROUND
|| eLineCap
== css::drawing::LineCap_SQUARE
))
132 glm::vec2
aBeginCapPoint1(rPolygon
.getB2DPoint(0).getX(), rPolygon
.getB2DPoint(0).getY());
133 glm::vec2
aBeginCapPoint2(rPolygon
.getB2DPoint(1).getX(), rPolygon
.getB2DPoint(1).getY());
135 glm::vec2
aEndCapPoint1(rPolygon
.getB2DPoint(nPoints
- 1).getX(), rPolygon
.getB2DPoint(nPoints
- 1).getY());
136 glm::vec2
aEndCapPoint2(rPolygon
.getB2DPoint(nPoints
- 2).getX(), rPolygon
.getB2DPoint(nPoints
- 2).getY());
138 if (eLineCap
== css::drawing::LineCap_ROUND
)
140 rBuilder
.appendRoundLineCapVertices(aBeginCapPoint1
, aBeginCapPoint2
);
141 rBuilder
.appendRoundLineCapVertices(aEndCapPoint1
, aEndCapPoint2
);
143 else if (eLineCap
== css::drawing::LineCap_SQUARE
)
145 rBuilder
.appendSquareLineCapVertices(aBeginCapPoint1
, aBeginCapPoint2
);
146 rBuilder
.appendSquareLineCapVertices(aEndCapPoint1
, aEndCapPoint2
);
151 inline void appendTrapezoid(std::vector
<Vertex
>& rVertices
, std::vector
<GLuint
>& rIndices
,
152 GLfloat x1
, GLfloat y1
, GLfloat x2
, GLfloat y2
,
153 GLfloat x3
, GLfloat y3
, GLfloat x4
, GLfloat y4
,
154 SalColor nColor
, GLfloat fTransparency
)
156 GLubyte nR
, nG
, nB
, nA
;
157 vcl::vertex::createColor(nColor
, fTransparency
, nR
, nG
, nB
, nA
);
159 GLuint zero
= rVertices
.size();
161 rVertices
.insert(rVertices
.end(), {
162 Vertex
{glm::vec2
{x1
, y1
}, glm::vec4
{nR
, nG
, nB
, nA
}, glm::vec4
{0.0f
, 0.0f
, 0.0f
, 0.0f
}},
163 Vertex
{glm::vec2
{x2
, y2
}, glm::vec4
{nR
, nG
, nB
, nA
}, glm::vec4
{0.0f
, 0.0f
, 0.0f
, 0.0f
}},
164 Vertex
{glm::vec2
{x3
, y3
}, glm::vec4
{nR
, nG
, nB
, nA
}, glm::vec4
{0.0f
, 0.0f
, 0.0f
, 0.0f
}},
165 Vertex
{glm::vec2
{x4
, y4
}, glm::vec4
{nR
, nG
, nB
, nA
}, glm::vec4
{0.0f
, 0.0f
, 0.0f
, 0.0f
}},
168 rIndices
.insert(rIndices
.end(), {
169 zero
+ 0, zero
+ 1, zero
+ 2,
170 zero
+ 2, zero
+ 1, zero
+ 3
174 void appendRectangle(std::vector
<Vertex
>& rVertices
, std::vector
<GLuint
>& rIndices
,
175 GLfloat x1
, GLfloat y1
, GLfloat x2
, GLfloat y2
,
176 SalColor nColor
, GLfloat fTransparency
)
178 GLubyte nR
, nG
, nB
, nA
;
179 vcl::vertex::createColor(nColor
, fTransparency
, nR
, nG
, nB
, nA
);
181 GLuint zero
= rVertices
.size();
183 rVertices
.insert(rVertices
.end(), {
184 Vertex
{glm::vec2
{x1
, y1
}, glm::vec4
{nR
, nG
, nB
, nA
}, glm::vec4
{0.0f
, 0.0f
, 0.0f
, 0.0f
}},
185 Vertex
{glm::vec2
{x2
, y1
}, glm::vec4
{nR
, nG
, nB
, nA
}, glm::vec4
{0.0f
, 0.0f
, 0.0f
, 0.0f
}},
186 Vertex
{glm::vec2
{x1
, y2
}, glm::vec4
{nR
, nG
, nB
, nA
}, glm::vec4
{0.0f
, 0.0f
, 0.0f
, 0.0f
}},
187 Vertex
{glm::vec2
{x2
, y2
}, glm::vec4
{nR
, nG
, nB
, nA
}, glm::vec4
{0.0f
, 0.0f
, 0.0f
, 0.0f
}},
190 rIndices
.insert(rIndices
.end(), {
191 zero
+ 0, zero
+ 1, zero
+ 2,
192 zero
+ 2, zero
+ 1, zero
+ 3
196 } // end anonymous namespace
198 void RenderList::addDrawPixel(long nX
, long nY
, SalColor nColor
)
200 if (nColor
== SALCOLOR_NONE
)
203 checkOverlapping(basegfx::B2DRange(nX
, nY
, nX
, nY
));
205 RenderParameters
& rRenderParameter
= maRenderEntries
.back().maTriangleParameters
;
206 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
207 nX
- 0.5f
, nY
- 0.5f
, nX
+ 0.5f
, nY
+ 0.5f
, nColor
, 0.0f
);
210 void RenderList::addDrawRectangle(long nX
, long nY
, long nWidth
, long nHeight
, double fTransparency
,
211 SalColor nLineColor
, SalColor nFillColor
)
213 if (nLineColor
== SALCOLOR_NONE
&& nFillColor
== SALCOLOR_NONE
)
215 if (fTransparency
== 1.0f
)
220 GLfloat
fX2(nX
+ nWidth
- 1);
221 GLfloat
fY2(nY
+ nHeight
- 1);
223 checkOverlapping(basegfx::B2DRange(fX1
, fY1
, fX2
, fY2
));
225 RenderParameters
& rRenderParameter
= maRenderEntries
.back().maTriangleParameters
;
227 // Draw rectangle stroke with line color
228 if (nLineColor
!= SALCOLOR_NONE
)
230 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
231 fX1
- 0.5f
, fY1
- 0.5f
, fX1
+ 0.5f
, fY2
+ 0.5f
, nLineColor
, fTransparency
);
232 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
233 fX1
- 0.5f
, fY1
- 0.5f
, fX2
+ 0.5f
, fY1
+ 0.5f
, nLineColor
, fTransparency
);
234 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
235 fX2
- 0.5f
, fY1
- 0.5f
, fX2
+ 0.5f
, fY2
+ 0.5f
, nLineColor
, fTransparency
);
236 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
237 fX1
- 0.5f
, fY2
- 0.5f
, fX2
+ 0.5f
, fY2
+ 0.5f
, nLineColor
, fTransparency
);
240 if (nFillColor
!= SALCOLOR_NONE
)
242 if (nLineColor
== SALCOLOR_NONE
)
244 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
245 fX1
- 0.5f
, fY1
- 0.5f
, fX1
+ 0.5f
, fY2
+ 0.5f
, nFillColor
, fTransparency
);
246 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
247 fX1
- 0.5f
, fY1
- 0.5f
, fX2
+ 0.5f
, fY1
+ 0.5f
, nFillColor
, fTransparency
);
248 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
249 fX2
- 0.5f
, fY1
- 0.5f
, fX2
+ 0.5f
, fY2
+ 0.5f
, nFillColor
, fTransparency
);
250 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
251 fX1
- 0.5f
, fY2
- 0.5f
, fX2
+ 0.5f
, fY2
+ 0.5f
, nFillColor
, fTransparency
);
253 // Draw rectangle fill with fill color
254 appendRectangle(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
,
255 fX1
+ 0.5f
, fY1
+ 0.5f
, fX2
- 0.5f
, fY2
- 0.5f
, nFillColor
, fTransparency
);
259 void RenderList::addDrawLine(long nX1
, long nY1
, long nX2
, long nY2
, SalColor nLineColor
, bool bUseAA
)
261 if (nLineColor
== SALCOLOR_NONE
)
264 checkOverlapping(basegfx::B2DRange(nX1
, nY1
, nX2
, nY2
));
266 RenderParameters
& rRenderParameter
= maRenderEntries
.back().maLineParameters
;
268 glm::vec2
aPoint1(nX1
, nY1
);
269 glm::vec2
aPoint2(nX2
, nY2
);
271 vcl::LineBuilder
aBuilder(rRenderParameter
.maVertices
, rRenderParameter
.maIndices
, nLineColor
, 0.0f
, 1.0f
, bUseAA
);
272 aBuilder
.appendLine(aPoint1
, aPoint2
);
275 void RenderList::addDrawPolyPolygon(const basegfx::B2DPolyPolygon
& rPolyPolygon
, double fTransparency
,
276 SalColor nLineColor
, SalColor nFillColor
, bool bUseAA
)
278 if (rPolyPolygon
.count() <= 0)
280 if (nLineColor
== SALCOLOR_NONE
&& nFillColor
== SALCOLOR_NONE
)
282 if (fTransparency
== 1.0)
285 checkOverlapping(rPolyPolygon
.getB2DRange());
287 RenderParameters
& rLineRenderParameter
= maRenderEntries
.back().maLineParameters
;
288 RenderParameters
& rTriangleRenderParameter
= maRenderEntries
.back().maTriangleParameters
;
290 if (nFillColor
!= SALCOLOR_NONE
)
292 basegfx::B2DTrapezoidVector aTrapezoidVector
;
293 basegfx::tools::trapezoidSubdivide(aTrapezoidVector
, rPolyPolygon
);
295 if (!aTrapezoidVector
.empty())
297 for (basegfx::B2DTrapezoid
& rTrapezoid
: aTrapezoidVector
)
299 GLfloat topX1
= rTrapezoid
.getTopXLeft();
300 GLfloat topX2
= rTrapezoid
.getTopXRight();
301 GLfloat topY
= rTrapezoid
.getTopY();
303 GLfloat bottomX1
= rTrapezoid
.getBottomXLeft();
304 GLfloat bottomX2
= rTrapezoid
.getBottomXRight();
305 GLfloat bottomY
= rTrapezoid
.getBottomY();
307 appendTrapezoid(rTriangleRenderParameter
.maVertices
, rTriangleRenderParameter
.maIndices
,
308 topX1
, topY
, topX2
, topY
,
309 bottomX1
, bottomY
, bottomX2
, bottomY
,
310 nFillColor
, fTransparency
);
315 if (nLineColor
!= SALCOLOR_NONE
|| bUseAA
)
317 SalColor nColor
= (nLineColor
== SALCOLOR_NONE
) ? nFillColor
: nLineColor
;
319 vcl::LineBuilder
aBuilder(rLineRenderParameter
.maVertices
, rLineRenderParameter
.maIndices
,
320 nColor
, fTransparency
, 1.0f
, bUseAA
);
322 for (const basegfx::B2DPolygon
& rPolygon
: rPolyPolygon
)
324 basegfx::B2DPolygon
aPolygon(rPolygon
);
325 if (rPolygon
.areControlPointsUsed())
326 aPolygon
= rPolygon
.getDefaultAdaptiveSubdivision();
328 sal_uInt32 nPoints
= aPolygon
.count();
332 GLfloat x1
, y1
, x2
, y2
;
333 sal_uInt32 index1
, index2
;
335 for (sal_uInt32 i
= 0; i
<= nPoints
; ++i
)
337 index1
= (i
) % nPoints
;
338 index2
= (i
+ 1) % nPoints
;
340 x1
= aPolygon
.getB2DPoint(index1
).getX();
341 y1
= aPolygon
.getB2DPoint(index1
).getY();
342 x2
= aPolygon
.getB2DPoint(index2
).getX();
343 y2
= aPolygon
.getB2DPoint(index2
).getY();
345 aBuilder
.appendLine(glm::vec2(x1
, y1
), glm::vec2(x2
, y2
));
351 bool RenderList::addDrawTextureWithMaskColor(OpenGLTexture
& rTexture
, SalColor nColor
, const SalTwoRect
& r2Rect
)
356 GLfloat fX1
= r2Rect
.mnDestX
;
357 GLfloat fY1
= r2Rect
.mnDestY
;
358 GLfloat fX2
= fX1
+ r2Rect
.mnDestWidth
;
359 GLfloat fY2
= fY1
+ r2Rect
.mnDestHeight
;
361 checkOverlapping(basegfx::B2DRange(fX1
, fY1
, fX2
, fY2
));
363 GLuint nTextureId
= rTexture
.Id();
365 RenderTextureParameters
& rTextureParameter
= maRenderEntries
.back().maTextureParametersMap
[nTextureId
];
366 rTextureParameter
.maTexture
= rTexture
;
368 rTexture
.FillCoords
<GL_TRIANGLES
>(rTextureParameter
.maTextureCoords
, r2Rect
, false);
370 vcl::vertex::addRectangle
<GL_TRIANGLES
>(rTextureParameter
.maVertices
, fX1
, fY1
, fX2
, fY2
);
371 vcl::vertex::addQuadColors
<GL_TRIANGLES
>(rTextureParameter
.maColors
, nColor
, 0.0f
);
376 void RenderList::addDrawPolyLine(const basegfx::B2DPolygon
& rPolygon
, double fTransparency
,
377 const basegfx::B2DVector
& rLineWidth
, basegfx::B2DLineJoin eLineJoin
,
378 css::drawing::LineCap eLineCap
, double fMiterMinimumAngle
,
379 SalColor nLineColor
, bool bUseAA
)
381 if (rPolygon
.count() <= 1)
383 if (nLineColor
== SALCOLOR_NONE
)
385 if (fTransparency
== 1.0)
388 const bool bIsHairline
= (rLineWidth
.getX() == rLineWidth
.getY()) && (rLineWidth
.getX() <= 1.2);
389 const float fLineWidth
= bIsHairline
? 1.0f
: rLineWidth
.getX();
391 basegfx::B2DPolygon
aPolygon(rPolygon
);
392 if (rPolygon
.areControlPointsUsed())
393 aPolygon
= rPolygon
.getDefaultAdaptiveSubdivision();
395 checkOverlapping(aPolygon
.getB2DRange());
397 RenderParameters
& rParameter
= maRenderEntries
.back().maLineParameters
;
399 vcl::LineBuilder
aBuilder(rParameter
.maVertices
, rParameter
.maIndices
,
400 nLineColor
, fTransparency
, fLineWidth
, bUseAA
);
402 appendPolyLine(aBuilder
, aPolygon
, eLineJoin
, eLineCap
, fMiterMinimumAngle
);
405 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */