-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathtmd_InferMinor.m
169 lines (146 loc) · 6.34 KB
/
tmd_InferMinor.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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
function dh = tmd_InferMinor(hc,constituents,t,s,h,p,N)
% tmd_InferMinor returns correction for 16 minor tidal constituents. Zeros
% are returned if not enough input for inference. This is based on Richard
% Ray's code perth2. For further description, see Ray, Richard D. "Global
% ocean tide models on the eve of TOPEX/POSEIDON." IEEE transactions on
% Geoscience and Remote Sensing 31.2 (1993): 355-364.
%
%% Syntax
%
% dh = tmd_InferMinor(hc,constituents,t,s,h,p,N)
%
%% Description
%
% dh = tmd_InferMinor(hc,constituents,t,s,h,p,N) returns the correction dh given
% complex hc for given constituents are in cell format,
% t is serial date in Matlab's datenum or datetime format. The variables
% s,h,p,N are the outputs of tmd_astrol, and must have the same dimensions as t.
%
%% Function history
% This function was written in Matlab by Lana Erofeeva, Oct 2004, based on
% Richard Ray's perth2 code.
%
% In June 2022 for TMD3.0, Chad Greene made the following changes:
% * Added documentation in function header.
% * Removed unused variables.
% * Replaced summation loop with a simple call to sum.
% * Replaced function return with function end.
% * Replaced if-statement error checks with assertions.
% * Added input checks.
% * Replaced nested loops previously used to create z8 with ismember.
% * Changed the inputs for speed and readability.
% * Now ignores any minor constituents that are in the list of major constituents.
%
% See also tmd_predict and tmd_astrol.
%% Initial error checks:
narginchk(7,7)
assert(isnumeric(hc)==1,'InferMinor input error: hc must be numeric.')
assert(iscell(constituents),'InferMinor input error: cid must be a cell array of constituents.')
assert(isequal(size(t),size(s),size(h),size(p),size(N)),'Time and astrol inputs must all have the same dimensions.')
%% Check sizes:
M = size(hc,2);
T = numel(t);
if T>1
assert(M==1 | M==T,'Error: If neither input times nor location are scalar, they must be vectors of the same length.')
M=T;
end
%% Create data based on input dimensions:
if isdatetime(t)
t = datenum(t);
end
t_1992 = t(:)-datenum(1992,1,1); % time (days) relatively Jan 1 1992 GMT corresponds to 48622mjd
t_hr = (t_1992 - floor(t_1992))*24;
t1 = 15.*t_hr;
t2 = 30.*t_hr;
cid8 = {'q1';'o1';'p1';'k1';'n2';'m2';'s2';'k2'};
% Reorder zmaj to correspond to cid8:
[Lia,Lib] = ismember(cid8,constituents);
z8 = hc(Lib,:);
assert(sum(Lia)>5,'Not enough constituents for inference.')
% Names of all 18 minor cons (some are repeated)
minor_cons = {'2q1';'sigma1';'rho1';'m1';'m1';'chi1';'pi1';'phi1';'theta1';...
'j1';'oo1';'2n2';'mu2';'nu2';'lambda2';'l2';'l2';'t2'};
% Get the indices of all the minor constituents that are in minor_cons and
% also in the tide model. The ~tilde below ensures we only solve the minor
% cons that aren't in the model, because we don't want to double count them.
ind = ~ismember(minor_cons,constituents);
%% Solve constants:
rad=pi/180;
PP=282.8;
zmin=zeros(18,M);
zmin(1,:) = 0.263 *z8(1,:) - 0.0252*z8(2,:); %2Q1
zmin(2,:) = 0.297 *z8(1,:) - 0.0264*z8(2,:); %sigma1
zmin(3,:) = 0.164 *z8(1,:) + 0.0048*z8(2,:); %rho1 +
zmin(4,:) = 0.0140*z8(2,:) + 0.0101*z8(4,:); %M1
zmin(5,:) = 0.0389*z8(2,:) + 0.0282*z8(4,:); %M1
zmin(6,:) = 0.0064*z8(2,:) + 0.0060*z8(4,:); %chi1
zmin(7,:) = 0.0030*z8(2,:) + 0.0171*z8(4,:); %pi1
zmin(8,:) =-0.0015*z8(2,:) + 0.0152*z8(4,:); %phi1
zmin(9,:) =-0.0065*z8(2,:) + 0.0155*z8(4,:); %theta1
zmin(10,:) =-0.0389*z8(2,:) + 0.0836*z8(4,:); %J1 +
zmin(11,:) =-0.0431*z8(2,:) + 0.0613*z8(4,:); %OO1 +
zmin(12,:) = 0.264 *z8(5,:) - 0.0253*z8(6,:); %2N2 +
zmin(13,:) = 0.298 *z8(5,:) - 0.0264*z8(6,:); %mu2 +
zmin(14,:) = 0.165 *z8(5,:) + 0.00487*z8(6,:); %nu2 +
zmin(15,:) = 0.0040*z8(6,:) + 0.0074*z8(7,:); %lambda2
zmin(16,:) = 0.0131*z8(6,:) + 0.0326*z8(7,:); %L2 +
zmin(17,:) = 0.0033*z8(6,:) + 0.0082*z8(7,:); %L2 +
zmin(18,:) = 0.0585*z8(7,:); %t2 +
arg=zeros(18,M);
arg(1,:) = t1 - 4.*s + h + 2.*p - 90; % 2Q1
arg(2,:) = t1 - 4.*s + 3.*h - 90; % sigma1
arg(3,:) = t1 - 3.*s + 3.*h - p - 90; % rho1
arg(4,:) = t1 - s + h - p + 90; % M1
arg(5,:) = t1 - s + h + p + 90; % M1
arg(6,:) = t1 - s + 3.*h - p + 90; % chi1
arg(7,:) = t1 - 2.*h + PP - 90; % pi1
arg(8,:) = t1 + 3.*h + 90; % phi1
arg(9,:) = t1 + s - h + p + 90; % theta1
arg(10,:) = t1 + s + h - p + 90; % J1
arg(11,:) = t1 + 2.*s + h + 90; % OO1
arg(12,:) = t2 - 4.*s + 2.*h + 2.*p; % 2N2
arg(13,:) = t2 - 4.*s + 4.*h; % mu2
arg(14,:) = t2 - 3.*s + 4.*h - p; % nu2
arg(15,:) = t2 - s + p + 180; % lambda2
arg(16,:) = t2 - s + 2.*h - p + 180; % L2
arg(17,:) = t2 - s + 2.*h + p; % L2
arg(18,:) = t2 - h + PP; % t2
% determine nodal corrections f and u:
sinn = sin(N*rad);
cosn = cos(N*rad);
sin2n = sin(2.*N*rad);
cos2n = cos(2.*N*rad);
f = ones(18,M);
f(1,:) = hypot(1.0 + 0.189*cosn - 0.0058*cos2n, 0.189*sinn - 0.0058*sin2n);
f(2,:) = f(1,:);
f(3,:) = f(1,:); % was mistakenly f(3,:) = f(1); in 2.04
f(4,:) = hypot(1.0 + 0.185*cosn, 0.185*sinn);
f(5,:) = hypot(1.0 + 0.201*cosn, 0.201*sinn);
f(6,:) = hypot(1.0 + 0.221*cosn, 0.221*sinn);
f(10,:) = hypot(1.0 + 0.198*cosn, 0.198*sinn);
f(11,:) = hypot(1.0 + 0.640*cosn + 0.134*cos2n,0.640*sinn + 0.134*sin2n);
f(12,:) = hypot(1.0 - 0.0373*cosn,0.0373*sinn);
f(13,:) = f(12,:);
f(14,:) = f(12,:);
f(16,:) = f(12,:);
f(17,:) = hypot(1.0 + 0.441*cosn,0.441*sinn);
u = zeros(18,M);
u(1,:) = atan2(0.189*sinn - 0.0058*sin2n,...
1.0 + 0.189*cosn - 0.0058*sin2n)/rad;
u(2,:) = u(1,:);
u(3,:) = u(1,:);
u(4,:) = atan2( 0.185*sinn, 1.0 + 0.185*cosn)/rad;
u(5,:) = atan2(-0.201*sinn, 1.0 + 0.201*cosn)/rad;
u(6,:) = atan2(-0.221*sinn, 1.0 + 0.221*cosn)/rad;
u(10,:) = atan2(-0.198*sinn, 1.0 + 0.198*cosn)/rad;
u(11,:) = atan2(-0.640*sinn - 0.134*sin2n,...
1.0 + 0.640*cosn + 0.134*cos2n)/rad;
u(12,:) = atan2(-0.0373*sinn, 1.0 - 0.0373*cosn)/rad;
u(13,:) = u(12,:);
u(14,:) = u(12,:);
u(16,:) = u(12,:);
u(17,:) = atan2(-0.441*sinn, 1.0 + 0.441*cosn)/rad;
%% Sum over all tides:
tmp = (arg(ind,:)+u(ind,:))*rad; % This tmp variable prevents performing the same operation twice on the next line.
dh = permute(sum(real(zmin(ind,:)).*f(ind,:).*cos(tmp) + imag(zmin(ind,:)).*f(ind,:).*sin(tmp)),[2 3 1]);
end