Circle.cc 3.76 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
#include "Circle.h"



Circle::Circle(QObject *parent)
    :   QObject (parent)
    ,   _circleRadius(0)
    ,   _circleOrigin(QPointF(0,0))
{

}

Circle::Circle(double radius, QObject *parent)
    :   QObject (parent)
15
    ,   _circleRadius(radius >= 0 ? radius : 0)
16 17 18 19 20 21
    ,   _circleOrigin(QPointF(0,0))
{
}

Circle::Circle(double radius, QPointF origin, QObject *parent)
    :   QObject (parent)
22
    ,   _circleRadius(radius >= 0 ? radius : 0)
23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95
    ,   _circleOrigin(origin)
{
}

Circle::Circle(const Circle &other, QObject *parent)
    :   QObject (parent)
{
    *this = other;
}

Circle &Circle::operator=(const Circle &other)
{
    this->setRadius(other.radius());
    this->setOrigin(other.origin());

    return *this;
}

/*!
 * \fn void Circle::setRadius(double radius)
 * Sets the radius of the circle to \a radius
 */
void Circle::setRadius(double radius)
{
    if ( radius >= 0 && !qFuzzyCompare(_circleRadius, radius) ) {
        _circleRadius = radius;

        emit radiusChanged(_circleRadius);
    }
}

/*!
 * \fn void Circle::setOrigin(const QPointF &origin)
 * Sets the origin of the circle to \a origin
 *
 * \sa QPointF
 */
void Circle::setOrigin(const QPointF &origin)
{
    if (origin != _circleOrigin) {
        _circleOrigin = origin;

        emit originChanged(_circleOrigin);
    }
}

/*!
 * \fn double Circle::radius() const
 * Returns the radius of the circle.
 */
double Circle::radius() const
{
    return _circleRadius;
}

/*!
 * \fn QPointF Circle::origin() const
 * Returns the origin of the circle.
 *
 * \sa QPointF
 */
QPointF Circle::origin() const
{
    return _circleOrigin;
}


/*!
 * \fn QPolygonF Circle::approximate(int numberOfCorners)
 * Returns a polygon with \a numberOfCorners corners which approximates the circle.
 *
 * \sa QPointF
 */
96
QPolygonF Circle::approximate(int numberOfCorners) const
97 98 99
{
    if ( numberOfCorners < 3)
        return QPolygonF();
100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
    return approximateSektor(numberOfCorners, 0, 2*M_PI);
}

QPolygonF Circle::approximate(double angleDiscretisation) const
{
    return approximateSektor(angleDiscretisation, 0, 2*M_PI);
}

QPolygonF Circle::approximateSektor(int numberOfCorners, double alpha1, double alpha2) const
{
    return approximateSektor((alpha2-alpha1)/double(numberOfCorners-1), alpha1, alpha2);
}

QPolygonF Circle::approximateSektor(double angleDiscretisation, double alpha1, double alpha2) const
{
    // truncate alpha1 to [0, 2*pi], fmod() does not work in this case
    alpha1              = PlanimetryCalculus::truncateAngle(alpha1);
    alpha2              = PlanimetryCalculus::truncateAngle(alpha2);
    double deltaAlpha   = PlanimetryCalculus::truncateAngle(alpha2 - alpha1);
    angleDiscretisation = PlanimetryCalculus::truncateAngle(angleDiscretisation);

    if (angleDiscretisation > deltaAlpha || qFuzzyIsNull(angleDiscretisation))
        return QPolygonF();
123 124 125 126


    QPolygonF polygon;
    QPointF vertex(-_circleRadius,0); // initial vertex
127
    double currentAngle = alpha1;
128
    // rotate the vertex numberOfCorners-1 times add the origin and append to the polygon.
129
    while(currentAngle < alpha2) {
Valentin Platzgummer's avatar
Valentin Platzgummer committed
130
        PlanimetryCalculus::rotateReference(vertex, currentAngle);
131
        polygon.append(vertex + _circleOrigin);
132 133 134 135
        currentAngle = PlanimetryCalculus::truncateAngle(currentAngle + angleDiscretisation);
    }

    // append last point if necessarry
Valentin Platzgummer's avatar
Valentin Platzgummer committed
136
    PlanimetryCalculus::rotateReference(vertex, alpha2);
137 138 139 140
    vertex = vertex + _circleOrigin;
    if (   !qFuzzyIsNull(PlanimetryCalculus::distance(polygon.first(), vertex))
        && !qFuzzyIsNull(PlanimetryCalculus::distance(polygon.last(), vertex )) ){
        polygon.append(vertex);
141 142 143 144 145
    }

    return polygon;
}

146 147 148 149 150
bool Circle::isNull() const
{
    return _circleRadius <= 0 ? true : false;
}

151 152 153 154 155 156
/*!
 * \class Circle
 * \brief Provies a circle with radius and origin.
 *
 * \sa QPointF
 */