-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathphastimate.m
76 lines (63 loc) · 3.08 KB
/
phastimate.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
function [phase, amplitude] = phastimate(data, D, edge, ord, hilbertwindow, varargin)
%PHASTIMATE Pre-stimulus phase estimation
% phase = PHASTIMATE(data, D, edge, ord, hilbertwindow, [offset_correction], [iterations], [armethod]) estimates the phase of the matrix data
%
% data is a time x channel matrix
% D is the digitalFilter object
% edge is the filter edge size (removed)
% ord is the order of the autoregressive yule-walker model
% hilbertwindow is the length of the window used for the hilberttransform
% iterations is the number of samples forward predicted (default is edge + hilbertwindow/2)
% armethod is the method used to estimate autoregressive parametres (default is @aryule)
%
% Example:
% Fs = 200;
% data = [sin([1:Fs]/Fs*5*2*pi)' sin([1:Fs]/Fs*6*2*pi)'] + randn(Fs,2);
% D = designfilt('bandpassfir', 'FilterOrder', round(Fs/5), 'CutoffFrequency1', 4, 'CutoffFrequency2', 7, 'SampleRate', Fs);
%
% phase = phastimate(data, D, 25, 20, 64)
if nargin < 6
offset_correction = 0;
else
offset_correction = varargin{1};
end
if nargin < 7
iterations = edge + ceil(hilbertwindow/2);
else
iterations = varargin{2};
assert(iterations > edge, 'iterations must be larger than the number of edge samples')
end
if nargin < 8
armethod = @aryule; %could be aryule, arburg
else
armethod = varargin{3};
end
% demean the data
data = detrend(data,'constant');
% filter the data
data_filtered = filtfilt(D, data); %note that filtfilt uses reflection and sets the initial values
data_filtered_withoutedge = data_filtered(edge+1:end-edge,:);
% determine AR parameters
[a, e, rc] = armethod(data_filtered_withoutedge, ord);
coefficients = -1 * flip(a(:, 2:end)');
% prepare matrix with the aditional time points for the forward prediction
data_filtered_withoutedge_predicted = [data_filtered_withoutedge; ones(iterations, size(data,2))];
% run the forward prediction
for i = iterations:-1:1
data_filtered_withoutedge_predicted(end-i+1,:) = ...
sum(coefficients .* data_filtered_withoutedge_predicted((end-i-ord+1):(end-i),:));
end
%hold on
%plot(data_filtered_withoutedge_predicted, '--')
%plot(data_filtered_withoutedge)
%TODO: de-mean again? Or just demean the window of data used for the
%hilbert transform? Or use a fancier method for detection of the zero line?
data_filtered_withoutedge_predicted_hilbertwindow = data_filtered_withoutedge_predicted(end-hilbertwindow+1:end,:);
% analytic signal and angle
data_filtered_withoutedge_predicted_hilbertwindow_analytic = hilbert(data_filtered_withoutedge_predicted_hilbertwindow);
%plot((size(data_filtered_withoutedge_predicted,1)-hilbertwindow+1):size(data_filtered_withoutedge_predicted,1), angle(data_filtered_withoutedge_predicted_hilbertwindow_analytic).*max(data_filtered_withoutedge_predicted_hilbertwindow(:))/pi)
%xpos = (size(data_filtered_withoutedge_predicted,1)-iterations+edge);
%line([xpos xpos], ylim(gca))
phase = angle(data_filtered_withoutedge_predicted_hilbertwindow_analytic(end-iterations+edge+offset_correction,:));
amplitude = mean(abs(data_filtered_withoutedge_predicted_hilbertwindow_analytic));
end