Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(439)

Side by Side Diff: src/core/SkPath.cpp

Issue 60203002: use quads for mixed radius rrects (Closed) Base URL: https://skia.googlecode.com/svn/trunk
Patch Set: Created 7 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « no previous file | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 1
2 /* 2 /*
3 * Copyright 2006 The Android Open Source Project 3 * Copyright 2006 The Android Open Source Project
4 * 4 *
5 * Use of this source code is governed by a BSD-style license that can be 5 * Use of this source code is governed by a BSD-style license that can be
6 * found in the LICENSE file. 6 * found in the LICENSE file.
7 */ 7 */
8 8
9 9
10 #include "SkBuffer.h" 10 #include "SkBuffer.h"
(...skipping 932 matching lines...) Expand 10 before | Expand all | Expand 10 after
943 943
944 matrix.setScale(SkScalarHalf(oval.width()), SkScalarHalf(oval.height())); 944 matrix.setScale(SkScalarHalf(oval.width()), SkScalarHalf(oval.height()));
945 matrix.postTranslate(oval.centerX(), oval.centerY()); 945 matrix.postTranslate(oval.centerX(), oval.centerY());
946 946
947 return SkBuildQuadArc(start, stop, 947 return SkBuildQuadArc(start, stop,
948 sweepAngle > 0 ? kCW_SkRotationDirection : 948 sweepAngle > 0 ? kCW_SkRotationDirection :
949 kCCW_SkRotationDirection, 949 kCCW_SkRotationDirection,
950 &matrix, pts); 950 &matrix, pts);
951 } 951 }
952 952
953 static void add_corner_arc(SkPath* path, const SkRect& rect,
954 SkScalar rx, SkScalar ry, int startAngle,
955 SkPath::Direction dir, bool forceMoveTo) {
956 // These two asserts are not sufficient, since really we want to know
957 // that the pair of radii (e.g. left and right, or top and bottom) sum
958 // to <= dimension, but we don't have that data here, so we just have
959 // these conservative asserts.
960 SkASSERT(0 <= rx && rx <= rect.width());
961 SkASSERT(0 <= ry && ry <= rect.height());
962
963 SkRect r;
964 r.set(-rx, -ry, rx, ry);
965
966 switch (startAngle) {
967 case 0:
968 r.offset(rect.fRight - r.fRight, rect.fBottom - r.fBottom);
969 break;
970 case 90:
971 r.offset(rect.fLeft - r.fLeft, rect.fBottom - r.fBottom);
972 break;
973 case 180: r.offset(rect.fLeft - r.fLeft, rect.fTop - r.fTop); break;
974 case 270: r.offset(rect.fRight - r.fRight, rect.fTop - r.fTop); break;
975 default: SkDEBUGFAIL("unexpected startAngle in add_corner_arc");
976 }
977
978 SkScalar start = SkIntToScalar(startAngle);
979 SkScalar sweep = SkIntToScalar(90);
980 if (SkPath::kCCW_Direction == dir) {
981 start += sweep;
982 sweep = -sweep;
983 }
984
985 path->arcTo(r, start, sweep, forceMoveTo);
986 }
987
988 void SkPath::addRoundRect(const SkRect& rect, const SkScalar radii[], 953 void SkPath::addRoundRect(const SkRect& rect, const SkScalar radii[],
989 Direction dir) { 954 Direction dir) {
990 SkRRect rrect; 955 SkRRect rrect;
991 rrect.setRectRadii(rect, (const SkVector*) radii); 956 rrect.setRectRadii(rect, (const SkVector*) radii);
992 this->addRRect(rrect, dir); 957 this->addRRect(rrect, dir);
993 } 958 }
994 959
960 /* The inline clockwise and counterclockwise round rect quad approximations
961 make it easier to see the symmetry patterns used by add corner quads.
962 Clockwise corner value
963 path->lineTo(rect.fLeft, rect.fTop + ry); 0 upper left
964 path->quadTo(rect.fLeft, rect.fTop + offPtY,
965 rect.fLeft + midPtX, rect.fTop + midPtY);
966 path->quadTo(rect.fLeft + offPtX, rect.fTop,
967 rect.fLeft + rx, rect.fTop);
968
969 path->lineTo(rect.fRight - rx, rect.fTop); 1 upper right
970 path->quadTo(rect.fRight - offPtX, rect.fTop,
971 rect.fRight - midPtX, rect.fTop + midPtY);
972 path->quadTo(rect.fRight, rect.fTop + offPtY,
973 rect.fRight, rect.fTop + ry);
974
975 path->lineTo(rect.fRight, rect.fBottom - ry); 2 lower right
976 path->quadTo(rect.fRight, rect.fBottom - offPtY,
977 rect.fRight - midPtX, rect.fBottom - midPtY);
978 path->quadTo(rect.fRight - offPtX, rect.fBottom,
979 rect.fRight - rx, rect.fBottom);
980
981 path->lineTo(rect.fLeft + rx, rect.fBottom); 3 lower left
982 path->quadTo(rect.fLeft + offPtX, rect.fBottom,
983 rect.fLeft + midPtX, rect.fBottom - midPtY);
984 path->quadTo(rect.fLeft, rect.fBottom - offPtY,
985 rect.fLeft, rect.fBottom - ry);
986
987 Counterclockwise
988 path->lineTo(rect.fLeft, rect.fBottom - ry); 3 lower left
989 path->quadTo(rect.fLeft, rect.fBottom - offPtY,
990 rect.fLeft + midPtX, rect.fBottom - midPtY);
991 path->quadTo(rect.fLeft + offPtX, rect.fBottom,
992 rect.fLeft + rx, rect.fBottom);
993
994 path->lineTo(rect.fRight - rx, rect.fBottom); 2 lower right
995 path->quadTo(rect.fRight - offPtX, rect.fBottom,
996 rect.fRight - midPtX, rect.fBottom - midPtY);
997 path->quadTo(rect.fRight, rect.fBottom - offPtY,
998 rect.fRight, rect.fBottom - ry);
999
1000 path->lineTo(rect.fRight, rect.fTop + ry); 1 upper right
1001 path->quadTo(rect.fRight, rect.fTop + offPtY,
1002 rect.fRight - midPtX, rect.fTop + midPtY);
1003 path->quadTo(rect.fRight - offPtX, rect.fTop,
1004 rect.fRight - rx, rect.fTop);
1005
1006 path->lineTo(rect.fLeft + rx, rect.fTop); 0 upper left
1007 path->quadTo(rect.fLeft + offPtX, rect.fTop,
1008 rect.fLeft + midPtX, rect.fTop + midPtY);
1009 path->quadTo(rect.fLeft, rect.fTop + offPtY,
1010 rect.fLeft, rect.fTop + ry);
1011 */
robertphillips 2013/11/05 16:47:56 Would it make sense to pass in the SkRRect itself
caryclark 2013/11/05 20:29:55 Done.
1012 static void add_corner_quads(SkPath* path, const SkRect& rect, const SkVector& r adii,
1013 SkRRect::Corner corner, SkPath::Direction dir) {
1014 SkScalar rx = radii.fX;
1015 SkScalar ry = radii.fY;
1016 // The mid point of the quadratic arc approximation is half way between the two
1017 // control points. The float epsilon adjustment moves the on curve point out by
1018 // two bits, distributing the convex test error between the round rect appro ximation
1019 // and the convex cross product sign equality test.
1020 SkScalar midPtX = rx - rx * (SK_Scalar1 + SK_ScalarTanPIOver8 + FLT_EPSILON * 4) / 2;
1021 SkScalar midPtY = ry - ry * (SK_Scalar1 + SK_ScalarTanPIOver8 + FLT_EPSILON * 4) / 2;
1022 SkScalar offPtX = rx - rx * SK_ScalarTanPIOver8;
1023 SkScalar offPtY = ry - ry * SK_ScalarTanPIOver8;
1024 static const int kCornerPts = 5;
1025 SkScalar xOff[kCornerPts];
1026 SkScalar yOff[kCornerPts];
1027
1028 if ((corner & 1) == (dir == SkPath::kCCW_Direction)) { // corners always al ternate direction
1029 SkASSERT(dir == SkPath::kCCW_Direction
1030 ? corner == SkRRect::kLowerLeft_Corner || corner == SkRRect::kUpper Right_Corner
1031 : corner == SkRRect::kUpperLeft_Corner || corner == SkRRect::kLower Right_Corner);
1032 xOff[0] = xOff[1] = 0;
1033 xOff[2] = midPtX;
1034 xOff[3] = offPtX;
1035 xOff[4] = rx;
1036 yOff[0] = ry;
1037 yOff[1] = offPtY;
1038 yOff[2] = midPtY;
1039 yOff[3] = yOff[4] = 0;
1040 } else {
1041 xOff[0] = rx;
1042 xOff[1] = offPtX;
1043 xOff[2] = midPtX;
1044 xOff[3] = xOff[4] = 0;
1045 yOff[0] = yOff[1] = 0;
1046 yOff[2] = midPtY;
1047 yOff[3] = offPtY;
1048 yOff[4] = ry;
1049 }
1050 if ((corner - 1) & 2) {
1051 SkASSERT(corner == SkRRect::kLowerLeft_Corner || corner == SkRRect::kUpp erLeft_Corner);
1052 for (int i = 0; i < kCornerPts; ++i) {
1053 xOff[i] = rect.fLeft + xOff[i]; // cw lower right, lower left
1054 }
1055 } else {
1056 SkASSERT(corner == SkRRect::kLowerRight_Corner || corner == SkRRect::kUp perRight_Corner);
1057 for (int i = 0; i < kCornerPts; ++i) {
1058 xOff[i] = rect.fRight - xOff[i];
1059 }
1060 }
1061 if (corner < SkRRect::kLowerRight_Corner) {
1062 for (int i = 0; i < kCornerPts; ++i) {
1063 yOff[i] = rect.fTop + yOff[i];
1064 }
1065 } else {
1066 for (int i = 0; i < kCornerPts; ++i) {
1067 yOff[i] = rect.fBottom - yOff[i];
1068 }
1069 }
1070
1071 SkPoint lastPt;
1072 SkAssertResult(path->getLastPt(&lastPt));
1073 if (lastPt.fX != xOff[0] || lastPt.fY != yOff[0]) {
1074 path->lineTo(xOff[0], yOff[0]);
1075 }
1076 if (rx || ry) {
1077 path->quadTo(xOff[1], yOff[1], xOff[2], yOff[2]);
1078 path->quadTo(xOff[3], yOff[3], xOff[4], yOff[4]);
1079 } else {
1080 path->lineTo(xOff[2], yOff[2]);
1081 path->lineTo(xOff[4], yOff[4]);
1082 }
1083 }
1084
995 void SkPath::addRRect(const SkRRect& rrect, Direction dir) { 1085 void SkPath::addRRect(const SkRRect& rrect, Direction dir) {
996 assert_known_direction(dir); 1086 assert_known_direction(dir);
997 1087
998 if (rrect.isEmpty()) { 1088 if (rrect.isEmpty()) {
999 return; 1089 return;
1000 } 1090 }
1001 1091
1002 const SkRect& bounds = rrect.getBounds(); 1092 const SkRect& bounds = rrect.getBounds();
1003 1093
1004 if (rrect.isRect()) { 1094 if (rrect.isRect()) {
1005 this->addRect(bounds, dir); 1095 this->addRect(bounds, dir);
1006 } else if (rrect.isOval()) { 1096 } else if (rrect.isOval()) {
1007 this->addOval(bounds, dir); 1097 this->addOval(bounds, dir);
1008 } else if (rrect.isSimple()) { 1098 } else if (rrect.isSimple()) {
1009 const SkVector& rad = rrect.getSimpleRadii(); 1099 const SkVector& rad = rrect.getSimpleRadii();
1010 this->addRoundRect(bounds, rad.x(), rad.y(), dir); 1100 this->addRoundRect(bounds, rad.x(), rad.y(), dir);
1011 } else { 1101 } else {
1102 fDirection = this->hasOnlyMoveTos() ? dir : kUnknown_Direction;
1103
1012 SkAutoPathBoundsUpdate apbu(this, bounds); 1104 SkAutoPathBoundsUpdate apbu(this, bounds);
1013 1105
1106 this->incReserve(21);
1014 if (kCW_Direction == dir) { 1107 if (kCW_Direction == dir) {
robertphillips 2013/11/05 16:47:56 Should 0 be 3 here? One of the Corner enum values
caryclark 2013/11/05 20:29:55 Done.
1015 add_corner_arc(this, bounds, rrect.fRadii[0].fX, rrect.fRadii[0].fY, 180, dir, true); 1108 this->moveTo(bounds.fLeft, bounds.fBottom - rrect.fRadii[0].fY); // bottom-left
1016 add_corner_arc(this, bounds, rrect.fRadii[1].fX, rrect.fRadii[1].fY, 270, dir, false); 1109 add_corner_quads(this, bounds, rrect.fRadii[0], SkRRect::kUpperLeft_ Corner, dir);
1017 add_corner_arc(this, bounds, rrect.fRadii[2].fX, rrect.fRadii[2].fY, 0, dir, false); 1110 add_corner_quads(this, bounds, rrect.fRadii[1], SkRRect::kUpperRight _Corner, dir);
1018 add_corner_arc(this, bounds, rrect.fRadii[3].fX, rrect.fRadii[3].fY, 90, dir, false); 1111 add_corner_quads(this, bounds, rrect.fRadii[2], SkRRect::kLowerRight _Corner, dir);
1112 add_corner_quads(this, bounds, rrect.fRadii[3], SkRRect::kLowerLeft_ Corner, dir);
1019 } else { 1113 } else {
1020 add_corner_arc(this, bounds, rrect.fRadii[0].fX, rrect.fRadii[0].fY, 180, dir, true); 1114 this->moveTo(bounds.fLeft, bounds.fTop + rrect.fRadii[0].fY); // to p-left
1021 add_corner_arc(this, bounds, rrect.fRadii[3].fX, rrect.fRadii[3].fY, 90, dir, false); 1115 add_corner_quads(this, bounds, rrect.fRadii[3], SkRRect::kLowerLeft_ Corner, dir);
1022 add_corner_arc(this, bounds, rrect.fRadii[2].fX, rrect.fRadii[2].fY, 0, dir, false); 1116 add_corner_quads(this, bounds, rrect.fRadii[2], SkRRect::kLowerRight _Corner, dir);
1023 add_corner_arc(this, bounds, rrect.fRadii[1].fX, rrect.fRadii[1].fY, 270, dir, false); 1117 add_corner_quads(this, bounds, rrect.fRadii[1], SkRRect::kUpperRight _Corner, dir);
1118 add_corner_quads(this, bounds, rrect.fRadii[0], SkRRect::kUpperLeft_ Corner, dir);
1024 } 1119 }
1025 this->close(); 1120 this->close();
1026 } 1121 }
1027 } 1122 }
1028 1123
1029 bool SkPath::hasOnlyMoveTos() const { 1124 bool SkPath::hasOnlyMoveTos() const {
1030 int count = fPathRef->countVerbs(); 1125 int count = fPathRef->countVerbs();
1031 const uint8_t* verbs = const_cast<const SkPathRef*>(fPathRef.get())->verbsMe mBegin(); 1126 const uint8_t* verbs = const_cast<const SkPathRef*>(fPathRef.get())->verbsMe mBegin();
1032 for (int i = 0; i < count; ++i) { 1127 for (int i = 0; i < count; ++i) {
1033 if (*verbs == kLine_Verb || 1128 if (*verbs == kLine_Verb ||
(...skipping 1968 matching lines...) Expand 10 before | Expand all | Expand 10 after
3002 switch (this->getFillType()) { 3097 switch (this->getFillType()) {
3003 case SkPath::kEvenOdd_FillType: 3098 case SkPath::kEvenOdd_FillType:
3004 case SkPath::kInverseEvenOdd_FillType: 3099 case SkPath::kInverseEvenOdd_FillType:
3005 w &= 1; 3100 w &= 1;
3006 break; 3101 break;
3007 default: 3102 default:
3008 break; 3103 break;
3009 } 3104 }
3010 return SkToBool(w); 3105 return SkToBool(w);
3011 } 3106 }
OLDNEW
« no previous file with comments | « no previous file | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698