| 1 | /**************************************************************************** |
| 2 | ** |
| 3 | ** Copyright (C) 2016 The Qt Company Ltd. |
| 4 | ** Contact: https://www.qt.io/licensing/ |
| 5 | ** |
| 6 | ** This file is part of the QtPositioning module of the Qt Toolkit. |
| 7 | ** |
| 8 | ** $QT_BEGIN_LICENSE:LGPL$ |
| 9 | ** Commercial License Usage |
| 10 | ** Licensees holding valid commercial Qt licenses may use this file in |
| 11 | ** accordance with the commercial license agreement provided with the |
| 12 | ** Software or, alternatively, in accordance with the terms contained in |
| 13 | ** a written agreement between you and The Qt Company. For licensing terms |
| 14 | ** and conditions see https://www.qt.io/terms-conditions. For further |
| 15 | ** information use the contact form at https://www.qt.io/contact-us. |
| 16 | ** |
| 17 | ** GNU Lesser General Public License Usage |
| 18 | ** Alternatively, this file may be used under the terms of the GNU Lesser |
| 19 | ** General Public License version 3 as published by the Free Software |
| 20 | ** Foundation and appearing in the file LICENSE.LGPL3 included in the |
| 21 | ** packaging of this file. Please review the following information to |
| 22 | ** ensure the GNU Lesser General Public License version 3 requirements |
| 23 | ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. |
| 24 | ** |
| 25 | ** GNU General Public License Usage |
| 26 | ** Alternatively, this file may be used under the terms of the GNU |
| 27 | ** General Public License version 2.0 or (at your option) the GNU General |
| 28 | ** Public license version 3 or any later version approved by the KDE Free |
| 29 | ** Qt Foundation. The licenses are as published by the Free Software |
| 30 | ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 |
| 31 | ** included in the packaging of this file. Please review the following |
| 32 | ** information to ensure the GNU General Public License requirements will |
| 33 | ** be met: https://www.gnu.org/licenses/gpl-2.0.html and |
| 34 | ** https://www.gnu.org/licenses/gpl-3.0.html. |
| 35 | ** |
| 36 | ** $QT_END_LICENSE$ |
| 37 | ** |
| 38 | ****************************************************************************/ |
| 39 | |
| 40 | #include "qgeoshape.h" |
| 41 | #include "qgeoshape_p.h" |
| 42 | #include "qgeorectangle.h" |
| 43 | #include "qgeocircle.h" |
| 44 | #include "qgeopath.h" |
| 45 | #include "qgeopolygon.h" |
| 46 | |
| 47 | |
| 48 | #ifndef QT_NO_DEBUG_STREAM |
| 49 | #include <QtCore/QDebug> |
| 50 | #endif |
| 51 | |
| 52 | #ifndef QT_NO_DATASTREAM |
| 53 | #include <QtCore/QDataStream> |
| 54 | #endif |
| 55 | |
| 56 | QT_BEGIN_NAMESPACE |
| 57 | |
| 58 | QGeoShapePrivate::QGeoShapePrivate(QGeoShape::ShapeType type) |
| 59 | : type(type) |
| 60 | { |
| 61 | } |
| 62 | |
| 63 | QGeoShapePrivate::~QGeoShapePrivate() |
| 64 | { |
| 65 | } |
| 66 | |
| 67 | bool QGeoShapePrivate::operator==(const QGeoShapePrivate &other) const |
| 68 | { |
| 69 | return type == other.type; |
| 70 | } |
| 71 | |
| 72 | /*! |
| 73 | \class QGeoShape |
| 74 | \inmodule QtPositioning |
| 75 | \ingroup QtPositioning-positioning |
| 76 | \since 5.2 |
| 77 | |
| 78 | \brief The QGeoShape class defines a geographic area. |
| 79 | |
| 80 | This class is the base class for classes which specify a geographic |
| 81 | area. |
| 82 | |
| 83 | For the sake of consistency, subclasses should describe the specific |
| 84 | details of the associated areas in terms of QGeoCoordinate instances |
| 85 | and distances in meters. |
| 86 | |
| 87 | This class is a \l Q_GADGET since Qt 5.5. It can be |
| 88 | \l{Cpp_value_integration_positioning}{directly used from C++ and QML}. |
| 89 | */ |
| 90 | |
| 91 | /*! |
| 92 | \enum QGeoShape::ShapeType |
| 93 | |
| 94 | Describes the type of the shape. |
| 95 | |
| 96 | \value UnknownType A shape of unknown type |
| 97 | \value RectangleType A rectangular shape |
| 98 | \value CircleType A circular shape |
| 99 | \value PathType A path type |
| 100 | \value PolygonType A polygon type |
| 101 | */ |
| 102 | |
| 103 | /*! |
| 104 | \property QGeoShape::type |
| 105 | \brief This property holds the type of this geo shape. |
| 106 | |
| 107 | While this property is introduced in Qt 5.5, the related accessor functions |
| 108 | exist since the first version of this class. |
| 109 | |
| 110 | \since 5.5 |
| 111 | */ |
| 112 | |
| 113 | /*! |
| 114 | \property QGeoShape::isValid |
| 115 | \brief This property holds the validity of the geo shape. |
| 116 | |
| 117 | A geo shape is considered to be invalid if some of the data that is required to |
| 118 | unambiguously describe the geo shape has not been set or has been set to an |
| 119 | unsuitable value depending on the subclass of this object. The default constructed |
| 120 | objects of this type are invalid. |
| 121 | |
| 122 | While this property is introduced in Qt 5.5, the related accessor functions |
| 123 | exist since the first version of this class. |
| 124 | |
| 125 | \since 5.5 |
| 126 | */ |
| 127 | |
| 128 | /*! |
| 129 | \property QGeoShape::isEmpty |
| 130 | \brief This property defines whether this geo shape is empty. |
| 131 | |
| 132 | An empty geo shape is a region which has a geometrical area of 0. |
| 133 | |
| 134 | While this property is introduced in Qt 5.5, the related accessor functions |
| 135 | exist since the first version of this class. |
| 136 | |
| 137 | \since 5.5 |
| 138 | */ |
| 139 | inline QGeoShapePrivate *QGeoShape::d_func() |
| 140 | { |
| 141 | return static_cast<QGeoShapePrivate *>(d_ptr.data()); |
| 142 | } |
| 143 | |
| 144 | inline const QGeoShapePrivate *QGeoShape::d_func() const |
| 145 | { |
| 146 | return static_cast<const QGeoShapePrivate *>(d_ptr.constData()); |
| 147 | } |
| 148 | |
| 149 | /*! |
| 150 | Constructs a new invalid geo shape of \l UnknownType. |
| 151 | */ |
| 152 | QGeoShape::QGeoShape() |
| 153 | { |
| 154 | } |
| 155 | |
| 156 | /*! |
| 157 | Constructs a new geo shape which is a copy of \a other. |
| 158 | */ |
| 159 | QGeoShape::QGeoShape(const QGeoShape &other) |
| 160 | : d_ptr(other.d_ptr) |
| 161 | { |
| 162 | } |
| 163 | |
| 164 | /*! |
| 165 | \internal |
| 166 | */ |
| 167 | QGeoShape::QGeoShape(QGeoShapePrivate *d) |
| 168 | : d_ptr(d) |
| 169 | { |
| 170 | } |
| 171 | |
| 172 | /*! |
| 173 | Destroys this geo shape. |
| 174 | */ |
| 175 | QGeoShape::~QGeoShape() |
| 176 | { |
| 177 | } |
| 178 | |
| 179 | /*! |
| 180 | Returns the type of this geo shape. |
| 181 | */ |
| 182 | QGeoShape::ShapeType QGeoShape::type() const |
| 183 | { |
| 184 | Q_D(const QGeoShape); |
| 185 | |
| 186 | if (d) |
| 187 | return d->type; |
| 188 | else |
| 189 | return UnknownType; |
| 190 | } |
| 191 | |
| 192 | /*! |
| 193 | Returns whether this geo shape is valid. |
| 194 | |
| 195 | */ |
| 196 | bool QGeoShape::isValid() const |
| 197 | { |
| 198 | Q_D(const QGeoShape); |
| 199 | |
| 200 | if (d) |
| 201 | return d->isValid(); |
| 202 | else |
| 203 | return false; |
| 204 | } |
| 205 | |
| 206 | /*! |
| 207 | Returns whether this geo shape is empty. |
| 208 | |
| 209 | An empty geo shape is a region which has a geometrical area of 0. |
| 210 | */ |
| 211 | bool QGeoShape::isEmpty() const |
| 212 | { |
| 213 | Q_D(const QGeoShape); |
| 214 | |
| 215 | if (d) |
| 216 | return d->isEmpty(); |
| 217 | else |
| 218 | return true; |
| 219 | } |
| 220 | |
| 221 | /*! |
| 222 | Returns whether the coordinate \a coordinate is contained within this geo shape. |
| 223 | */ |
| 224 | bool QGeoShape::contains(const QGeoCoordinate &coordinate) const |
| 225 | { |
| 226 | Q_D(const QGeoShape); |
| 227 | |
| 228 | if (d) |
| 229 | return d->contains(coordinate); |
| 230 | else |
| 231 | return false; |
| 232 | } |
| 233 | |
| 234 | /*! |
| 235 | Returns a QGeoRectangle representing the geographical bounding rectangle of the |
| 236 | geo shape, that defines the latitudinal/longitudinal bounds of the geo shape. |
| 237 | |
| 238 | \since 5.9 |
| 239 | */ |
| 240 | QGeoRectangle QGeoShape::boundingGeoRectangle() const |
| 241 | { |
| 242 | Q_D(const QGeoShape); |
| 243 | |
| 244 | if (d) |
| 245 | return d->boundingGeoRectangle(); |
| 246 | else |
| 247 | return QGeoRectangle(); |
| 248 | } |
| 249 | |
| 250 | /*! |
| 251 | Returns the coordinate located at the geometric center of the geo shape. |
| 252 | |
| 253 | \since 5.5 |
| 254 | */ |
| 255 | QGeoCoordinate QGeoShape::center() const |
| 256 | { |
| 257 | Q_D(const QGeoShape); |
| 258 | |
| 259 | if (d) |
| 260 | return d->center(); |
| 261 | else |
| 262 | return QGeoCoordinate(); |
| 263 | } |
| 264 | |
| 265 | /*! |
| 266 | \deprecated |
| 267 | |
| 268 | This method used to extend the geo shape to also cover the coordinate \a coordinate. |
| 269 | |
| 270 | It currently only works for \l QGeoCircle and \l QGeoRectangle, on which the functionality remains, |
| 271 | now also accessible through QGeoCircle::extendCircle and QGeoRectangle::extendRectangle. |
| 272 | |
| 273 | This method should therefore not be called on a generic QGeoShape any longer, as the behavior for |
| 274 | other shape types is undefined. |
| 275 | |
| 276 | \sa QGeoRectangle::extendRectangle, QGeoCircle::extendCircle |
| 277 | */ |
| 278 | void QGeoShape::extendShape(const QGeoCoordinate &coordinate) |
| 279 | { |
| 280 | Q_D(QGeoShape); |
| 281 | |
| 282 | if (d) |
| 283 | d->extendShape(coordinate); |
| 284 | } |
| 285 | |
| 286 | |
| 287 | /*! |
| 288 | Returns true if the \a other geo shape is equivalent to this geo shape, otherwise returns |
| 289 | false. |
| 290 | */ |
| 291 | bool QGeoShape::operator==(const QGeoShape &other) const |
| 292 | { |
| 293 | Q_D(const QGeoShape); |
| 294 | |
| 295 | if (d == other.d_func()) |
| 296 | return true; |
| 297 | |
| 298 | if (!d || !(other.d_func())) |
| 299 | return false; |
| 300 | |
| 301 | return *d == *other.d_func(); |
| 302 | } |
| 303 | |
| 304 | /*! |
| 305 | Returns true if the \a other geo shape is not equivalent to this geo shape, otherwise returns |
| 306 | false. |
| 307 | */ |
| 308 | bool QGeoShape::operator!=(const QGeoShape &other) const |
| 309 | { |
| 310 | return !(*this == other); |
| 311 | } |
| 312 | |
| 313 | /*! |
| 314 | Assigns \a other to this geo shape and returns a reference to this geo shape. |
| 315 | */ |
| 316 | QGeoShape &QGeoShape::operator=(const QGeoShape &other) |
| 317 | { |
| 318 | if (this == &other) |
| 319 | return *this; |
| 320 | |
| 321 | d_ptr = other.d_ptr; |
| 322 | return *this; |
| 323 | } |
| 324 | |
| 325 | /*! |
| 326 | Returns a string representation of this geo shape. |
| 327 | |
| 328 | \since 5.5 |
| 329 | */ |
| 330 | QString QGeoShape::toString() const |
| 331 | { |
| 332 | return QStringLiteral("QGeoShape(%1)" ).arg(a: type()); |
| 333 | } |
| 334 | |
| 335 | #ifndef QT_NO_DEBUG_STREAM |
| 336 | QDebug operator<<(QDebug dbg, const QGeoShape &shape) |
| 337 | { |
| 338 | QDebugStateSaver saver(dbg); |
| 339 | dbg.nospace() << "QGeoShape(" ; |
| 340 | switch (shape.type()) { |
| 341 | case QGeoShape::UnknownType: |
| 342 | dbg << "Unknown" ; |
| 343 | break; |
| 344 | case QGeoShape::RectangleType: |
| 345 | dbg << "Rectangle" ; |
| 346 | break; |
| 347 | case QGeoShape::PathType: |
| 348 | dbg << "Path" ; |
| 349 | break; |
| 350 | case QGeoShape::PolygonType: |
| 351 | dbg << "Polygon" ; |
| 352 | break; |
| 353 | case QGeoShape::CircleType: |
| 354 | dbg << "Circle" ; |
| 355 | } |
| 356 | |
| 357 | dbg << ')'; |
| 358 | |
| 359 | return dbg; |
| 360 | } |
| 361 | #endif |
| 362 | |
| 363 | #ifndef QT_NO_DATASTREAM |
| 364 | QDataStream &operator<<(QDataStream &stream, const QGeoShape &shape) |
| 365 | { |
| 366 | stream << quint32(shape.type()); |
| 367 | switch (shape.type()) { |
| 368 | case QGeoShape::UnknownType: |
| 369 | break; |
| 370 | case QGeoShape::RectangleType: { |
| 371 | QGeoRectangle r = shape; |
| 372 | stream << r.topLeft() << r.bottomRight(); |
| 373 | break; |
| 374 | } |
| 375 | case QGeoShape::CircleType: { |
| 376 | QGeoCircle c = shape; |
| 377 | stream << c.center() << c.radius(); |
| 378 | break; |
| 379 | } |
| 380 | case QGeoShape::PathType: { |
| 381 | QGeoPath p = shape; |
| 382 | stream << p.path().size(); |
| 383 | for (const auto &c: p.path()) |
| 384 | stream << c; |
| 385 | break; |
| 386 | } |
| 387 | case QGeoShape::PolygonType: { |
| 388 | QGeoPolygon p = shape; |
| 389 | stream << p.path().size(); |
| 390 | for (const auto &c: p.path()) |
| 391 | stream << c; |
| 392 | break; |
| 393 | } |
| 394 | } |
| 395 | |
| 396 | return stream; |
| 397 | } |
| 398 | |
| 399 | QDataStream &operator>>(QDataStream &stream, QGeoShape &shape) |
| 400 | { |
| 401 | quint32 type; |
| 402 | stream >> type; |
| 403 | |
| 404 | switch (type) { |
| 405 | case QGeoShape::UnknownType: |
| 406 | shape = QGeoShape(); |
| 407 | break; |
| 408 | case QGeoShape::RectangleType: { |
| 409 | QGeoCoordinate tl; |
| 410 | QGeoCoordinate br; |
| 411 | stream >> tl >> br; |
| 412 | shape = QGeoRectangle(tl, br); |
| 413 | break; |
| 414 | } |
| 415 | case QGeoShape::CircleType: { |
| 416 | QGeoCoordinate c; |
| 417 | qreal r; |
| 418 | stream >> c >> r; |
| 419 | shape = QGeoCircle(c, r); |
| 420 | break; |
| 421 | } |
| 422 | case QGeoShape::PathType: { |
| 423 | QList<QGeoCoordinate> l; |
| 424 | QGeoCoordinate c; |
| 425 | int sz; |
| 426 | stream >> sz; |
| 427 | for (int i = 0; i < sz; i++) { |
| 428 | stream >> c; |
| 429 | l.append(t: c); |
| 430 | } |
| 431 | shape = QGeoPath(l); |
| 432 | break; |
| 433 | } |
| 434 | case QGeoShape::PolygonType: { |
| 435 | QList<QGeoCoordinate> l; |
| 436 | QGeoCoordinate c; |
| 437 | int sz; |
| 438 | stream >> sz; |
| 439 | for (int i = 0; i < sz; i++) { |
| 440 | stream >> c; |
| 441 | l.append(t: c); |
| 442 | } |
| 443 | shape = QGeoPolygon(l); |
| 444 | break; |
| 445 | } |
| 446 | } |
| 447 | |
| 448 | return stream; |
| 449 | } |
| 450 | #endif |
| 451 | |
| 452 | QT_END_NAMESPACE |
| 453 | |
| 454 | #include "moc_qgeoshape.cpp" |
| 455 | |