forked from amforte/Topographic-Analysis-Kit
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRemoveFlats.m
127 lines (110 loc) · 3.46 KB
/
RemoveFlats.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
function [DEMn,MASK]=RemoveFlats(dem,strength)
%
% Usage:
% [DEMn,MASK]=RemoveFlats(dem,strength);
%
% Description:
% Function takes DEM and attempts a semi-automated routine to remove flat areas with some input from
% the user to select areas considred to be flat. This function sometimes works reliably, but will
% never produce as clean a result as manually clipping out flat areas in gis software (but it's
% a lot faster!)
%
% Required Inputs:
% dem - either full path of dem file as either an ascii text file or geotiff OR
% the name of a GRIDobj of a DEM stored in the workspace
% strength - integer value between 1 and 4 that controls how aggressively the function defines
% flat areas, specifically related to the size of the neighborhood the function uses to
% connect ares of similar elevation. A strength of 1 = a 3x3 neighborhood, 2=5x5, 3=7x7, and
% 4=9x9. If the results of the function do not capture enough of the flat areas in the MASK,
% increase the strength and rerun. Similarly, if the function erroneously includes areas that
% are not part of what you consider the flats, try decreasing the strength.
%
% Outputs:
% DEMn - Version of the DEM with idenitifed flat areas masked out (values set to nan)
% MASK - Logical GRIDobj, true where area was identified as a flat.
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
% Function Written by Adam M. Forte - Updated : 06/18/18 %
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
% Parse Inputs
p = inputParser;
p.FunctionName = 'RemoveFlats';
addRequired(p,'dem',@(x) isa(x,'GRIDobj') | ischar(x));
addRequired(p,'strength',@(x) isnumeric(x) && x<=4 && mod(x,1)==0);
parse(p,dem,strength);
dem=p.Results.dem;
strength=p.Results.strength;
disp('Preparing Grids...')
% Define kernels
nhood1=ones(3,3);
if strength==1
nhood2=ones(3,3);
elseif strength==2
nhood2=ones(5,5);
elseif strength==3
nhood2=ones(7,7);
elseif strength==4
nhood2=ones(9,9);
else
if isdeployed
errordlg('Input to "strength" is not recognized, must be an integer between 1 and 4')
end
error('Input to "strength" is not recognized, must be an integer between 1 and 4')
end
% Check type of input
if isa(dem,'GRIDobj');
DEM=dem;
elseif ischar(dem);
disp('Loading DEM')
DEM=GRIDobj(dem);
else
if isdeployed
errordlg('Input for dem not recognized as either a GRIDobj or character')
end
error('Input for dem not recognized as either a GRIDobj or character')
end
% Fill sinks
[DEMs]=fillsinks(DEM);
% Identify flats
FL=GRIDobj(DEMs);
FL=erode(DEMs,nhood1)==DEMs;
CFL=dilate(FL,nhood2);
% Label flats
L=bwlabel(CFL.Z);
[xdim,ydim]=getcoordinates(DEM);
L=GRIDobj(xdim,ydim,L);
% Generate gradient for visualizing
G=gradient8(DEM);
% Prompt user to choose sinks
f1=figure(1);
hold on
title('Gradient of DEM: Select areas that you consider sinks and then press enter')
imageschs(DEM,G,'colormap','jet','caxis',[0 1]);
if ~verLessThan('matlab','9.5')
disableDefaultInteractivity(gca);
end
hold off
[x,y]=ginput;
close(f1)
% Find label for flats
[ix]=coord2ind(DEM,x,y);
l=L.Z(ix);
l=unique(l);
% Generate mask
MASK=GRIDobj(DEM);
for ii=1:numel(l)
MASK.Z(L.Z==l(ii))=1;
end
MASK.Z=logical(MASK.Z);
% Produce new DEM
DEMn=DEM;
DEMn.Z(MASK.Z)=nan;
% Generate output to test results
f1=figure(1);
hold on
imageschs(DEMn,MASK);
if ~verLessThan('matlab','9.5')
disableDefaultInteractivity(gca);
end
hold off
end