-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy pathTEgroup_stats.m
341 lines (296 loc) · 14.4 KB
/
TEgroup_stats.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
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
function TEgroup_stats(cfg,filesTEpermtest)
% TEGROUP_STATS: The function uses a permutation test to compare TE values
% between two groups of subjects or between two conditions. TEgroup_prepare
% has to be run on all data sets and TE values have to be estimated using
% InteractionDelayReconstruction_calculate (or
% TEsurrogatestats/_ensemble.m).
%
% !!!!!!!! DATA HAS TO BE PREPARED AND ANALYZED FOR GROUP STATICIS !!!!!!!!
%
%
%
% * REFERENCE INFORMATION
%
% - transfer entropy
% - The concept of TE appears in Schreiber's article,
% "Measuring Information Transfer", Phys. Rev. Lett. 85, 461 - 464
% (2000).
% - For the estimation of probability densities needed for the TE
% computation, the function implements the Kraskov-Stoegbauer-
% Grassberger estimator described in Kraskov et al. "Estimating
% mutual information", Phys. Rev. E 69 (6) 066138, (2004).
%
% - permutation test
% - Maris & Oostenveld (2007). Nonparametric statistical testing of
% EEG- and MEG-data. J. of Neuroscience Methods, 164, 177-190.
%
%
% * DEPENDENCIES
% - Package TSTOOL is used at nearest neighbors searches
% required for the KSG estimator. (Gnu Public License)
% http://www.dpi.physik.uni-goettingen.de/tstool/
% - The following Matlab toolboxes:
% - statistic toolbox
% - The functions
% - transferentropy
% - TEcmc
% - TEperm
% - TEvalues
% - TEwait
% - TEconsoleoutput
%
%
% * INPUT PARAMETERS
%
% filesTEpermtest = cell array of strings containing files names of data
% sets for group statistics (this assumes the function
% is called from the folder containing the files); OR
% full paths to data sets (function can be called from
% anywhere). Note that the order of the files has to
% match the order of entries in the design matrix
% (along the 2nd dimension)
%
% cfg.design = matrix containing a row with subject number and a row
% with independent variable representing the order of
% the data input.
% example:
% datasets: 1 2 3 4 5 1 2 3 4 5
% conditions: 1 1 1 1 1 2 2 2 2 2
% cfg.uvar = row in cfg.design which contains the dataset number
% (in the example: 1)
% cfg.ivar = row in cfg.design which contains the independent
% variable (in the example: 2)
% cfg.rawvalues = use raw TE values for statistic (default=1), if set
% to 0, distance between raw TE and mean surrogate TE
% value is used (4th column in TEpermtest table)
% cfg.alpha = significance level for statistical shift test,
% permutation test and correction for multiple
% comparison (default = 0.05)
% cfg.numpermutation = nr of permutations in permutation test
% (default = 190100)
% cfg.permstatstype = 'mean' to use the distribution of the mean
% differences and 'depsamplesT' or
% 'indepsamplesT' for distribution of the
% t-values. (default = 'mean')
% cfg.tail = '1' tail or '2' tailed test of significance (for the
% permutation tests) (default = 2)
% cfg.correctm = correction method used for correction of the multiple
% comparison problem - False discovery rate 'FDR' or
% Bonferroni correction 'BONF' (default = 'FDR')
% cfg.fileidout = string for the output path and the first part of the
% output filename. TEgroup_stats will write two files
% with suffix 'TE_output.mat' (containing the raw
% values used in the statistical test) and
% 'TEpermtestgroup_output.mat' (containing the output
% of the statistical test).
%
%
% * OUTPUT PARAMETERS (ouput is saved to the location specified in
% cfg.fileidout)
%
% TEpermtestgroup (saved in file '*TE_output.mat')
% .TEpermvalues = matrix with size:
% (channelpair,value)
% The last dimension "value" includes:
% 1 - p_values of the statistic within the
% distribution given by the permutations
% 2 - 1 (0), if the statistics is significant at
% the prescribed alpha level (or not)
% 3 - 1 (0), if the statistics is significant
% after correction for multiple comparisons
% (or not)
% 4 - test statistic mean difference or tvalue
% of mean difference depending on
% cfg.permstatstype
% 5 - 1 (0), if instantaneous mixing/volume
% conduction exists (or not)
% 6 - mean estimated interaction delay u from the
% group with the larger TE value
%
% .nr2cmc = number used for correction for multiple
% comparisons (returned by TEcmc)
% .correctm = method used for correction for multiple
% comparisons (returned by TEcmc)
% .dimord = dimensions of TEpermvalues
% .cfg = configuration file used for group statistics
% .sgncmb = labels of channel combinations (source ->
% target)
% .numpermutation = number of permutations
% .nrdatasets = number of datasets that entered the analysis
% .groupprepare = results of the function TEgroup_prepare
% from the data
%
%
% Raw data used for statistical testing (saved in file '*TE_output.mat')
% TEresult1 = raw TE values for group/condition 1
% TEresult2 = raw TE values for group/condition 2
% TEresultmean = mean over raw values per group/condition
%
%
% This program is free software; you can redistribute it and/or modify
% it under the terms of the GNU General Public License as published by
% the Free Software Foundation;
%
% This program is distributed in the hope that it will be useful,
% but WITHOUT ANY WARRANTY;
%
% Version 1.0 by Michael Lindner, Raul Vicente, Michael Wibral
% Frankfurt 2010
%
% CHANGELOG:
% 2013-06-25 PW: changed ival/uval to ivar/uvar in the function's help
% 2013-06-25 PW: nrchannels was not written into cfg structure, but is used
% later to construct the output structures
% 2013-10-20 PW: TEpermtestgroup.TEpermvalues now returns the mean interaction
% delay u for channel combinations with significant differences (mean value
% from the group with the higher TE value)
% 2015-11-05 PW: function now handles unordered data correctly (relevant
% for dependent samples t-test, assures that corresponding data sets are
% substracted from each other)
% 2015-11-10 PW: allow user to switch between raw TE values and surrogate-
% corrected TE values for test
%% define logging levels
LOG_INFO_MAJOR = 1;
LOG_INFO_MINOR = 2;
%% load Data
% -------------------------------------------------------------------------
nrdatasets = length(filesTEpermtest);
cfg.maxtrials = 0; % get the max. no. trials in any data set
allTEpermtest = cell(1,nrdatasets); % holds all data for stat test
u_values = [];
for ll = 1:nrdatasets
% load files containing 'TEpermteset'
varinfile = who('-file',filesTEpermtest{ll});
load(filesTEpermtest{ll},varinfile{1});
allTEpermtest{ll} = eval(varinfile{1});
% check data for TEprepare/TEproupprepare structure
if ~isfield(allTEpermtest{ll}, 'groupprepare')
error(strcat('TRENTOOL ERROR: Field .groupprepare is missing in dataset ',filesTEpermtest{ll}));
end
% check code
if allTEpermtest{1}.groupprepare.code ~= allTEpermtest{ll}.groupprepare.code
error('TRENTOOL ERROR: Only datasets, which were prepared together with TEgroup_prepare can be analysed together! Code is not identical')
end
% get the max no. trials in any of the data sets that enter the test
if max(max(allTEpermtest{1}.TEprepare.nrtrials))>cfg.maxtrials
cfg.maxtrials = max(max(allTEpermtest{1}.TEprepare.nrtrials));
end
u_values = cat(2,u_values,allTEpermtest{ll}.TEpermvalues(:,6));
end
clear ll
% get number of channel combinations
nrchannelcombi = size(allTEpermtest{1}.sgncmb,1);
cfg.verbosity = TEpermtest.TEprepare.cfg.verbosity;
%% check cfg
% -------------------------------------------------------------------------
if ~isfield(cfg, 'alpha'), cfg.alpha = 0.05; end;
if ~isfield(cfg, 'correctm'), cfg.correctm = 'FDR'; end;
if ~isfield(cfg, 'tail'), cfg.tail = 2; end;
if ~isfield(cfg, 'rawvalues'), cfg.rawvalues = 1; end;
if ~isfield(cfg, 'verbosity')
cfg.verbosity = TEpermtest.TEprepare.cfg.verbosity;
end
if ~isfield(cfg, 'design'),
error('TRENTOOL ERROR: cfg.design must be defined, see help!');
end;
if ~isfield(cfg, 'ivar'),
error('TRENTOOL ERROR: cfg.ivar must be defined, see help!');
end;
if ~isfield(cfg, 'uvar'),
error('TRENTOOL ERROR: cfg.uvar must be defined, see help!');
end;
if size(cfg.design,2) ~= nrdatasets
error('TRENTOOL ERROR: cfg.design must have the same number of columns than number of datasets exist, see help!');
end
if ~isfield(cfg, 'permstatstype'), cfg.permstatstype = 'mean'; end;
if ~strcmp(cfg.permstatstype , 'mean') && ~strcmp(cfg.permstatstype , 'indepsamplesT') && ~strcmp(cfg.permstatstype , 'depsamplesT')
error('TRENTOOL ERROR: wrong cfg.permstatstype - use ''mean'' ''depsamplesT'' or ''indepsamplesT'', see help!');
end
if ~isfield(cfg, 'fileidout'),
error('TRENTOOL ERROR: cfg.fileidout must be defined, see help!');
end;
%% Define conditions
% -------------------------------------------------------------------------
TEconsoleoutput(cfg.verbosity, 'Defining conditions', LOG_INFO_MINOR);
conds = squeeze(cfg.design(cfg.ivar,:));
condtype = unique(conds);
nrconds = length(condtype);
if nrconds ~=2
fprintf(strcat(['TRENTOOL ERROR: You defined ',num2str(nrconds),' conditions.\n']))
error('TRENTOOL ERROR: You have to define two conditions in cfg.design, see help!');
end
units = squeeze(cfg.design(cfg.uvar,:));
unittype = unique(units);
nrunits = length(unittype);
condindex1 = find(conds == condtype(1));
condindex2 = find(conds == condtype(2));
msg = {'Total no. data sets:' num2str(nrunits);
'Condition 1, ind data sets: ' num2str(condindex1);
'Condition 2, ind data sets: ' num2str(condindex2)
};
TEconsoleoutput(cfg.verbosity, msg, LOG_INFO_MINOR);
u_mean1 = mean(u_values(:,condindex1),2);
u_mean2 = mean(u_values(:,condindex2),2);
clear u_values;
%% check nr of permutations
% -------------------------------------------------------------------------
TEconsoleoutput(cfg.verbosity, 'Checking number of permutations', LOG_INFO_MINOR);
cfg.numpermutation = TEchecknumperm( ...
cfg, nrchannelcombi, length(condindex1), length(condindex2));
%% Create TEresultmean (mean over trials per subject)
% -------------------------------------------------------------------------
% looping over conditions and subjects makes sure we have the same order of
% subjects within conditions -> this is important for unsorted data
% entering a dependent samples t-test
subject = 1;
sorted_design = [];
TEresultmean.TEmat = zeros(nrchannelcombi,nrdatasets);
for c = condtype
for u = unittype
ind = cfg.design(cfg.uvar,:) == u & cfg.design(cfg.ivar,:) == c;
if cfg.rawvalues
TEresultmean.TEmat(:,subject) = squeeze(mean(allTEpermtest{ind}.TEmat, 2));
else
TEresultmean.TEmat(:,subject) = allTEpermtest{ind}.TEpermvalues(:,4);
end
%TEresultmean.TEmat(:,subject) = squeeze(mean(allTEpermtest{ind}.TEmat, 2));
sorted_design(:,subject) = [u;c]; % for debugging
subject = subject + 1;
end
end
%% permutation tests
% -------------------------------------------------------------------------
TEconsoleoutput(cfg.verbosity, 'Preparing condition matrices', LOG_INFO_MINOR);
TEresult1.TEmat = TEresultmean.TEmat(:,1:nrunits);
TEresult2.TEmat = TEresultmean.TEmat(:,nrunits+1:end);
TEpermtestgroup = TEperm(cfg,TEresult1,TEresult2);
% add mean u value from group with bigger TE value
ind_u_1 = TEpermtestgroup.TEpermvalues(:,2) & TEpermtestgroup.TEpermvalues(:,4)>0;
ind_u_2 = TEpermtestgroup.TEpermvalues(:,2) & TEpermtestgroup.TEpermvalues(:,4)<0;
u_vec = sum([u_mean1.*ind_u_1 u_mean2.*ind_u_2],2);
TEpermtestgroup.TEpermvalues = [TEpermtestgroup.TEpermvalues u_vec];
% add info to output structure (keep TEprepare info relevant for all subjects)
TEpermtestgroup.dimord = 'chanpair_value';
TEpermtestgroup.cfg = cfg;
TEpermtestgroup.sgncmb = allTEpermtest{1}.sgncmb;
TEpermtestgroup.numpermutation = cfg.numpermutation;
TEpermtestgroup.nrdatasets = nrdatasets;
TEpermtestgroup.TEgroupprepare = allTEpermtest{1}.groupprepare;
TEpermtestgroup.TEgroupprepare.ensemblemethod = allTEpermtest{1}.TEprepare.ensemblemethod;
TEpermtestgroup.TEgroupprepare.cfg = allTEpermtest{1}.TEprepare.cfg;
TEpermtestgroup.TEgroupprepare.channelcombi = allTEpermtest{1}.TEprepare.channelcombi;
TEpermtestgroup.TEgroupprepare.channelcombilabel = allTEpermtest{1}.TEprepare.channelcombilabel;
TEpermtestgroup.TEgroupprepare.channellabel = allTEpermtest{1}.TEprepare.channellabel;
TEpermtestgroup.TEgroupprepare.timeindices = allTEpermtest{1}.TEprepare.timeindices;
TEpermtestgroup.TEgroupprepare.optdim = allTEpermtest{1}.TEprepare.optdim;
%% save results
% -------------------------------------------------------------------------
toi = allTEpermtest{1}.cfg.toi;
TEconsoleoutput(cfg.verbosity, 'Saving results', LOG_INFO_MINOR);
% this saves data for each group, that enters the group statistics
savename1 = strcat(cfg.fileidout,'_time',num2str(toi(1)),'-',num2str(toi(2)),'s_TEpermtestgroup_data.mat');
save(savename1, 'TEresultmean','TEresult1','TEresult2','-v7.3');
% this saves the output of the group statistics
save(strcat(cfg.fileidout,'_time',num2str(toi(1)),'-',num2str(toi(2)),'s_TEpermtestgroup_output.mat'), ...
'TEpermtestgroup','-v7.3');
end