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