OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 } |
OLD | NEW |