|
| 1 | +#include "ink/strokes/internal/brush_tip_extrusion.h" |
| 2 | + |
| 3 | +#include <algorithm> |
| 4 | +#include <cstdlib> |
| 5 | + |
| 6 | +#include "absl/container/inlined_vector.h" |
| 7 | +#include "absl/log/absl_check.h" |
| 8 | +#include "absl/types/span.h" |
| 9 | +#include "ink/geometry/distance.h" |
| 10 | +#include "ink/geometry/internal/circle.h" |
| 11 | +#include "ink/geometry/rect.h" |
| 12 | +#include "ink/strokes/internal/brush_tip_shape.h" |
| 13 | +#include "ink/strokes/internal/brush_tip_state.h" |
| 14 | +#include "ink/strokes/internal/rounded_polygon.h" |
| 15 | + |
| 16 | +namespace ink::strokes_internal { |
| 17 | +namespace { |
| 18 | + |
| 19 | +using ::ink::geometry_internal::Circle; |
| 20 | + |
| 21 | +// Constructs a `RoundedPolygon` representing the shape that results from |
| 22 | +// connecting `first` and `second` by the tangents of the circles indicated by |
| 23 | +// `indices`. The returned shape starts at `indices.left.first`, proceeds to |
| 24 | +// `indices.right.first`, then jumps to `indices.right.second` and proceeds to |
| 25 | +// `indices.left.second`, always moving counter-clockwise per |
| 26 | +// `GetNextPerimeterIndexCcw`. `indices` is expected to be the result of calling |
| 27 | +// `GetTangentCircleIndices(first, second)`. `first` must not contain `second`, |
| 28 | +// and vice versa. |
| 29 | +// |
| 30 | +// `offset` specifies how much the returned `RoundedPolygon` should be offset |
| 31 | +// from the actual joined shape. This value must be >= 0. |
| 32 | +// |
| 33 | +// Note that not all of the component circles of `first` and `second` are |
| 34 | +// guaranteed to be contained in the returned shape; e.g. you could have two |
| 35 | +// rectangular `BrushTipShape`s that form a cross, which would leave two |
| 36 | +// circles outside the `RoundedPolygon`. |
| 37 | +static RoundedPolygon ConstructJoinedShape( |
| 38 | + const BrushTipShape& first, const BrushTipShape& second, |
| 39 | + const BrushTipShape::TangentCircleIndices& indices, float offset) { |
| 40 | + ABSL_DCHECK(!first.Contains(second)); |
| 41 | + ABSL_DCHECK(!second.Contains(first)); |
| 42 | + ABSL_CHECK_GE(offset, 0); |
| 43 | + |
| 44 | + // Each `BrushTipShape` has at most four circles, so we need at most eight |
| 45 | + // for the `RoundedPolygon`. |
| 46 | + absl::InlinedVector<Circle, 8> circles; |
| 47 | + |
| 48 | + auto add_circles = [&circles, offset](const BrushTipShape& shape, |
| 49 | + int first_index, int last_index) { |
| 50 | + auto add_one_circle_with_offset = [&circles, offset](const Circle& circle) { |
| 51 | + circles.push_back({circle.Center(), circle.Radius() + offset}); |
| 52 | + }; |
| 53 | + if (first_index == last_index) { |
| 54 | + add_one_circle_with_offset(shape.PerimeterCircles()[first_index]); |
| 55 | + } else { |
| 56 | + for (int index = first_index; index != last_index; |
| 57 | + index = shape.GetNextPerimeterIndexCcw(index)) { |
| 58 | + add_one_circle_with_offset(shape.PerimeterCircles()[index]); |
| 59 | + } |
| 60 | + add_one_circle_with_offset(shape.PerimeterCircles()[last_index]); |
| 61 | + } |
| 62 | + }; |
| 63 | + |
| 64 | + add_circles(first, indices.left.first, indices.right.first); |
| 65 | + add_circles(second, indices.right.second, indices.left.second); |
| 66 | + |
| 67 | + return RoundedPolygon(circles); |
| 68 | +} |
| 69 | + |
| 70 | +BrushTipExtrusion::TangentQuality EvaluateTangentQualityInternal( |
| 71 | + const BrushTipShape& first, const BrushTipShape& second) { |
| 72 | + if (first.Contains(second)) { |
| 73 | + return BrushTipExtrusion::TangentQuality::kNoTangentsFirstContainsSecond; |
| 74 | + } |
| 75 | + |
| 76 | + if ((second.Contains(first))) { |
| 77 | + return BrushTipExtrusion::TangentQuality::kNoTangentsSecondContainsFirst; |
| 78 | + } |
| 79 | + |
| 80 | + // If we have two circles that don't contain each other, then we can always |
| 81 | + // construct good tangents. |
| 82 | + // NOMUTANTS -- this is just a short-circuit for performance. |
| 83 | + if (first.PerimeterCircles().size() == 1 && |
| 84 | + second.PerimeterCircles().size() == 1) { |
| 85 | + return BrushTipExtrusion::TangentQuality::kGoodTangents; |
| 86 | + } |
| 87 | + |
| 88 | + // Fetch the indices of the circles that will be connect the two shapes. |
| 89 | + BrushTipShape::TangentCircleIndices indices = |
| 90 | + BrushTipShape::GetTangentCircleIndices(first, second); |
| 91 | + |
| 92 | + // If the first circle is immediately *clockwise* to the last circle for |
| 93 | + // each shape, then all circles contribute to the boundary of the joined |
| 94 | + // shape and there are no unused circles. |
| 95 | + // NOMUTANTS -- This is just a short-circuit for performance. |
| 96 | + if (first.GetNextPerimeterIndexCw(indices.left.first) == |
| 97 | + indices.right.first && |
| 98 | + second.GetNextPerimeterIndexCw(indices.right.second) == |
| 99 | + indices.left.second) { |
| 100 | + return BrushTipExtrusion::TangentQuality::kGoodTangents; |
| 101 | + } |
| 102 | + |
| 103 | + // In order to avoid false-negatives from `RoundedPolygon::ContainsCircle` due |
| 104 | + // to floating-point precision issues, we enlarge the joined shape by a small |
| 105 | + // amount. |
| 106 | + Rect first_bounds = first.Bounds(); |
| 107 | + Rect second_bounds = second.Bounds(); |
| 108 | + float max_absolute_coordinate = std::max( |
| 109 | + {std::abs(first_bounds.XMin()), std::abs(first_bounds.XMax()), |
| 110 | + std::abs(first_bounds.YMin()), std::abs(first_bounds.YMax()), |
| 111 | + std::abs(second_bounds.XMin()), std::abs(second_bounds.XMax()), |
| 112 | + std::abs(second_bounds.YMin()), std::abs(second_bounds.YMax())}); |
| 113 | + float offset = 1e-6 * max_absolute_coordinate; |
| 114 | + |
| 115 | + // Construct the joined shape, with the offset. |
| 116 | + RoundedPolygon joined_shape = |
| 117 | + ConstructJoinedShape(first, second, indices, offset); |
| 118 | + |
| 119 | + // Finally, check whether the unused circles are contained inside the joined |
| 120 | + // shape. |
| 121 | + for (int index = first.GetNextPerimeterIndexCcw(indices.right.first); |
| 122 | + index != indices.left.first; |
| 123 | + index = first.GetNextPerimeterIndexCcw(index)) { |
| 124 | + if (!joined_shape.ContainsCircle(first.PerimeterCircles()[index])) |
| 125 | + return BrushTipExtrusion::TangentQuality:: |
| 126 | + kBadTangentsJoinedShapeDoesNotCoverInputShapes; |
| 127 | + } |
| 128 | + for (int index = second.GetNextPerimeterIndexCcw(indices.left.second); |
| 129 | + index != indices.right.second; |
| 130 | + index = second.GetNextPerimeterIndexCcw(index)) { |
| 131 | + if (!joined_shape.ContainsCircle(second.PerimeterCircles()[index])) |
| 132 | + return BrushTipExtrusion::TangentQuality:: |
| 133 | + kBadTangentsJoinedShapeDoesNotCoverInputShapes; |
| 134 | + } |
| 135 | + |
| 136 | + return BrushTipExtrusion::TangentQuality::kGoodTangents; |
| 137 | +} |
| 138 | + |
| 139 | +bool StatesHaveEqualShapeProperties(const BrushTipState& first, |
| 140 | + const BrushTipState& second) { |
| 141 | + return first.width == second.width && first.height == second.height && |
| 142 | + first.percent_radius == second.percent_radius && |
| 143 | + first.rotation == second.rotation && first.slant == second.slant && |
| 144 | + first.pinch == second.pinch; |
| 145 | +} |
| 146 | + |
| 147 | +} // namespace |
| 148 | + |
| 149 | +BrushTipExtrusion::TangentQuality BrushTipExtrusion::EvaluateTangentQuality( |
| 150 | + const BrushTipExtrusion& first, const BrushTipExtrusion& second, |
| 151 | + float travel_threshold) { |
| 152 | + if ((Distance(first.GetState().position, second.GetState().position)) > |
| 153 | + travel_threshold && |
| 154 | + StatesHaveEqualShapeProperties(first.GetState(), second.GetState())) { |
| 155 | + return TangentQuality::kGoodTangents; |
| 156 | + } |
| 157 | + |
| 158 | + return EvaluateTangentQualityInternal(first.GetShape(), second.GetShape()); |
| 159 | +} |
| 160 | + |
| 161 | +} // namespace ink::strokes_internal |
0 commit comments