Skip to content

Commit

Permalink
#110 rename kwarg and use dict for election
Browse files Browse the repository at this point in the history
  • Loading branch information
debboutr committed Apr 7, 2020
1 parent 836b33c commit 3dba26c
Showing 1 changed file with 29 additions and 25 deletions.
54 changes: 29 additions & 25 deletions pysheds/grid.py
Original file line number Diff line number Diff line change
Expand Up @@ -550,7 +550,7 @@ def resize(self, data, new_shape, out_suffix='_resized', inplace=True,
return self._output_handler(data=data, out_name=out_name, properties=grid_props,
inplace=inplace, metadata=metadata)

def nearest_cell(self, x, y, affine=None, reside=False):
def nearest_cell(self, x, y, affine=None, snap='corner'):
"""
Returns the index of the cell (column, row) closest
to a given geographical coordinate.
Expand All @@ -565,11 +565,11 @@ def nearest_cell(self, x, y, affine=None, reside=False):
Affine transformation that defines the translation between
geographic x/y coordinate and array row/column coordinate.
Defaults to self.affine.
reside : bool
Indicates the cell indexing method. If False, will resolve to
snapping the (x,y) geometry to the index of the nearest top-left
cell corner. If True, will return the index of the cell that the
geometry resides.
snap : str
Indicates the cell indexing method. If "corner", will resolve to
snapping the (x,y) geometry to the index of the nearest top-left
cell corner. If "center", will return the index of the cell that
the geometry falls within.
Returns
-------
x_i, y_i : tuple of ints
Expand All @@ -581,8 +581,8 @@ def nearest_cell(self, x, y, affine=None, reside=False):
assert isinstance(affine, Affine)
except:
raise TypeError('affine must be an Affine instance.')
index_method = np.floor if reside else np.around
col, row = index_method(~affine * (x, y)).astype(int)
snap_dict = {'corner': np.around, 'center': np.floor}
col, row = snap_dict[snap](~affine * (x, y)).astype(int)
return col, row

def set_bbox(self, new_bbox):
Expand Down Expand Up @@ -843,7 +843,7 @@ def facet_flow(self, e0, e1, e2, d1=1, d2=1):
def catchment(self, x, y, data, pour_value=None, out_name='catch', dirmap=None,
nodata_in=None, nodata_out=0, xytype='index', routing='d8',
recursionlimit=15000, inplace=True, apply_mask=False, ignore_metadata=False,
reside=False, **kwargs):
snap='corner', **kwargs):
"""
Delineates a watershed from a given pour point (x, y).
Expand Down Expand Up @@ -890,8 +890,10 @@ def catchment(self, x, y, data, pour_value=None, out_name='catch', dirmap=None,
If True, "mask" the output using self.mask.
ignore_metadata : bool
If False, require a valid affine transform and crs.
reside : bool
If True, index point geometry to it's residing cell.
snap : str
Function to use on array for indexing:
'corner' : numpy.around()
'center' : numpy.floor()
"""
# TODO: Why does this use set_dirmap but flowdir doesn't?
dirmap = self._set_dirmap(dirmap, data)
Expand All @@ -917,7 +919,7 @@ def catchment(self, x, y, data, pour_value=None, out_name='catch', dirmap=None,
dirmap=dirmap, nodata_in=nodata_in, nodata_out=nodata_out,
xytype=xytype, recursionlimit=recursionlimit, inplace=inplace,
apply_mask=apply_mask, ignore_metadata=ignore_metadata,
properties=properties, metadata=metadata, reside=reside, **kwargs)
properties=properties, metadata=metadata, snap=snap, **kwargs)
elif routing.lower() == 'dinf':
return self._dinf_catchment(x, y, fdir=fdir, pour_value=pour_value, out_name=out_name,
dirmap=dirmap, nodata_in=nodata_in, nodata_out=nodata_out,
Expand All @@ -928,7 +930,7 @@ def catchment(self, x, y, data, pour_value=None, out_name='catch', dirmap=None,
def _d8_catchment(self, x, y, fdir=None, pour_value=None, out_name='catch', dirmap=None,
nodata_in=None, nodata_out=0, xytype='index', recursionlimit=15000,
inplace=True, apply_mask=False, ignore_metadata=False, properties={},
metadata={}, reside=False, **kwargs):
metadata={}, snap='corner', **kwargs):

# Vectorized Recursive algorithm:
# for each cell j, recursively search through grid to determine
Expand All @@ -951,7 +953,7 @@ def d8_catchment_search(cells):
# to given geographic coordinate
# Valid if the dataset is a view.
if xytype == 'label':
x, y = self.nearest_cell(x, y, fdir.affine, reside)
x, y = self.nearest_cell(x, y, fdir.affine, snap)
# get the flattened index of the pour point
pour_point = np.ravel_multi_index(np.array([y, x]),
fdir.shape)
Expand Down Expand Up @@ -985,7 +987,7 @@ def d8_catchment_search(cells):
def _dinf_catchment(self, x, y, fdir=None, pour_value=None, out_name='catch', dirmap=None,
nodata_in=None, nodata_out=0, xytype='index', recursionlimit=15000,
inplace=True, apply_mask=False, ignore_metadata=False, properties={},
metadata={}, reside=False, **kwargs):
metadata={}, snap='corner', **kwargs):
# Filter warnings due to invalid values
np.warnings.filterwarnings(action='ignore', message='Invalid value encountered',
category=RuntimeWarning)
Expand Down Expand Up @@ -1036,7 +1038,7 @@ def dinf_catchment_search(cells):
# TODO: This relies on the bbox of the grid instance, not the dataset
# Valid if the dataset is a view.
if xytype == 'label':
x, y = self.nearest_cell(x, y, fdir.affine, reside)
x, y = self.nearest_cell(x, y, fdir.affine, snap)
# get the flattened index of the pour point
pour_point = np.ravel_multi_index(np.array([y, x]),
fdir.shape)
Expand Down Expand Up @@ -1504,7 +1506,7 @@ def _remove_dinf_cycles(self, fdir_0, fdir_1, startnodes, max_cycles=2):
def flow_distance(self, x, y, data, weights=None, dirmap=None, nodata_in=None,
nodata_out=0, out_name='dist', routing='d8', method='shortest',
inplace=True, xytype='index', apply_mask=True, ignore_metadata=False,
reside=False, **kwargs):
snap='corner', **kwargs):
"""
Generates an array representing the topological distance from each cell
to the outlet.
Expand Down Expand Up @@ -1548,8 +1550,10 @@ def flow_distance(self, x, y, data, weights=None, dirmap=None, nodata_in=None,
If True, "mask" the output using self.mask.
ignore_metadata : bool
If False, require a valid affine transform and CRS.
reside : bool
If True, index point geometry to it's residing cell.
snap : str
Function to use on array for indexing:
'corner' : numpy.around()
'center' : numpy.floor()
"""
if not _HAS_SCIPY:
raise ImportError('flow_distance requires scipy.sparse module')
Expand All @@ -1576,20 +1580,20 @@ def flow_distance(self, x, y, data, weights=None, dirmap=None, nodata_in=None,
xytype=xytype, apply_mask=apply_mask,
ignore_metadata=ignore_metadata,
properties=properties, metadata=metadata,
reside=reside, **kwargs)
snap=snap, **kwargs)
elif routing.lower() == 'dinf':
return self._dinf_flow_distance(x, y, fdir, weights=weights, dirmap=dirmap,
nodata_in=nodata_in, nodata_out=nodata_out,
out_name=out_name, method=method, inplace=inplace,
xytype=xytype, apply_mask=apply_mask,
ignore_metadata=ignore_metadata,
properties=properties, metadata=metadata,
reside=reside, **kwargs)
snap=snap, **kwargs)

def _d8_flow_distance(self, x, y, fdir, weights=None, dirmap=None, nodata_in=None,
nodata_out=0, out_name='dist', method='shortest', inplace=True,
xytype='index', apply_mask=True, ignore_metadata=False, properties={},
metadata={}, reside=False, **kwargs):
metadata={}, snap='corner', **kwargs):
# Construct flat index onto flow direction array
domain = np.arange(fdir.size)
fdir_orig_type = fdir.dtype
Expand All @@ -1607,7 +1611,7 @@ def _d8_flow_distance(self, x, y, fdir, weights=None, dirmap=None, nodata_in=Non
startnodes, endnodes = self._construct_matching(fdir, domain,
dirmap=dirmap)
if xytype == 'label':
x, y = self.nearest_cell(x, y, fdir.affine, reside)
x, y = self.nearest_cell(x, y, fdir.affine, snap)
# TODO: Currently the size of weights is hard to understand
if weights is not None:
weights = weights.ravel()
Expand Down Expand Up @@ -1637,7 +1641,7 @@ def _d8_flow_distance(self, x, y, fdir, weights=None, dirmap=None, nodata_in=Non
def _dinf_flow_distance(self, x, y, fdir, weights=None, dirmap=None, nodata_in=None,
nodata_out=0, out_name='dist', method='shortest', inplace=True,
xytype='index', apply_mask=True, ignore_metadata=False,
properties={}, metadata={}, reside=False, **kwargs):
properties={}, metadata={}, snap='corner', **kwargs):
# Filter warnings due to invalid values
np.warnings.filterwarnings(action='ignore', message='Invalid value encountered',
category=RuntimeWarning)
Expand Down Expand Up @@ -1670,7 +1674,7 @@ def _dinf_flow_distance(self, x, y, fdir, weights=None, dirmap=None, nodata_in=N
assert(startnodes.size == endnodes_0.size)
assert(startnodes.size == endnodes_1.size)
if xytype == 'label':
x, y = self.nearest_cell(x, y, fdir.affine, reside)
x, y = self.nearest_cell(x, y, fdir.affine, snap)
# TODO: Currently the size of weights is hard to understand
if weights is not None:
if isinstance(weights, list) or isinstance(weights, tuple):
Expand Down

0 comments on commit 3dba26c

Please sign in to comment.