-
Notifications
You must be signed in to change notification settings - Fork 5
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Sg/calc angles #50
Merged
Merged
Sg/calc angles #50
Changes from 4 commits
Commits
Show all changes
13 commits
Select commit
Hold shift + click to select a range
2aab374
Add angle code
skygering 6a3bad6
Merge branch 'main' into sg/calc_angles
skygering 5c85147
Finish angle code
skygering 25e8934
Add tests back in
skygering 1ddd181
Change hole angles to interior
skygering ac0f4b8
Fix angle docs
skygering 0070049
Add multipoints
skygering 2e0457e
Update doc manifest
skygering 89b3ae4
Update makedocs keywords
skygering 7c6bea8
Add documenter source code to gitignore
skygering 7bb3058
Update comment examples to find error
skygering 83820f3
More comment example fixes
skygering 336c7ee
More document fixes
skygering File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
@@ -0,0 +1,164 @@ | ||||||
# # Angles | ||||||
export angles | ||||||
|
||||||
#= | ||||||
## What is angles? | ||||||
|
||||||
Angles are the angles formed by a given geometries line segments, if it has line segments. | ||||||
|
||||||
To provide an example, consider this rectangle: | ||||||
```@example angles | ||||||
using GeometryOps | ||||||
using GeometryOps.GeometryBasics | ||||||
using Makie | ||||||
|
||||||
rect = Polygon([Point(0,0), Point(0,1), Point(1,1), Point(1,0), Point(0, 0)]) | ||||||
f, a, p = poly(rect; axis = (; aspect = DataAspect())) | ||||||
``` | ||||||
This is clearly a rectangle, with angles of 90 degrees. | ||||||
```@example angles | ||||||
lines!(a, rect; color = 1:length(coordinates(rect))+1) | ||||||
f | ||||||
angles(area) | ||||||
``` | ||||||
|
||||||
## Implementation | ||||||
|
||||||
This is the GeoInterface-compatible implementation. First, we implement a | ||||||
wrapper method that dispatches to the correct implementation based on the | ||||||
geometry trait. This is also used in the implementation, since it's a lot less | ||||||
work! | ||||||
=# | ||||||
|
||||||
""" | ||||||
angles(geom, ::Type{T} = Float64) | ||||||
|
||||||
Returns the angles of a geometry or collection of geometries. | ||||||
This is computed differently for different geometries: | ||||||
|
||||||
- The angles of a point is an empty vector. | ||||||
- The angles of a single line segment is an empty vector. | ||||||
- The angles of a linestring or linearring is a vector of angles formed by the curve. | ||||||
- The angles of a polygin is a vector of vectors of angles formed by each ring. | ||||||
- The angles of a multi-geometry collection is a vector of the angles of each of the | ||||||
sub-geometries as defined above. | ||||||
|
||||||
Result will be a Vector, or nested set of vectors, of type T where an optional argument with | ||||||
a default value of Float64. | ||||||
""" | ||||||
angles(geom, ::Type{T} = Float64) where T <: AbstractFloat = | ||||||
_angles(T, GI.trait(geom), geom) | ||||||
|
||||||
# Points and single line segments have no angles | ||||||
_angles(::Type{T}, ::Union{GI.PointTrait, GI.LineTrait}, geom) where T = T[] | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe MultiPointTrait is missing here?
Suggested change
|
||||||
|
||||||
#= The angles of a linestring are the angles formed by the line. If the first and last point | ||||||
are not explicitly repeated, the geom is not considered closed. The angles should all be on | ||||||
one side of the line, but a particular side is not guaranteed by this function. =# | ||||||
function _angles(::Type{T}, ::Union{GI.LineStringTrait}, geom) where T | ||||||
npoints = GI.npoint(geom) | ||||||
first_last_equal = equals(GI.getpoint(geom, 1), GI.getpoint(geom, npoints)) | ||||||
angle_list = Vector{T}(undef, npoints - (first_last_equal ? 1 : 2)) | ||||||
_find_angles!( | ||||||
T, angle_list, geom; | ||||||
offset = first_last_equal, close_geom = false, | ||||||
) | ||||||
return angle_list | ||||||
end | ||||||
|
||||||
#= The angles of a linearring are the angles within the closed line and include the angles | ||||||
formed by connecting the first and last points of the curve. =# | ||||||
function _angles(::Type{T}, ::GI.LinearRingTrait, geom) where T | ||||||
npoints = GI.npoint(geom) | ||||||
first_last_equal = equals(GI.getpoint(geom, 1), GI.getpoint(geom, npoints)) | ||||||
angle_list = Vector{T}(undef, npoints - (first_last_equal ? 1 : 0)) | ||||||
_find_angles!( | ||||||
T, angle_list, geom; | ||||||
offset = true, close_geom = !first_last_equal, | ||||||
) | ||||||
return angle_list | ||||||
end | ||||||
|
||||||
#= The angles of a polygon is a vector of lists angles of its rings. Note that | ||||||
this means that the angles of any holes will be the exterior angles of the holes | ||||||
outside of the geometry, rather than the interior angles=# | ||||||
_angles(::Type{T}, ::GI.PolygonTrait, geom) where T = | ||||||
[_angles(T, GI.LinearRingTrait(), g) for g in GI.getring(geom)] | ||||||
|
||||||
# Angles of a multi-geometry is simply a list of the angles of its sub-geometries. | ||||||
_angles(::Type{T}, ::MultiGeomTrait, geom) where T = [angles(g, T) for g in GI.getgeom(geom)] | ||||||
skygering marked this conversation as resolved.
Show resolved
Hide resolved
|
||||||
|
||||||
#= | ||||||
Find angles of a curve and insert the values into the angle_list. If offset is true, then | ||||||
save space for the angle at the first vertex, as the curve is closed, at the front of | ||||||
angle_list. If close_geom is true, then despite the first and last point not being | ||||||
explicitly repeated, the curve is closed and the angle of the last point should be added to | ||||||
angle_list. | ||||||
=# | ||||||
function _find_angles!(::Type{T}, angle_list, geom; offset, close_geom) where T | ||||||
local p1, prev_p1_diff, p2_p1_diff | ||||||
local start_point, start_diff | ||||||
local extreem_idx, extreem_x, extreem_y | ||||||
i_offset = offset ? 1 : 0 | ||||||
# Loop through the curve and find each of the angels | ||||||
for (i, p2) in enumerate(GI.getpoint(geom)) | ||||||
xp2, yp2 = GI.x(p2), GI.y(p2) | ||||||
#= Find point with smallest x values (and smallest y in case of a tie) as this point | ||||||
is know to be convex. =# | ||||||
if i == 1 || (xp2 < extreem_x || (xp2 == extreem_x && yp2 < extreem_y)) | ||||||
extreem_idx = i | ||||||
extreem_x, extreem_y = xp2, yp2 | ||||||
end | ||||||
if i > 1 | ||||||
p2_p1_diff = (xp2 - GI.x(p1), yp2 - GI.y(p1)) | ||||||
if i == 2 | ||||||
start_point = p1 | ||||||
start_diff = p2_p1_diff | ||||||
else | ||||||
angle_list[i - 2 + i_offset] = _diffs_calc_angle(T, prev_p1_diff, p2_p1_diff) | ||||||
end | ||||||
prev_p1_diff = -1 .* p2_p1_diff | ||||||
end | ||||||
p1 = p2 | ||||||
end | ||||||
# If the last point of geometry should be the same as the first, calculate closing angle | ||||||
if close_geom | ||||||
p2_p1_diff = (GI.x(start_point) - GI.x(p1), GI.y(start_point) - GI.y(p1)) | ||||||
angle_list[end] = _diffs_calc_angle(T, prev_p1_diff, p2_p1_diff) | ||||||
prev_p1_diff = -1 .* p2_p1_diff | ||||||
end | ||||||
# If needed, calculate first angle corresponding to the first point | ||||||
if offset | ||||||
angle_list[1] = _diffs_calc_angle(T, prev_p1_diff, start_diff) | ||||||
end | ||||||
#= Make sure that all of the angles are on the same side of the line and inside of the | ||||||
closed ring if the input geometry is closed. =# | ||||||
convex_sgn = sign(angle_list[extreem_idx]) | ||||||
for i in eachindex(angle_list) | ||||||
idx_sgn = sign(angle_list[i]) | ||||||
if idx_sgn == -1 | ||||||
angle_list[i] = abs(angle_list[i]) | ||||||
end | ||||||
if idx_sgn != convex_sgn | ||||||
angle_list[i] = 360 - angle_list[i] | ||||||
end | ||||||
end | ||||||
return | ||||||
end | ||||||
|
||||||
#= | ||||||
Calculate the angle between two vectors defined by the previous and current Δx and Δys. | ||||||
Angle will have a sign corresponding to the sign of the cross product between the two | ||||||
vectors. All angles of one sign in a given geometry are convex, while those of the other | ||||||
sign are concave. However, the sign corresponding to each of these can vary based on | ||||||
geometry and thus you must compare to an angle that is know to be convex or concave. | ||||||
=# | ||||||
function _diffs_calc_angle(::Type{T}, (Δx_prev, Δy_prev), (Δx_curr, Δy_curr)) where T | ||||||
cross_prod = Δx_prev * Δy_curr - Δy_prev * Δx_curr | ||||||
dot_prod = Δx_prev * Δx_curr + Δy_prev * Δy_curr | ||||||
prev_mag = max(sqrt(Δx_prev^2 + Δy_prev^2), eps(T)) | ||||||
curr_mag = max(sqrt(Δx_curr^2 + Δy_curr^2), eps(T)) | ||||||
val = clamp(dot_prod / (prev_mag * curr_mag), -one(T), one(T)) | ||||||
angle = real(acos(val) * 180 / π) | ||||||
return angle * (cross_prod < 0 ? -1 : 1) | ||||||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,30 @@ | ||
pt1 = GI.Point((0.0, 0.0)) | ||
l1 = GI.Line([(0.0, 0.0), (0.0, 1.0)]) | ||
|
||
concave_coords = [(0.0, 0.0), (0.0, 1.0), (-1.0, 1.0), (-1.0, 2.0), (2.0, 2.0), (2.0, 0.0), (0.0, 0.0)] | ||
l2 = GI.LineString(concave_coords) | ||
l3 = GI.LineString(concave_coords[1:(end - 1)]) | ||
r1 = GI.LinearRing(concave_coords) | ||
r2 = GI.LinearRing(concave_coords[1:(end - 1)]) | ||
concave_angles = [90.0, 270.0, 90.0, 90.0, 90.0, 90.0] | ||
|
||
p1 = GI.Polygon([[(1.0, 1.0), (1.0, 2.0), (2.0, 2.0), (2.0, 1.0), (1.0, 1.0)]]) | ||
p2 = GI.Polygon([[(0.0, 0.0), (0.0, 4.0), (3.0, 0.0), (0.0, 0.0)]]) | ||
p3 = GI.Polygon([[(-3.0, -2.0), (0.0,0.0), (5.0, 0.0), (-3.0, -2.0)]]) | ||
p4 = GI.Polygon([r1]) | ||
|
||
# Points and lines | ||
@test isempty(GO.angles(pt1)) | ||
@test isempty(GO.angles(l1)) | ||
|
||
# LineStrings and Linear Rings | ||
@test all(isapprox.(GO.angles(l2), concave_angles, atol = 1e-3)) | ||
@test all(isapprox.(GO.angles(l3), concave_angles[2:(end - 1)], atol = 1e-3)) | ||
@test all(isapprox.(GO.angles(r1), concave_angles, atol = 1e-3)) | ||
@test all(isapprox.(GO.angles(r2), concave_angles, atol = 1e-3)) | ||
|
||
# Polygons | ||
@test all(isapprox.(GO.angles(p1)[1], [90.0, 90.0, 90.0, 90.0], atol = 1e-3)) | ||
@test all(isapprox.(GO.angles(p2)[1], [90.0, 36.8699, 53.1301], atol = 1e-3)) | ||
@test all(isapprox.(GO.angles(p3)[1], [19.6538, 146.3099, 14.0362], atol = 1e-3)) | ||
@test all(isapprox.(GO.angles(p4)[1], concave_angles, atol = 1e-3)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks like this isn't used now