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

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: add disable direction check 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 */
1012 static void add_corner_quads(SkPath* path, const SkRRect& rrect,
1013 SkRRect::Corner corner, SkPath::Direction dir) {
1014 const SkRect& rect = rrect.rect();
1015 const SkVector& radii = rrect.radii(corner);
1016 SkScalar rx = radii.fX;
1017 SkScalar ry = radii.fY;
1018 // The mid point of the quadratic arc approximation is half way between the two
1019 // control points. The float epsilon adjustment moves the on curve point out by
1020 // two bits, distributing the convex test error between the round rect appro ximation
1021 // and the convex cross product sign equality test.
reed1 2013/11/08 14:04:32 In the comment, can we point to a unittest that wo
caryclark 2013/11/08 15:22:30 Mike's suspicion paid off. The 2 bit bump is only
1022 SkScalar midPtX = rx - rx * (SK_Scalar1 + SK_ScalarTanPIOver8 + FLT_EPSILON * 4) / 2;
1023 SkScalar midPtY = ry - ry * (SK_Scalar1 + SK_ScalarTanPIOver8 + FLT_EPSILON * 4) / 2;
1024 SkScalar offPtX = rx - rx * SK_ScalarTanPIOver8;
1025 SkScalar offPtY = ry - ry * SK_ScalarTanPIOver8;
1026 static const int kCornerPts = 5;
1027 SkScalar xOff[kCornerPts];
1028 SkScalar yOff[kCornerPts];
1029
1030 if ((corner & 1) == (dir == SkPath::kCCW_Direction)) { // corners always al ternate direction
1031 SkASSERT(dir == SkPath::kCCW_Direction
1032 ? corner == SkRRect::kLowerLeft_Corner || corner == SkRRect::kUpper Right_Corner
1033 : corner == SkRRect::kUpperLeft_Corner || corner == SkRRect::kLower Right_Corner);
1034 xOff[0] = xOff[1] = 0;
1035 xOff[2] = midPtX;
1036 xOff[3] = offPtX;
1037 xOff[4] = rx;
1038 yOff[0] = ry;
1039 yOff[1] = offPtY;
1040 yOff[2] = midPtY;
1041 yOff[3] = yOff[4] = 0;
1042 } else {
1043 xOff[0] = rx;
1044 xOff[1] = offPtX;
1045 xOff[2] = midPtX;
1046 xOff[3] = xOff[4] = 0;
1047 yOff[0] = yOff[1] = 0;
1048 yOff[2] = midPtY;
1049 yOff[3] = offPtY;
1050 yOff[4] = ry;
1051 }
1052 if ((corner - 1) & 2) {
1053 SkASSERT(corner == SkRRect::kLowerLeft_Corner || corner == SkRRect::kUpp erLeft_Corner);
1054 for (int i = 0; i < kCornerPts; ++i) {
1055 xOff[i] = rect.fLeft + xOff[i];
1056 }
1057 } else {
1058 SkASSERT(corner == SkRRect::kLowerRight_Corner || corner == SkRRect::kUp perRight_Corner);
1059 for (int i = 0; i < kCornerPts; ++i) {
1060 xOff[i] = rect.fRight - xOff[i];
1061 }
1062 }
1063 if (corner < SkRRect::kLowerRight_Corner) {
1064 for (int i = 0; i < kCornerPts; ++i) {
1065 yOff[i] = rect.fTop + yOff[i];
1066 }
1067 } else {
1068 for (int i = 0; i < kCornerPts; ++i) {
1069 yOff[i] = rect.fBottom - yOff[i];
1070 }
1071 }
1072
1073 SkPoint lastPt;
1074 SkAssertResult(path->getLastPt(&lastPt));
1075 if (lastPt.fX != xOff[0] || lastPt.fY != yOff[0]) {
1076 path->lineTo(xOff[0], yOff[0]);
1077 }
1078 if (rx || ry) {
1079 path->quadTo(xOff[1], yOff[1], xOff[2], yOff[2]);
1080 path->quadTo(xOff[3], yOff[3], xOff[4], yOff[4]);
1081 } else {
1082 path->lineTo(xOff[2], yOff[2]);
1083 path->lineTo(xOff[4], yOff[4]);
1084 }
1085 }
1086
995 void SkPath::addRRect(const SkRRect& rrect, Direction dir) { 1087 void SkPath::addRRect(const SkRRect& rrect, Direction dir) {
996 assert_known_direction(dir); 1088 assert_known_direction(dir);
997 1089
998 if (rrect.isEmpty()) { 1090 if (rrect.isEmpty()) {
999 return; 1091 return;
1000 } 1092 }
1001 1093
1002 const SkRect& bounds = rrect.getBounds(); 1094 const SkRect& bounds = rrect.getBounds();
1003 1095
1004 if (rrect.isRect()) { 1096 if (rrect.isRect()) {
1005 this->addRect(bounds, dir); 1097 this->addRect(bounds, dir);
1006 } else if (rrect.isOval()) { 1098 } else if (rrect.isOval()) {
1007 this->addOval(bounds, dir); 1099 this->addOval(bounds, dir);
1100 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1008 } else if (rrect.isSimple()) { 1101 } else if (rrect.isSimple()) {
1009 const SkVector& rad = rrect.getSimpleRadii(); 1102 const SkVector& rad = rrect.getSimpleRadii();
1010 this->addRoundRect(bounds, rad.x(), rad.y(), dir); 1103 this->addRoundRect(bounds, rad.x(), rad.y(), dir);
1104 #endif
1011 } else { 1105 } else {
1106 fDirection = this->hasOnlyMoveTos() ? dir : kUnknown_Direction;
1107
1012 SkAutoPathBoundsUpdate apbu(this, bounds); 1108 SkAutoPathBoundsUpdate apbu(this, bounds);
1109 SkAutoDisableDirectionCheck(this);
1013 1110
1111 this->incReserve(21);
1014 if (kCW_Direction == dir) { 1112 if (kCW_Direction == dir) {
1015 add_corner_arc(this, bounds, rrect.fRadii[0].fX, rrect.fRadii[0].fY, 180, dir, true); 1113 this->moveTo(bounds.fLeft,
1016 add_corner_arc(this, bounds, rrect.fRadii[1].fX, rrect.fRadii[1].fY, 270, dir, false); 1114 bounds.fBottom - rrect.fRadii[SkRRect::kLowerLeft_Corne r].fY);
1017 add_corner_arc(this, bounds, rrect.fRadii[2].fX, rrect.fRadii[2].fY, 0, dir, false); 1115 add_corner_quads(this, rrect, SkRRect::kUpperLeft_Corner, dir);
1018 add_corner_arc(this, bounds, rrect.fRadii[3].fX, rrect.fRadii[3].fY, 90, dir, false); 1116 add_corner_quads(this, rrect, SkRRect::kUpperRight_Corner, dir);
1117 add_corner_quads(this, rrect, SkRRect::kLowerRight_Corner, dir);
1118 add_corner_quads(this, rrect, SkRRect::kLowerLeft_Corner, dir);
1019 } else { 1119 } else {
1020 add_corner_arc(this, bounds, rrect.fRadii[0].fX, rrect.fRadii[0].fY, 180, dir, true); 1120 this->moveTo(bounds.fLeft,
1021 add_corner_arc(this, bounds, rrect.fRadii[3].fX, rrect.fRadii[3].fY, 90, dir, false); 1121 bounds.fTop + rrect.fRadii[SkRRect::kUpperLeft_Corner]. fY);
1022 add_corner_arc(this, bounds, rrect.fRadii[2].fX, rrect.fRadii[2].fY, 0, dir, false); 1122 add_corner_quads(this, rrect, SkRRect::kLowerLeft_Corner, dir);
1023 add_corner_arc(this, bounds, rrect.fRadii[1].fX, rrect.fRadii[1].fY, 270, dir, false); 1123 add_corner_quads(this, rrect, SkRRect::kLowerRight_Corner, dir);
1124 add_corner_quads(this, rrect, SkRRect::kUpperRight_Corner, dir);
1125 add_corner_quads(this, rrect, SkRRect::kUpperLeft_Corner, dir);
1024 } 1126 }
1025 this->close(); 1127 this->close();
1026 } 1128 }
1027 } 1129 }
1028 1130
1029 bool SkPath::hasOnlyMoveTos() const { 1131 bool SkPath::hasOnlyMoveTos() const {
1030 int count = fPathRef->countVerbs(); 1132 int count = fPathRef->countVerbs();
1031 const uint8_t* verbs = const_cast<const SkPathRef*>(fPathRef.get())->verbsMe mBegin(); 1133 const uint8_t* verbs = const_cast<const SkPathRef*>(fPathRef.get())->verbsMe mBegin();
1032 for (int i = 0; i < count; ++i) { 1134 for (int i = 0; i < count; ++i) {
1033 if (*verbs == kLine_Verb || 1135 if (*verbs == kLine_Verb ||
(...skipping 15 matching lines...) Expand all
1049 assert_known_direction(dir); 1151 assert_known_direction(dir);
1050 1152
1051 if (rx < 0 || ry < 0) { 1153 if (rx < 0 || ry < 0) {
1052 SkErrorInternals::SetError( kInvalidArgument_SkError, 1154 SkErrorInternals::SetError( kInvalidArgument_SkError,
1053 "I got %f and %f as radii to SkPath::AddRoun dRect, " 1155 "I got %f and %f as radii to SkPath::AddRoun dRect, "
1054 "but negative radii are not allowed.", 1156 "but negative radii are not allowed.",
1055 SkScalarToDouble(rx), SkScalarToDouble(ry) ) ; 1157 SkScalarToDouble(rx), SkScalarToDouble(ry) ) ;
1056 return; 1158 return;
1057 } 1159 }
1058 1160
1161 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1059 SkScalar w = rect.width(); 1162 SkScalar w = rect.width();
1060 SkScalar halfW = SkScalarHalf(w); 1163 SkScalar halfW = SkScalarHalf(w);
1061 SkScalar h = rect.height(); 1164 SkScalar h = rect.height();
1062 SkScalar halfH = SkScalarHalf(h); 1165 SkScalar halfH = SkScalarHalf(h);
1063 1166
1064 if (halfW <= 0 || halfH <= 0) { 1167 if (halfW <= 0 || halfH <= 0) {
1065 return; 1168 return;
1066 } 1169 }
1067 1170
1068 bool skip_hori = rx >= halfW; 1171 bool skip_hori = rx >= halfW;
1069 bool skip_vert = ry >= halfH; 1172 bool skip_vert = ry >= halfH;
1070 1173
1071 if (skip_hori && skip_vert) { 1174 if (skip_hori && skip_vert) {
1072 this->addOval(rect, dir); 1175 this->addOval(rect, dir);
1073 return; 1176 return;
1074 } 1177 }
1075 1178
1076 fDirection = this->hasOnlyMoveTos() ? dir : kUnknown_Direction; 1179 fDirection = this->hasOnlyMoveTos() ? dir : kUnknown_Direction;
1077 1180
1078 SkAutoPathBoundsUpdate apbu(this, rect); 1181 SkAutoPathBoundsUpdate apbu(this, rect);
1079 SkAutoDisableDirectionCheck(this); 1182 SkAutoDisableDirectionCheck(this);
1080 1183
1081 if (skip_hori) { 1184 if (skip_hori) {
1082 rx = halfW; 1185 rx = halfW;
1083 } else if (skip_vert) { 1186 } else if (skip_vert) {
1084 ry = halfH; 1187 ry = halfH;
1085 } 1188 }
1086 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1087 SkScalar sx = SkScalarMul(rx, CUBIC_ARC_FACTOR); 1189 SkScalar sx = SkScalarMul(rx, CUBIC_ARC_FACTOR);
1088 SkScalar sy = SkScalarMul(ry, CUBIC_ARC_FACTOR); 1190 SkScalar sy = SkScalarMul(ry, CUBIC_ARC_FACTOR);
1089 1191
1090 this->incReserve(17); 1192 this->incReserve(17);
1091 #else
1092 // The mid point of the quadratic arc approximation is half way between the two
1093 // control points. The float epsilon adjustment moves the on curve point out by
1094 // two bits, distributing the convex test error between the round rect approxima tion
1095 // and the convex cross product sign equality test.
1096 SkScalar midPtX = rx * (SK_Scalar1 + SK_ScalarTanPIOver8 + FLT_EPSILON * 4) / 2;
1097 SkScalar midPtY = ry * (SK_Scalar1 + SK_ScalarTanPIOver8 + FLT_EPSILON * 4) / 2;
1098
1099 SkScalar offPtX = rx * SK_ScalarTanPIOver8;
1100 SkScalar offPtY = ry * SK_ScalarTanPIOver8;
1101
1102 this->incReserve(21);
1103 #endif
1104 this->moveTo(rect.fRight - rx, rect.fTop); // top-right 1193 this->moveTo(rect.fRight - rx, rect.fTop); // top-right
1105 if (dir == kCCW_Direction) { 1194 if (dir == kCCW_Direction) {
1106 if (!skip_hori) { 1195 if (!skip_hori) {
1107 this->lineTo(rect.fLeft + rx, rect.fTop); // top 1196 this->lineTo(rect.fLeft + rx, rect.fTop); // top
1108 } 1197 }
1109 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1110 this->cubicTo(rect.fLeft + rx - sx, rect.fTop, 1198 this->cubicTo(rect.fLeft + rx - sx, rect.fTop,
1111 rect.fLeft, rect.fTop + ry - sy, 1199 rect.fLeft, rect.fTop + ry - sy,
1112 rect.fLeft, rect.fTop + ry); // top-left 1200 rect.fLeft, rect.fTop + ry); // top-left
1113 #else
1114 this->quadTo(rect.fLeft + rx - offPtX, rect.fTop,
1115 rect.fLeft + rx - midPtX, rect.fTop + ry - midPtY);
1116 this->quadTo(rect.fLeft, rect.fTop + ry - offPtY,
1117 rect.fLeft, rect.fTop + ry);
1118 #endif
1119 if (!skip_vert) { 1201 if (!skip_vert) {
1120 this->lineTo(rect.fLeft, rect.fBottom - ry); // left 1202 this->lineTo(rect.fLeft, rect.fBottom - ry); // left
1121 } 1203 }
1122 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1123 this->cubicTo(rect.fLeft, rect.fBottom - ry + sy, 1204 this->cubicTo(rect.fLeft, rect.fBottom - ry + sy,
1124 rect.fLeft + rx - sx, rect.fBottom, 1205 rect.fLeft + rx - sx, rect.fBottom,
1125 rect.fLeft + rx, rect.fBottom); // bot-left 1206 rect.fLeft + rx, rect.fBottom); // bot-left
1126 #else
1127 this->quadTo(rect.fLeft, rect.fBottom - ry + offPtY,
1128 rect.fLeft + rx - midPtX, rect.fBottom - ry + midPtY);
1129 this->quadTo(rect.fLeft + rx - offPtX, rect.fBottom,
1130 rect.fLeft + rx, rect.fBottom);
1131 #endif
1132 if (!skip_hori) { 1207 if (!skip_hori) {
1133 this->lineTo(rect.fRight - rx, rect.fBottom); // bottom 1208 this->lineTo(rect.fRight - rx, rect.fBottom); // bottom
1134 } 1209 }
1135 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1136 this->cubicTo(rect.fRight - rx + sx, rect.fBottom, 1210 this->cubicTo(rect.fRight - rx + sx, rect.fBottom,
1137 rect.fRight, rect.fBottom - ry + sy, 1211 rect.fRight, rect.fBottom - ry + sy,
1138 rect.fRight, rect.fBottom - ry); // bot-right 1212 rect.fRight, rect.fBottom - ry); // bot-right
1139 #else
1140 this->quadTo(rect.fRight - rx + offPtX, rect.fBottom,
1141 rect.fRight - rx + midPtX, rect.fBottom - ry + midPtY);
1142 this->quadTo(rect.fRight, rect.fBottom - ry + offPtY,
1143 rect.fRight, rect.fBottom - ry);
1144 #endif
1145 if (!skip_vert) { 1213 if (!skip_vert) {
1146 this->lineTo(rect.fRight, rect.fTop + ry); // right 1214 this->lineTo(rect.fRight, rect.fTop + ry); // right
1147 } 1215 }
1148 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1149 this->cubicTo(rect.fRight, rect.fTop + ry - sy, 1216 this->cubicTo(rect.fRight, rect.fTop + ry - sy,
1150 rect.fRight - rx + sx, rect.fTop, 1217 rect.fRight - rx + sx, rect.fTop,
1151 rect.fRight - rx, rect.fTop); // top-right 1218 rect.fRight - rx, rect.fTop); // top-right
1152 #else
1153 this->quadTo(rect.fRight, rect.fTop + ry - offPtY,
1154 rect.fRight - rx + midPtX, rect.fTop + ry - midPtY);
1155 this->quadTo(rect.fRight - rx + offPtX, rect.fTop,
1156 rect.fRight - rx, rect.fTop);
1157 #endif
1158 } else { 1219 } else {
1159 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1160 this->cubicTo(rect.fRight - rx + sx, rect.fTop, 1220 this->cubicTo(rect.fRight - rx + sx, rect.fTop,
1161 rect.fRight, rect.fTop + ry - sy, 1221 rect.fRight, rect.fTop + ry - sy,
1162 rect.fRight, rect.fTop + ry); // top-right 1222 rect.fRight, rect.fTop + ry); // top-right
1163 #else
1164 this->quadTo(rect.fRight - rx + offPtX, rect.fTop,
1165 rect.fRight - rx + midPtX, rect.fTop + ry - midPtY);
1166 this->quadTo(rect.fRight, rect.fTop + ry - offPtY,
1167 rect.fRight, rect.fTop + ry);
1168 #endif
1169 if (!skip_vert) { 1223 if (!skip_vert) {
1170 this->lineTo(rect.fRight, rect.fBottom - ry); // right 1224 this->lineTo(rect.fRight, rect.fBottom - ry); // right
1171 } 1225 }
1172 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1173 this->cubicTo(rect.fRight, rect.fBottom - ry + sy, 1226 this->cubicTo(rect.fRight, rect.fBottom - ry + sy,
1174 rect.fRight - rx + sx, rect.fBottom, 1227 rect.fRight - rx + sx, rect.fBottom,
1175 rect.fRight - rx, rect.fBottom); // bot-right 1228 rect.fRight - rx, rect.fBottom); // bot-right
1176 #else
1177 this->quadTo(rect.fRight, rect.fBottom - ry + offPtY,
1178 rect.fRight - rx + midPtX, rect.fBottom - ry + midPtY);
1179 this->quadTo(rect.fRight - rx + offPtX, rect.fBottom,
1180 rect.fRight - rx, rect.fBottom);
1181 #endif
1182 if (!skip_hori) { 1229 if (!skip_hori) {
1183 this->lineTo(rect.fLeft + rx, rect.fBottom); // bottom 1230 this->lineTo(rect.fLeft + rx, rect.fBottom); // bottom
1184 } 1231 }
1185 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1186 this->cubicTo(rect.fLeft + rx - sx, rect.fBottom, 1232 this->cubicTo(rect.fLeft + rx - sx, rect.fBottom,
1187 rect.fLeft, rect.fBottom - ry + sy, 1233 rect.fLeft, rect.fBottom - ry + sy,
1188 rect.fLeft, rect.fBottom - ry); // bot-left 1234 rect.fLeft, rect.fBottom - ry); // bot-left
1189 #else
1190 this->quadTo(rect.fLeft + rx - offPtX, rect.fBottom,
1191 rect.fLeft + rx - midPtX, rect.fBottom - ry + midPtY);
1192 this->quadTo(rect.fLeft, rect.fBottom - ry + offPtY,
1193 rect.fLeft, rect.fBottom - ry);
1194 #endif
1195 if (!skip_vert) { 1235 if (!skip_vert) {
1196 this->lineTo(rect.fLeft, rect.fTop + ry); // left 1236 this->lineTo(rect.fLeft, rect.fTop + ry); // left
1197 } 1237 }
1198 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1199 this->cubicTo(rect.fLeft, rect.fTop + ry - sy, 1238 this->cubicTo(rect.fLeft, rect.fTop + ry - sy,
1200 rect.fLeft + rx - sx, rect.fTop, 1239 rect.fLeft + rx - sx, rect.fTop,
1201 rect.fLeft + rx, rect.fTop); // top-left 1240 rect.fLeft + rx, rect.fTop); // top-left
1202 #else
1203 this->quadTo(rect.fLeft, rect.fTop + ry - offPtY,
1204 rect.fLeft + rx - midPtX, rect.fTop + ry - midPtY);
1205 this->quadTo(rect.fLeft + rx - offPtX, rect.fTop,
1206 rect.fLeft + rx, rect.fTop);
1207 #endif
1208 if (!skip_hori) { 1241 if (!skip_hori) {
1209 this->lineTo(rect.fRight - rx, rect.fTop); // top 1242 this->lineTo(rect.fRight - rx, rect.fTop); // top
1210 } 1243 }
1211 } 1244 }
1212 this->close(); 1245 this->close();
1246 #else
1247 SkRRect rrect;
1248 rrect.setRectXY(rect, rx, ry);
1249 this->addRRect(rrect, dir);
1250 #endif
1213 } 1251 }
1214 1252
1215 void SkPath::addOval(const SkRect& oval, Direction dir) { 1253 void SkPath::addOval(const SkRect& oval, Direction dir) {
1216 assert_known_direction(dir); 1254 assert_known_direction(dir);
1217 1255
1218 /* If addOval() is called after previous moveTo(), 1256 /* If addOval() is called after previous moveTo(),
1219 this path is still marked as an oval. This is used to 1257 this path is still marked as an oval. This is used to
1220 fit into WebKit's calling sequences. 1258 fit into WebKit's calling sequences.
1221 We can't simply check isEmpty() in this case, as additional 1259 We can't simply check isEmpty() in this case, as additional
1222 moveTo() would mark the path non empty. 1260 moveTo() would mark the path non empty.
(...skipping 1779 matching lines...) Expand 10 before | Expand all | Expand 10 after
3002 switch (this->getFillType()) { 3040 switch (this->getFillType()) {
3003 case SkPath::kEvenOdd_FillType: 3041 case SkPath::kEvenOdd_FillType:
3004 case SkPath::kInverseEvenOdd_FillType: 3042 case SkPath::kInverseEvenOdd_FillType:
3005 w &= 1; 3043 w &= 1;
3006 break; 3044 break;
3007 default: 3045 default:
3008 break; 3046 break;
3009 } 3047 }
3010 return SkToBool(w); 3048 return SkToBool(w);
3011 } 3049 }
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