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

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: remove stale comment 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.
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);
robertphillips 2013/11/06 13:31:58 I don't think this needs to be #ifdef protected. C
caryclark 2013/11/06 13:39:55 This is ifdef protected to avoid recursive functio
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);
robertphillips 2013/11/06 13:31:58 Do we need a "SkAutoDisableDirectionCheck(this);"
caryclark 2013/11/06 13:39:55 I didn't know about this helper. Yes, it looks lik
caryclark 2013/11/06 13:58:28 Done.
1013 1109
1110 this->incReserve(21);
1014 if (kCW_Direction == dir) { 1111 if (kCW_Direction == dir) {
1015 add_corner_arc(this, bounds, rrect.fRadii[0].fX, rrect.fRadii[0].fY, 180, dir, true); 1112 this->moveTo(bounds.fLeft,
1016 add_corner_arc(this, bounds, rrect.fRadii[1].fX, rrect.fRadii[1].fY, 270, dir, false); 1113 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); 1114 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); 1115 add_corner_quads(this, rrect, SkRRect::kUpperRight_Corner, dir);
1116 add_corner_quads(this, rrect, SkRRect::kLowerRight_Corner, dir);
1117 add_corner_quads(this, rrect, SkRRect::kLowerLeft_Corner, dir);
1019 } else { 1118 } else {
1020 add_corner_arc(this, bounds, rrect.fRadii[0].fX, rrect.fRadii[0].fY, 180, dir, true); 1119 this->moveTo(bounds.fLeft,
1021 add_corner_arc(this, bounds, rrect.fRadii[3].fX, rrect.fRadii[3].fY, 90, dir, false); 1120 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); 1121 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); 1122 add_corner_quads(this, rrect, SkRRect::kLowerRight_Corner, dir);
1123 add_corner_quads(this, rrect, SkRRect::kUpperRight_Corner, dir);
1124 add_corner_quads(this, rrect, SkRRect::kUpperLeft_Corner, dir);
1024 } 1125 }
1025 this->close(); 1126 this->close();
1026 } 1127 }
1027 } 1128 }
1028 1129
1029 bool SkPath::hasOnlyMoveTos() const { 1130 bool SkPath::hasOnlyMoveTos() const {
1030 int count = fPathRef->countVerbs(); 1131 int count = fPathRef->countVerbs();
1031 const uint8_t* verbs = const_cast<const SkPathRef*>(fPathRef.get())->verbsMe mBegin(); 1132 const uint8_t* verbs = const_cast<const SkPathRef*>(fPathRef.get())->verbsMe mBegin();
1032 for (int i = 0; i < count; ++i) { 1133 for (int i = 0; i < count; ++i) {
1033 if (*verbs == kLine_Verb || 1134 if (*verbs == kLine_Verb ||
(...skipping 15 matching lines...) Expand all
1049 assert_known_direction(dir); 1150 assert_known_direction(dir);
1050 1151
1051 if (rx < 0 || ry < 0) { 1152 if (rx < 0 || ry < 0) {
1052 SkErrorInternals::SetError( kInvalidArgument_SkError, 1153 SkErrorInternals::SetError( kInvalidArgument_SkError,
1053 "I got %f and %f as radii to SkPath::AddRoun dRect, " 1154 "I got %f and %f as radii to SkPath::AddRoun dRect, "
1054 "but negative radii are not allowed.", 1155 "but negative radii are not allowed.",
1055 SkScalarToDouble(rx), SkScalarToDouble(ry) ) ; 1156 SkScalarToDouble(rx), SkScalarToDouble(ry) ) ;
1056 return; 1157 return;
1057 } 1158 }
1058 1159
1160 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1059 SkScalar w = rect.width(); 1161 SkScalar w = rect.width();
1060 SkScalar halfW = SkScalarHalf(w); 1162 SkScalar halfW = SkScalarHalf(w);
1061 SkScalar h = rect.height(); 1163 SkScalar h = rect.height();
1062 SkScalar halfH = SkScalarHalf(h); 1164 SkScalar halfH = SkScalarHalf(h);
1063 1165
1064 if (halfW <= 0 || halfH <= 0) { 1166 if (halfW <= 0 || halfH <= 0) {
1065 return; 1167 return;
1066 } 1168 }
1067 1169
1068 bool skip_hori = rx >= halfW; 1170 bool skip_hori = rx >= halfW;
1069 bool skip_vert = ry >= halfH; 1171 bool skip_vert = ry >= halfH;
1070 1172
1071 if (skip_hori && skip_vert) { 1173 if (skip_hori && skip_vert) {
1072 this->addOval(rect, dir); 1174 this->addOval(rect, dir);
1073 return; 1175 return;
1074 } 1176 }
1075 1177
1076 fDirection = this->hasOnlyMoveTos() ? dir : kUnknown_Direction; 1178 fDirection = this->hasOnlyMoveTos() ? dir : kUnknown_Direction;
1077 1179
1078 SkAutoPathBoundsUpdate apbu(this, rect); 1180 SkAutoPathBoundsUpdate apbu(this, rect);
1079 SkAutoDisableDirectionCheck(this); 1181 SkAutoDisableDirectionCheck(this);
1080 1182
1081 if (skip_hori) { 1183 if (skip_hori) {
1082 rx = halfW; 1184 rx = halfW;
1083 } else if (skip_vert) { 1185 } else if (skip_vert) {
1084 ry = halfH; 1186 ry = halfH;
1085 } 1187 }
1086 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1087 SkScalar sx = SkScalarMul(rx, CUBIC_ARC_FACTOR); 1188 SkScalar sx = SkScalarMul(rx, CUBIC_ARC_FACTOR);
1088 SkScalar sy = SkScalarMul(ry, CUBIC_ARC_FACTOR); 1189 SkScalar sy = SkScalarMul(ry, CUBIC_ARC_FACTOR);
1089 1190
1090 this->incReserve(17); 1191 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 1192 this->moveTo(rect.fRight - rx, rect.fTop); // top-right
1105 if (dir == kCCW_Direction) { 1193 if (dir == kCCW_Direction) {
1106 if (!skip_hori) { 1194 if (!skip_hori) {
1107 this->lineTo(rect.fLeft + rx, rect.fTop); // top 1195 this->lineTo(rect.fLeft + rx, rect.fTop); // top
1108 } 1196 }
1109 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1110 this->cubicTo(rect.fLeft + rx - sx, rect.fTop, 1197 this->cubicTo(rect.fLeft + rx - sx, rect.fTop,
1111 rect.fLeft, rect.fTop + ry - sy, 1198 rect.fLeft, rect.fTop + ry - sy,
1112 rect.fLeft, rect.fTop + ry); // top-left 1199 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) { 1200 if (!skip_vert) {
1120 this->lineTo(rect.fLeft, rect.fBottom - ry); // left 1201 this->lineTo(rect.fLeft, rect.fBottom - ry); // left
1121 } 1202 }
1122 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1123 this->cubicTo(rect.fLeft, rect.fBottom - ry + sy, 1203 this->cubicTo(rect.fLeft, rect.fBottom - ry + sy,
1124 rect.fLeft + rx - sx, rect.fBottom, 1204 rect.fLeft + rx - sx, rect.fBottom,
1125 rect.fLeft + rx, rect.fBottom); // bot-left 1205 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) { 1206 if (!skip_hori) {
1133 this->lineTo(rect.fRight - rx, rect.fBottom); // bottom 1207 this->lineTo(rect.fRight - rx, rect.fBottom); // bottom
1134 } 1208 }
1135 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1136 this->cubicTo(rect.fRight - rx + sx, rect.fBottom, 1209 this->cubicTo(rect.fRight - rx + sx, rect.fBottom,
1137 rect.fRight, rect.fBottom - ry + sy, 1210 rect.fRight, rect.fBottom - ry + sy,
1138 rect.fRight, rect.fBottom - ry); // bot-right 1211 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) { 1212 if (!skip_vert) {
1146 this->lineTo(rect.fRight, rect.fTop + ry); // right 1213 this->lineTo(rect.fRight, rect.fTop + ry); // right
1147 } 1214 }
1148 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1149 this->cubicTo(rect.fRight, rect.fTop + ry - sy, 1215 this->cubicTo(rect.fRight, rect.fTop + ry - sy,
1150 rect.fRight - rx + sx, rect.fTop, 1216 rect.fRight - rx + sx, rect.fTop,
1151 rect.fRight - rx, rect.fTop); // top-right 1217 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 { 1218 } else {
1159 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1160 this->cubicTo(rect.fRight - rx + sx, rect.fTop, 1219 this->cubicTo(rect.fRight - rx + sx, rect.fTop,
1161 rect.fRight, rect.fTop + ry - sy, 1220 rect.fRight, rect.fTop + ry - sy,
1162 rect.fRight, rect.fTop + ry); // top-right 1221 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) { 1222 if (!skip_vert) {
1170 this->lineTo(rect.fRight, rect.fBottom - ry); // right 1223 this->lineTo(rect.fRight, rect.fBottom - ry); // right
1171 } 1224 }
1172 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1173 this->cubicTo(rect.fRight, rect.fBottom - ry + sy, 1225 this->cubicTo(rect.fRight, rect.fBottom - ry + sy,
1174 rect.fRight - rx + sx, rect.fBottom, 1226 rect.fRight - rx + sx, rect.fBottom,
1175 rect.fRight - rx, rect.fBottom); // bot-right 1227 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) { 1228 if (!skip_hori) {
1183 this->lineTo(rect.fLeft + rx, rect.fBottom); // bottom 1229 this->lineTo(rect.fLeft + rx, rect.fBottom); // bottom
1184 } 1230 }
1185 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1186 this->cubicTo(rect.fLeft + rx - sx, rect.fBottom, 1231 this->cubicTo(rect.fLeft + rx - sx, rect.fBottom,
1187 rect.fLeft, rect.fBottom - ry + sy, 1232 rect.fLeft, rect.fBottom - ry + sy,
1188 rect.fLeft, rect.fBottom - ry); // bot-left 1233 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) { 1234 if (!skip_vert) {
1196 this->lineTo(rect.fLeft, rect.fTop + ry); // left 1235 this->lineTo(rect.fLeft, rect.fTop + ry); // left
1197 } 1236 }
1198 #ifdef SK_IGNORE_QUAD_RR_CORNERS_OPT
1199 this->cubicTo(rect.fLeft, rect.fTop + ry - sy, 1237 this->cubicTo(rect.fLeft, rect.fTop + ry - sy,
1200 rect.fLeft + rx - sx, rect.fTop, 1238 rect.fLeft + rx - sx, rect.fTop,
1201 rect.fLeft + rx, rect.fTop); // top-left 1239 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) { 1240 if (!skip_hori) {
1209 this->lineTo(rect.fRight - rx, rect.fTop); // top 1241 this->lineTo(rect.fRight - rx, rect.fTop); // top
1210 } 1242 }
1211 } 1243 }
1212 this->close(); 1244 this->close();
1245 #else
1246 SkRRect rrect;
1247 rrect.setRectXY(rect, rx, ry);
1248 this->addRRect(rrect, dir);
1249 #endif
1213 } 1250 }
1214 1251
1215 void SkPath::addOval(const SkRect& oval, Direction dir) { 1252 void SkPath::addOval(const SkRect& oval, Direction dir) {
1216 assert_known_direction(dir); 1253 assert_known_direction(dir);
1217 1254
1218 /* If addOval() is called after previous moveTo(), 1255 /* If addOval() is called after previous moveTo(),
1219 this path is still marked as an oval. This is used to 1256 this path is still marked as an oval. This is used to
1220 fit into WebKit's calling sequences. 1257 fit into WebKit's calling sequences.
1221 We can't simply check isEmpty() in this case, as additional 1258 We can't simply check isEmpty() in this case, as additional
1222 moveTo() would mark the path non empty. 1259 moveTo() would mark the path non empty.
(...skipping 1779 matching lines...) Expand 10 before | Expand all | Expand 10 after
3002 switch (this->getFillType()) { 3039 switch (this->getFillType()) {
3003 case SkPath::kEvenOdd_FillType: 3040 case SkPath::kEvenOdd_FillType:
3004 case SkPath::kInverseEvenOdd_FillType: 3041 case SkPath::kInverseEvenOdd_FillType:
3005 w &= 1; 3042 w &= 1;
3006 break; 3043 break;
3007 default: 3044 default:
3008 break; 3045 break;
3009 } 3046 }
3010 return SkToBool(w); 3047 return SkToBool(w);
3011 } 3048 }
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