2017-07-28 16:27:46 +02:00
|
|
|
/*
|
|
|
|
* \brief Utilities for calculating bezier curves
|
|
|
|
* \date 2017-07-28
|
|
|
|
* \author Norman Feske
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 Genode Labs GmbH
|
|
|
|
*
|
|
|
|
* This file is part of the Genode OS framework, which is distributed
|
|
|
|
* under the terms of the GNU Affero General Public License version 3.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _INCLUDE__GEMS__BEZIER_H_
|
|
|
|
#define _INCLUDE__GEMS__BEZIER_H_
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate quadratic bezier curve
|
|
|
|
*
|
2024-05-22 17:22:53 +02:00
|
|
|
* \param draw_line_fn functor to be called to draw a line segment
|
|
|
|
* \param levels number of sub divisions
|
2017-07-28 16:27:46 +02:00
|
|
|
*
|
|
|
|
* The coordinates are specified in clock-wise order with point 1 being
|
|
|
|
* the start and point 3 the end of the curve.
|
|
|
|
*/
|
|
|
|
static inline void bezier(long x1, long y1, long x2, long y2, long x3, long y3,
|
2024-05-22 17:22:53 +02:00
|
|
|
auto const &draw_line_fn, unsigned levels)
|
2017-07-28 16:27:46 +02:00
|
|
|
{
|
|
|
|
if (levels-- == 0) {
|
2024-05-22 17:22:53 +02:00
|
|
|
draw_line_fn(x1, y1, x3, y3);
|
2017-07-28 16:27:46 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
long const x12 = (x1 + x2) / 2, y12 = (y1 + y2) / 2,
|
|
|
|
x23 = (x2 + x3) / 2, y23 = (y2 + y3) / 2,
|
|
|
|
x123 = (x12 + x23) / 2, y123 = (y12 + y23) / 2;
|
|
|
|
|
2024-05-22 17:22:53 +02:00
|
|
|
bezier(x1, y1, x12, y12, x123, y123, draw_line_fn, levels);
|
|
|
|
bezier(x123, y123, x23, y23, x3, y3, draw_line_fn, levels);
|
2017-07-28 16:27:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate cubic bezier curve
|
|
|
|
*
|
|
|
|
* The arguments correspond to those of the quadratic version but with point 4
|
|
|
|
* being the end of the curve.
|
|
|
|
*/
|
|
|
|
static inline void bezier(long x1, long y1, long x2, long y2,
|
|
|
|
long x3, long y3, long x4, long y4,
|
2024-05-22 17:22:53 +02:00
|
|
|
auto const &draw_line_fn, unsigned levels)
|
2017-07-28 16:27:46 +02:00
|
|
|
{
|
|
|
|
if (levels-- == 0) {
|
2024-05-22 17:22:53 +02:00
|
|
|
draw_line_fn(x1, y1, x4, y4);
|
2017-07-28 16:27:46 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
long const x12 = (x1 + x2) / 2, y12 = (y1 + y2) / 2,
|
|
|
|
x23 = (x2 + x3) / 2, y23 = (y2 + y3) / 2,
|
|
|
|
x34 = (x3 + x4) / 2, y34 = (y3 + y4) / 2,
|
|
|
|
x123 = (x12 + x23) / 2, y123 = (y12 + y23) / 2,
|
|
|
|
x234 = (x23 + x34) / 2, y234 = (y23 + y34) / 2,
|
|
|
|
x1234 = (x123 + x234) / 2, y1234 = (y123 + y234) / 2;
|
|
|
|
|
2024-05-22 17:22:53 +02:00
|
|
|
bezier(x1, y1, x12, y12, x123, y123, x1234, y1234, draw_line_fn, levels);
|
|
|
|
bezier(x1234, y1234, x234, y234, x34, y34, x4, y4, draw_line_fn, levels);
|
2017-07-28 16:27:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* _INCLUDE__GEMS__BEZIER_H_ */
|