-
Notifications
You must be signed in to change notification settings - Fork 31
/
Copy pathdefstruct.m
104 lines (91 loc) · 3.36 KB
/
defstruct.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
function defstruct(name,fields,values)
% DEFSTRUCT(name,fields,values)
%
% Assigns default values to a named structure
%
% INPUT:
%
% name A string, enclosed in single quotes, with a structure name
% fields A cell array with fields that you want the structure to have
% values A cell array with the values by which you populate those fields
%
% OUTPUT:
%
% None. The variable appears as if by magic into your workspace or
% will be available inside your function.
%
% SEE ALSO:
%
% CELLNAN, STRUCTNAN, STRUCT
%
% Last modified by fjsimons-at-alum.mit.edu, 08/16/2021
if ~ischar(name),
error(sprintf(['The first argument of DEFSTRUCT',...
'has to be a string with a structure name']));
end
% Always do it is our default here
si=1;
% Maybe it's only a single field name and value pair
if isstr(fields);
fields={fields};
end
if ~iscell(values);
values={values};
end
% Check input lengths
difer(length(values)-length(fields),[],1,NaN)
% If the STRUCTURE exists at all...
if evalin('caller',[ 'exist(''' name ''',''var'')'])
% ... and it's empty, do it; but don't do it if it's non empty
% This as the whole structure variable, not any piece of it
si=evalin('caller',[ 'isempty(' name ')']);
% Now if the STRUCTURE is not empty, SOME of it may be empty
if ~si && evalin('caller',[ 'isstruct(' name ')']);
% Which of the fieldnames exist?
fn=evalin('caller',['fieldnames(' name ')'])';
% For now they all exist
isv=logical(zeros(1,length(fn)));
% But now we know which ones are the empty ones
for index=1:length(fn)
isv(index)=evalin('caller',['isempty(' name '.' fn{index} ')']);
end
% Then there are the fields that did exist and were nonempty
oldvalues=cellnan(1,1,sum(~isv)); ondex=0;
for index=find(~isv)
ondex=ondex+1;
oldvalues{ondex}=evalin('caller',[ name '.' fn{index} ]);
end
% Variables that did exist but were empty for which we have values
[ifn,ivs]=intersect(fields,fn(isv),'stable');
% Variables that existed but were empty for which we don't have news
bfn=setdiff(fn(isv),ifn);
% Variables that didn't exist for which we have values
[newfields,ivf]=setdiff(fields,fn,'stable');
% Combine the new fields, the old values, and the new values and
% remember that where we have no information the default is [] anyhow
% Remember the difference between smooth and curly parentheses!
fields={newfields{:} fn{~isv} ifn{:} bfn{:}};
values={values{ivf} oldvalues{:} values{ivs} };
% So in the end you always have to do something!
si=1;
end
end
% Do it or not, either the whole thing or just the pieces of it
if si
% Do it exactly like OPTIMSET
structinput=cell(2,length(fields));
% Fields go in first row
structinput(1,:)=fields;
% []'s go in second row, remember there may be hollow empties at the end
structinput(2,1:length(values))=values;
% Turn it into correctly ordered comma separated list and call struct
S=struct(structinput{:});
% Order output. NOT. Usually, we expect a certain ordering!
% S=orderfields(S);
% Put in the caller workspace
assignin('caller',name,S);
end
% We may not need to build the entire structure and then (re)assign it -
% we might just build the updates and merge it in the workspace? That
% might be more efficient if there is a large number of knowns and a
% small number of updates