1 /* 2 * Copyright 2012 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8 #ifndef SkRRect_DEFINED 9 #define SkRRect_DEFINED 10 11 #include "include/core/SkPoint.h" 12 #include "include/core/SkRect.h" 13 14 class SkPath; 15 class SkMatrix; 16 class SkString; 17 18 /** \class SkRRect 19 SkRRect describes a rounded rectangle with a bounds and a pair of radii for each corner. 20 The bounds and radii can be set so that SkRRect describes: a rectangle with sharp corners; 21 a circle; an oval; or a rectangle with one or more rounded corners. 22 23 SkRRect allows implementing CSS properties that describe rounded corners. 24 SkRRect may have up to eight different radii, one for each axis on each of its four 25 corners. 26 27 SkRRect may modify the provided parameters when initializing bounds and radii. 28 If either axis radii is zero or less: radii are stored as zero; corner is square. 29 If corner curves overlap, radii are proportionally reduced to fit within bounds. 30 */ 31 class SK_API SkRRect { 32 public: 33 34 /** Initializes bounds at (0, 0), the origin, with zero width and height. 35 Initializes corner radii to (0, 0), and sets type of kEmpty_Type. 36 37 @return empty SkRRect 38 */ 39 SkRRect() = default; 40 41 /** Initializes to copy of rrect bounds and corner radii. 42 43 @param rrect bounds and corner to copy 44 @return copy of rrect 45 */ 46 SkRRect(const SkRRect& rrect) = default; 47 48 /** Copies rrect bounds and corner radii. 49 50 @param rrect bounds and corner to copy 51 @return copy of rrect 52 */ 53 SkRRect& operator=(const SkRRect& rrect) = default; 54 55 /** \enum SkRRect::Type 56 Type describes possible specializations of SkRRect. Each Type is 57 exclusive; a SkRRect may only have one type. 58 59 Type members become progressively less restrictive; larger values of 60 Type have more degrees of freedom than smaller values. 61 */ 62 enum Type { 63 kEmpty_Type, //!< zero width or height 64 kRect_Type, //!< non-zero width and height, and zeroed radii 65 kOval_Type, //!< non-zero width and height filled with radii 66 kSimple_Type, //!< non-zero width and height with equal radii 67 kNinePatch_Type, //!< non-zero width and height with axis-aligned radii 68 kComplex_Type, //!< non-zero width and height with arbitrary radii 69 kLastType = kComplex_Type, //!< largest Type value 70 }; 71 getType()72 Type getType() const { 73 SkASSERT(this->isValid()); 74 return static_cast<Type>(fType); 75 } 76 type()77 Type type() const { return this->getType(); } 78 isEmpty()79 inline bool isEmpty() const { return kEmpty_Type == this->getType(); } isRect()80 inline bool isRect() const { return kRect_Type == this->getType(); } isOval()81 inline bool isOval() const { return kOval_Type == this->getType(); } isSimple()82 inline bool isSimple() const { return kSimple_Type == this->getType(); } isNinePatch()83 inline bool isNinePatch() const { return kNinePatch_Type == this->getType(); } isComplex()84 inline bool isComplex() const { return kComplex_Type == this->getType(); } 85 86 /** Returns span on the x-axis. This does not check if result fits in 32-bit float; 87 result may be infinity. 88 89 @return rect().fRight minus rect().fLeft 90 */ width()91 SkScalar width() const { return fRect.width(); } 92 93 /** Returns span on the y-axis. This does not check if result fits in 32-bit float; 94 result may be infinity. 95 96 @return rect().fBottom minus rect().fTop 97 */ height()98 SkScalar height() const { return fRect.height(); } 99 100 /** Returns top-left corner radii. If type() returns kEmpty_Type, kRect_Type, 101 kOval_Type, or kSimple_Type, returns a value representative of all corner radii. 102 If type() returns kNinePatch_Type or kComplex_Type, at least one of the 103 remaining three corners has a different value. 104 105 @return corner radii for simple types 106 */ getSimpleRadii()107 SkVector getSimpleRadii() const { 108 return fRadii[0]; 109 } 110 111 /** Sets bounds to zero width and height at (0, 0), the origin. Sets 112 corner radii to zero and sets type to kEmpty_Type. 113 */ setEmpty()114 void setEmpty() { *this = SkRRect(); } 115 116 /** Sets bounds to sorted rect, and sets corner radii to zero. 117 If set bounds has width and height, and sets type to kRect_Type; 118 otherwise, sets type to kEmpty_Type. 119 120 @param rect bounds to set 121 */ setRect(const SkRect & rect)122 void setRect(const SkRect& rect) { 123 if (!this->initializeRect(rect)) { 124 return; 125 } 126 127 memset(fRadii, 0, sizeof(fRadii)); 128 fType = kRect_Type; 129 130 SkASSERT(this->isValid()); 131 } 132 133 /** Initializes bounds at (0, 0), the origin, with zero width and height. 134 Initializes corner radii to (0, 0), and sets type of kEmpty_Type. 135 136 @return empty SkRRect 137 */ MakeEmpty()138 static SkRRect MakeEmpty() { return SkRRect(); } 139 140 /** Initializes to copy of r bounds and zeroes corner radii. 141 142 @param r bounds to copy 143 @return copy of r 144 */ MakeRect(const SkRect & r)145 static SkRRect MakeRect(const SkRect& r) { 146 SkRRect rr; 147 rr.setRect(r); 148 return rr; 149 } 150 151 /** Sets bounds to oval, x-axis radii to half oval.width(), and all y-axis radii 152 to half oval.height(). If oval bounds is empty, sets to kEmpty_Type. 153 Otherwise, sets to kOval_Type. 154 155 @param oval bounds of oval 156 @return oval 157 */ MakeOval(const SkRect & oval)158 static SkRRect MakeOval(const SkRect& oval) { 159 SkRRect rr; 160 rr.setOval(oval); 161 return rr; 162 } 163 164 /** Sets to rounded rectangle with the same radii for all four corners. 165 If rect is empty, sets to kEmpty_Type. 166 Otherwise, if xRad and yRad are zero, sets to kRect_Type. 167 Otherwise, if xRad is at least half rect.width() and yRad is at least half 168 rect.height(), sets to kOval_Type. 169 Otherwise, sets to kSimple_Type. 170 171 @param rect bounds of rounded rectangle 172 @param xRad x-axis radius of corners 173 @param yRad y-axis radius of corners 174 @return rounded rectangle 175 */ MakeRectXY(const SkRect & rect,SkScalar xRad,SkScalar yRad)176 static SkRRect MakeRectXY(const SkRect& rect, SkScalar xRad, SkScalar yRad) { 177 SkRRect rr; 178 rr.setRectXY(rect, xRad, yRad); 179 return rr; 180 } 181 182 /** Sets bounds to oval, x-axis radii to half oval.width(), and all y-axis radii 183 to half oval.height(). If oval bounds is empty, sets to kEmpty_Type. 184 Otherwise, sets to kOval_Type. 185 186 @param oval bounds of oval 187 */ 188 void setOval(const SkRect& oval); 189 190 /** Sets to rounded rectangle with the same radii for all four corners. 191 If rect is empty, sets to kEmpty_Type. 192 Otherwise, if xRad or yRad is zero, sets to kRect_Type. 193 Otherwise, if xRad is at least half rect.width() and yRad is at least half 194 rect.height(), sets to kOval_Type. 195 Otherwise, sets to kSimple_Type. 196 197 @param rect bounds of rounded rectangle 198 @param xRad x-axis radius of corners 199 @param yRad y-axis radius of corners 200 201 example: https://fiddle.skia.org/c/@RRect_setRectXY 202 */ 203 void setRectXY(const SkRect& rect, SkScalar xRad, SkScalar yRad); 204 205 /** Sets bounds to rect. Sets radii to (leftRad, topRad), (rightRad, topRad), 206 (rightRad, bottomRad), (leftRad, bottomRad). 207 208 If rect is empty, sets to kEmpty_Type. 209 Otherwise, if leftRad and rightRad are zero, sets to kRect_Type. 210 Otherwise, if topRad and bottomRad are zero, sets to kRect_Type. 211 Otherwise, if leftRad and rightRad are equal and at least half rect.width(), and 212 topRad and bottomRad are equal at least half rect.height(), sets to kOval_Type. 213 Otherwise, if leftRad and rightRad are equal, and topRad and bottomRad are equal, 214 sets to kSimple_Type. Otherwise, sets to kNinePatch_Type. 215 216 Nine patch refers to the nine parts defined by the radii: one center rectangle, 217 four edge patches, and four corner patches. 218 219 @param rect bounds of rounded rectangle 220 @param leftRad left-top and left-bottom x-axis radius 221 @param topRad left-top and right-top y-axis radius 222 @param rightRad right-top and right-bottom x-axis radius 223 @param bottomRad left-bottom and right-bottom y-axis radius 224 */ 225 void setNinePatch(const SkRect& rect, SkScalar leftRad, SkScalar topRad, 226 SkScalar rightRad, SkScalar bottomRad); 227 228 /** Sets bounds to rect. Sets radii array for individual control of all for corners. 229 230 If rect is empty, sets to kEmpty_Type. 231 Otherwise, if one of each corner radii are zero, sets to kRect_Type. 232 Otherwise, if all x-axis radii are equal and at least half rect.width(), and 233 all y-axis radii are equal at least half rect.height(), sets to kOval_Type. 234 Otherwise, if all x-axis radii are equal, and all y-axis radii are equal, 235 sets to kSimple_Type. Otherwise, sets to kNinePatch_Type. 236 237 @param rect bounds of rounded rectangle 238 @param radii corner x-axis and y-axis radii 239 240 example: https://fiddle.skia.org/c/@RRect_setRectRadii 241 */ 242 void setRectRadii(const SkRect& rect, const SkVector radii[4]); 243 244 /** \enum SkRRect::Corner 245 The radii are stored: top-left, top-right, bottom-right, bottom-left. 246 */ 247 enum Corner { 248 kUpperLeft_Corner, //!< index of top-left corner radii 249 kUpperRight_Corner, //!< index of top-right corner radii 250 kLowerRight_Corner, //!< index of bottom-right corner radii 251 kLowerLeft_Corner, //!< index of bottom-left corner radii 252 }; 253 254 /** Returns bounds. Bounds may have zero width or zero height. Bounds right is 255 greater than or equal to left; bounds bottom is greater than or equal to top. 256 Result is identical to getBounds(). 257 258 @return bounding box 259 */ rect()260 const SkRect& rect() const { return fRect; } 261 262 /** Returns scalar pair for radius of curve on x-axis and y-axis for one corner. 263 Both radii may be zero. If not zero, both are positive and finite. 264 265 @return x-axis and y-axis radii for one corner 266 */ radii(Corner corner)267 SkVector radii(Corner corner) const { return fRadii[corner]; } 268 269 /** Returns bounds. Bounds may have zero width or zero height. Bounds right is 270 greater than or equal to left; bounds bottom is greater than or equal to top. 271 Result is identical to rect(). 272 273 @return bounding box 274 */ getBounds()275 const SkRect& getBounds() const { return fRect; } 276 277 /** Returns true if bounds and radii in a are equal to bounds and radii in b. 278 279 a and b are not equal if either contain NaN. a and b are equal if members 280 contain zeroes with different signs. 281 282 @param a SkRect bounds and radii to compare 283 @param b SkRect bounds and radii to compare 284 @return true if members are equal 285 */ 286 friend bool operator==(const SkRRect& a, const SkRRect& b) { 287 return a.fRect == b.fRect && SkScalarsEqual(&a.fRadii[0].fX, &b.fRadii[0].fX, 8); 288 } 289 290 /** Returns true if bounds and radii in a are not equal to bounds and radii in b. 291 292 a and b are not equal if either contain NaN. a and b are equal if members 293 contain zeroes with different signs. 294 295 @param a SkRect bounds and radii to compare 296 @param b SkRect bounds and radii to compare 297 @return true if members are not equal 298 */ 299 friend bool operator!=(const SkRRect& a, const SkRRect& b) { 300 return a.fRect != b.fRect || !SkScalarsEqual(&a.fRadii[0].fX, &b.fRadii[0].fX, 8); 301 } 302 303 /** Copies SkRRect to dst, then insets dst bounds by dx and dy, and adjusts dst 304 radii by dx and dy. dx and dy may be positive, negative, or zero. dst may be 305 SkRRect. 306 307 If either corner radius is zero, the corner has no curvature and is unchanged. 308 Otherwise, if adjusted radius becomes negative, pins radius to zero. 309 If dx exceeds half dst bounds width, dst bounds left and right are set to 310 bounds x-axis center. If dy exceeds half dst bounds height, dst bounds top and 311 bottom are set to bounds y-axis center. 312 313 If dx or dy cause the bounds to become infinite, dst bounds is zeroed. 314 315 @param dx added to rect().fLeft, and subtracted from rect().fRight 316 @param dy added to rect().fTop, and subtracted from rect().fBottom 317 @param dst insets bounds and radii 318 319 example: https://fiddle.skia.org/c/@RRect_inset 320 */ 321 void inset(SkScalar dx, SkScalar dy, SkRRect* dst) const; 322 323 /** Insets bounds by dx and dy, and adjusts radii by dx and dy. dx and dy may be 324 positive, negative, or zero. 325 326 If either corner radius is zero, the corner has no curvature and is unchanged. 327 Otherwise, if adjusted radius becomes negative, pins radius to zero. 328 If dx exceeds half bounds width, bounds left and right are set to 329 bounds x-axis center. If dy exceeds half bounds height, bounds top and 330 bottom are set to bounds y-axis center. 331 332 If dx or dy cause the bounds to become infinite, bounds is zeroed. 333 334 @param dx added to rect().fLeft, and subtracted from rect().fRight 335 @param dy added to rect().fTop, and subtracted from rect().fBottom 336 */ inset(SkScalar dx,SkScalar dy)337 void inset(SkScalar dx, SkScalar dy) { 338 this->inset(dx, dy, this); 339 } 340 341 /** Outsets dst bounds by dx and dy, and adjusts radii by dx and dy. dx and dy may be 342 positive, negative, or zero. 343 344 If either corner radius is zero, the corner has no curvature and is unchanged. 345 Otherwise, if adjusted radius becomes negative, pins radius to zero. 346 If dx exceeds half dst bounds width, dst bounds left and right are set to 347 bounds x-axis center. If dy exceeds half dst bounds height, dst bounds top and 348 bottom are set to bounds y-axis center. 349 350 If dx or dy cause the bounds to become infinite, dst bounds is zeroed. 351 352 @param dx subtracted from rect().fLeft, and added to rect().fRight 353 @param dy subtracted from rect().fTop, and added to rect().fBottom 354 @param dst outset bounds and radii 355 */ outset(SkScalar dx,SkScalar dy,SkRRect * dst)356 void outset(SkScalar dx, SkScalar dy, SkRRect* dst) const { 357 this->inset(-dx, -dy, dst); 358 } 359 360 /** Outsets bounds by dx and dy, and adjusts radii by dx and dy. dx and dy may be 361 positive, negative, or zero. 362 363 If either corner radius is zero, the corner has no curvature and is unchanged. 364 Otherwise, if adjusted radius becomes negative, pins radius to zero. 365 If dx exceeds half bounds width, bounds left and right are set to 366 bounds x-axis center. If dy exceeds half bounds height, bounds top and 367 bottom are set to bounds y-axis center. 368 369 If dx or dy cause the bounds to become infinite, bounds is zeroed. 370 371 @param dx subtracted from rect().fLeft, and added to rect().fRight 372 @param dy subtracted from rect().fTop, and added to rect().fBottom 373 */ outset(SkScalar dx,SkScalar dy)374 void outset(SkScalar dx, SkScalar dy) { 375 this->inset(-dx, -dy, this); 376 } 377 378 /** Translates SkRRect by (dx, dy). 379 380 @param dx offset added to rect().fLeft and rect().fRight 381 @param dy offset added to rect().fTop and rect().fBottom 382 */ offset(SkScalar dx,SkScalar dy)383 void offset(SkScalar dx, SkScalar dy) { 384 fRect.offset(dx, dy); 385 } 386 387 /** Returns SkRRect translated by (dx, dy). 388 389 @param dx offset added to rect().fLeft and rect().fRight 390 @param dy offset added to rect().fTop and rect().fBottom 391 @return SkRRect bounds offset by (dx, dy), with unchanged corner radii 392 */ makeOffset(SkScalar dx,SkScalar dy)393 SkRRect SK_WARN_UNUSED_RESULT makeOffset(SkScalar dx, SkScalar dy) const { 394 return SkRRect(fRect.makeOffset(dx, dy), fRadii, fType); 395 } 396 397 /** Returns true if rect is inside the bounds and corner radii, and if 398 SkRRect and rect are not empty. 399 400 @param rect area tested for containment 401 @return true if SkRRect contains rect 402 403 example: https://fiddle.skia.org/c/@RRect_contains 404 */ 405 bool contains(const SkRect& rect) const; 406 407 /** Returns true if bounds and radii values are finite and describe a SkRRect 408 SkRRect::Type that matches getType(). All SkRRect methods construct valid types, 409 even if the input values are not valid. Invalid SkRRect data can only 410 be generated by corrupting memory. 411 412 @return true if bounds and radii match type() 413 414 example: https://fiddle.skia.org/c/@RRect_isValid 415 */ 416 bool isValid() const; 417 418 static constexpr size_t kSizeInMemory = 12 * sizeof(SkScalar); 419 420 /** Writes SkRRect to buffer. Writes kSizeInMemory bytes, and returns 421 kSizeInMemory, the number of bytes written. 422 423 @param buffer storage for SkRRect 424 @return bytes written, kSizeInMemory 425 426 example: https://fiddle.skia.org/c/@RRect_writeToMemory 427 */ 428 size_t writeToMemory(void* buffer) const; 429 430 /** Reads SkRRect from buffer, reading kSizeInMemory bytes. 431 Returns kSizeInMemory, bytes read if length is at least kSizeInMemory. 432 Otherwise, returns zero. 433 434 @param buffer memory to read from 435 @param length size of buffer 436 @return bytes read, or 0 if length is less than kSizeInMemory 437 438 example: https://fiddle.skia.org/c/@RRect_readFromMemory 439 */ 440 size_t readFromMemory(const void* buffer, size_t length); 441 442 /** Transforms by SkRRect by matrix, storing result in dst. 443 Returns true if SkRRect transformed can be represented by another SkRRect. 444 Returns false if matrix contains transformations that are not axis aligned. 445 446 Asserts in debug builds if SkRRect equals dst. 447 448 @param matrix SkMatrix specifying the transform 449 @param dst SkRRect to store the result 450 @return true if transformation succeeded. 451 452 example: https://fiddle.skia.org/c/@RRect_transform 453 */ 454 bool transform(const SkMatrix& matrix, SkRRect* dst) const; 455 456 /** Writes text representation of SkRRect to standard output. 457 Set asHex true to generate exact binary representations 458 of floating point numbers. 459 460 @param asHex true if SkScalar values are written as hexadecimal 461 462 example: https://fiddle.skia.org/c/@RRect_dump 463 */ 464 void dump(bool asHex) const; 465 SkString dumpToString(bool asHex) const; 466 467 /** Writes text representation of SkRRect to standard output. The representation 468 may be directly compiled as C++ code. Floating point values are written 469 with limited precision; it may not be possible to reconstruct original 470 SkRRect from output. 471 */ dump()472 void dump() const { this->dump(false); } 473 474 /** Writes text representation of SkRRect to standard output. The representation 475 may be directly compiled as C++ code. Floating point values are written 476 in hexadecimal to preserve their exact bit pattern. The output reconstructs the 477 original SkRRect. 478 */ dumpHex()479 void dumpHex() const { this->dump(true); } 480 481 private: 482 static bool AreRectAndRadiiValid(const SkRect&, const SkVector[4]); 483 SkRRect(const SkRect & rect,const SkVector radii[4],int32_t type)484 SkRRect(const SkRect& rect, const SkVector radii[4], int32_t type) 485 : fRect(rect) 486 , fRadii{radii[0], radii[1], radii[2], radii[3]} 487 , fType(type) {} 488 489 /** 490 * Initializes fRect. If the passed in rect is not finite or empty the rrect will be fully 491 * initialized and false is returned. Otherwise, just fRect is initialized and true is returned. 492 */ 493 bool initializeRect(const SkRect&); 494 495 void computeType(); 496 bool checkCornerContainment(SkScalar x, SkScalar y) const; 497 // Returns true if the radii had to be scaled to fit rect 498 bool scaleRadii(); 499 500 SkRect fRect = SkRect::MakeEmpty(); 501 // Radii order is UL, UR, LR, LL. Use Corner enum to index into fRadii[] 502 SkVector fRadii[4] = {{0, 0}, {0, 0}, {0,0}, {0,0}}; 503 // use an explicitly sized type so we're sure the class is dense (no uninitialized bytes) 504 int32_t fType = kEmpty_Type; 505 // TODO: add padding so we can use memcpy for flattening and not copy uninitialized data 506 507 // to access fRadii directly 508 friend class SkPath; 509 friend class SkRRectPriv; 510 }; 511 512 #endif 513