Skip to content
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

Add spherical circle, range, polygon #306

Closed
wants to merge 2 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions regions/shapes/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
from .point import *
from .polygon import *
from .rectangle import *
from .range import *
from .annulus import *
from .line import *
from .text import *
14 changes: 13 additions & 1 deletion regions/shapes/circle.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@
from ..core.attributes import (ScalarSky, ScalarPix, QuantityLength,
ScalarLength, RegionVisual, RegionMeta)

__all__ = ['CirclePixelRegion', 'CircleSkyRegion']
__all__ = ['CirclePixelRegion', 'CircleSkyRegion', 'CircleSphericalRegion']


class CirclePixelRegion(PixelRegion):
Expand Down Expand Up @@ -195,3 +195,15 @@ def to_pixel(self, wcs):
center, scale, _ = skycoord_to_pixel_scale_angle(self.center, wcs)
radius = self.radius.to('deg').value * scale
return CirclePixelRegion(center, radius, self.meta, self.visual)


class CircleSphericalRegion(CircleSkyRegion):
"""Spherical circle sky region.

TODO: document
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See my comment on #276 . I'd be happy to help review and/or write.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@gpdf - Thanks!
I'll add some docs in this PR, and then you can review.

"""

def contains(self, skycoord, wcs=None):
"""Defined by spherical distance."""
separation = self.center.separation(skycoord)
return separation < self.radius
72 changes: 72 additions & 0 deletions regions/shapes/range.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
# Licensed under a 3-clause BSD style license - see LICENSE.rst
from astropy.coordinates import Angle, SkyCoord
from regions.core.attributes import QuantityLength
from ..core import SkyRegion, PixelRegion, PixCoord

__all__ = ["RangePixelRegion", "RangeSphericalRegion"]

inf = Angle(float("inf"), "deg")


# TODO: needed?
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think this is needed. Isn't it covered exactly by the existing planar rectangle as long as you choose the right parameterization? It might be thought of as a "convenience function". Maybe an additional constructor / factory function for the planar rectangle?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think there's a few small advantages to introducingRangePixelRegion (instead of using RectanglePixelRegion):

  • Clear code, symmetry, being able to round-trip when doing to_pixel and from_pixel isntead of mixing over to the rectangle classes
  • I'm not sure how the +- inf case would be handled. I don't think one can handle the min=-inf, max=42 case, there's no center that will result in a rectangle extending that range?
  • It's not a lot of work (30 lines of code?)

I'm still not sure if it's worth adding, but my feeling is that it might be the best way to go to add it?

@gpdf - Thoughts?

If no addition, what exactly whould be the alternative?
Should RangeSphericalRegion not offer to_pixel at all, or should it go to RectanglePixelRegion?

class RangePixelRegion(PixelRegion):
"""
A range in pixel coordinates.

See `RangeSphericalRegion`.
"""


class RangeSphericalRegion(SkyRegion):
"""
A range in longitude and latitude.

Use case: http://www.ivoa.net/documents/SIA/20151223/REC-SIA-2.0-20151223.html#toc12

Parameters
----------
lon_min, lon_max : `~astropy.coordinates.Angle`
Range in longitude
lat_min, lat_max : `~astropy.coordinates.Angle`
Range in latitude
frame : `~astropy.coordinates.BaseCoordinateFrame`
Sky frame
meta : `~regions.RegionMeta` object, optional
A dictionary which stores the meta attributes of this region.
visual : `~regions.RegionVisual` object, optional
A dictionary which stores the visual meta attributes of this region.
"""
# TODO: should we use these properties, or SkyCoord for min / max,
# or a center-based representation?
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please don't use a center-based representation. That way lies the madness of ADQL's ill-fated BOX.

A true range-based representation is fundamentally not relocatable in latitude - its shape is inherently different for different latitudes.

A geometrically natural center-based representation needs to be fully relocatable - it should have the same shape anywhere on the sphere as long as its size parameters stay the same. (It's a coordinate-free geometrical concept.)

But it's so likely to be confused with the coordinate-range region that people have a lot of trouble with it. BOX is being proposed for withdrawal from ADQL because it caused so much trouble of this kind.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK, I'll leave the current representation then, as four angles and a separate frame attribute.

_params = ('lon_min', 'lon_max', 'lat_min', 'lat_max')
lon_min = QuantityLength('lon_min')
lon_max = QuantityLength('lon_max')
lat_min = QuantityLength('lat_min')
lat_max = QuantityLength('lat_max')

def __init__(self, lon_min=-inf, lon_max=inf, lat_min=-inf, lat_max=inf, frame="icrs", meta=None, visual=None):
self.lon_min = lon_min
self.lon_max = lon_max
self.lat_min = lat_min
self.lat_max = lat_max
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Python supports IEEE floating point -Inf and +Inf, so it should be possible, with care, to map these ranges exactly onto their IVOA SIAv2/SODA interpretation. It may require a bit of thinking to handle the edge case of a region that actually covers the entire sky.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK, I'll add test cases.

# TODO: Is there a helper function to make a frame without a
self.frame = SkyCoord(0, 0, unit="deg", frame=frame).frame
self.meta = meta or {}
self.visual = visual or {}

def to_pixel(self, wcs):
sky_min = SkyCoord(self.lon_min, self.lat_min, frame=self.frame)
sky_max = SkyCoord(self.lon_max, self.lat_max, frame=self.frame)

pix_min = PixCoord.from_sky(sky_min, self.wcs)
pix_max = PixCoord.from_sky(sky_max, self.wcs)
return RangePixelRegion(pix_min.x, pix_max.x, pix_min.y, pix_max.y, self.meta, self.visual)

def contains(self, skycoord, wcs=None):
skycoord = skycoord.transform_to(self.frame)
return (
(self.lon_min <= skycoord.data.lon) &
(self.lon_max >= skycoord.data.lon) &
(self.lat_min <= skycoord.data.lat) &
(self.lat_max >= skycoord.data.lat)
)
13 changes: 12 additions & 1 deletion regions/shapes/tests/test_circle.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@

from ...core import PixCoord
from ...tests.helpers import make_simple_wcs
from ..circle import CirclePixelRegion, CircleSkyRegion
from ..circle import CirclePixelRegion, CircleSkyRegion, CircleSphericalRegion
from .utils import HAS_MATPLOTLIB # noqa
from .test_common import BaseTestPixelRegion, BaseTestSkyRegion

Expand Down Expand Up @@ -101,3 +101,14 @@ def test_contains(self, wcs):
position = SkyCoord([1, 3] * u.deg, [2, 4] * u.deg)
# 1,2 is outside, 3,4 is the center and is inside
assert all(self.reg.contains(position, wcs) == np.array([False, True], dtype='bool'))

class TestCircleSphericalRegion:
@staticmethod
def test_contains(wcs):
# TODO: pick a wcs and point where sky and spherical differs
reg_sky = CircleSkyRegion(SkyCoord(0 * u.deg, 80 * u.deg), radius=12 * u.deg)
reg_sph = CircleSphericalRegion(SkyCoord(0 * u.deg, 80 * u.deg), radius=12 * u.deg)
point = SkyCoord(180 * u.deg, 89 * u.deg)
assert reg_sky.contains(point, wcs)
assert reg_sph.contains(point, wcs)
assert reg_sph.contains(point)
23 changes: 23 additions & 0 deletions regions/shapes/tests/test_range.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
# Licensed under a 3-clause BSD style license - see LICENSE.rst
from numpy.testing import assert_equal
import astropy.units as u
from astropy.coordinates import SkyCoord
from ..range import RangeSphericalRegion

# TODO: use test cases with finite & infinite values from here:
# http://www.ivoa.net/documents/SIA/20151223/REC-SIA-2.0-20151223.html#toc12

def test_range_default():
region = RangeSphericalRegion()
assert region.contains(SkyCoord("0d 0d"))

def test_range_finite():
region = RangeSphericalRegion(12 * u.deg, 12.5 * u.deg, 34 * u.deg, 36.0 * u.deg)
assert region.contains(SkyCoord("12d 34d"))
assert not region.contains(SkyCoord("11d 34d"))
assert not region.contains(SkyCoord("13d 34d"))
assert not region.contains(SkyCoord("12d 33d"))
assert not region.contains(SkyCoord("12d 37d"))

mask = region.contains(SkyCoord([12, 13] * u.deg, [34, 34] * u.deg))
assert_equal(mask, [True, False])