-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathpos1_respiration_heart_rate16s_detection.m
217 lines (156 loc) · 9.8 KB
/
pos1_respiration_heart_rate16s_detection.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
% Respiration rate and heart rate detection of signals collected with a respiratory frequency
% of 0.0625 Hz for 12 cycles of inhalation and exhalation
%% Recording of LSM6DS3 accelerometer (STMicroelectronics) signals
% datasheet: https://www.st.com/resource/en/datasheet/lsm6ds3.pdf
% file: signals/pos1_accelerometer_data_16sbreathing.txt
%
% They were recorded with a sampling rate of 202 Hz and
% a resolution of 0.244 mg/LSB
fprintf('Loading of pos1_accelerometer_data_16sbreathing.txt \n');
load signals/pos1_accelerometer_data_16sbreathing.txt % data composed by the following columns: time, gFx, gFy, gFz, TgF
fprintf('Loading of pos2_accelerometer_data_16sbreathing.txt \n');
load signals/pos2_accelerometer_data_16sbreathing.txt % data composed by the following columns: time, gFx, gFy, gFz, TgF
%% Data acquisition step
fprintf('*Data acquisition step* \n');
% remove time values duplicated
[~,uidx] = unique(pos1_accelerometer_data_16sbreathing(:,1),'stable');
pos1_accelerometer_data = pos1_accelerometer_data_16sbreathing(uidx,:);
[~,uidx] = unique(pos2_accelerometer_data_16sbreathing(:,1),'stable');
pos2_accelerometer_data = pos2_accelerometer_data_16sbreathing(uidx,:);
% remove first 1010 rows (first 5 seconds of recording)
pos1_accelerometer_data(1:1010,:) = [];
pos2_accelerometer_data(1:1010,:) = [];
% length of accelerometer_data
pos1_accelerometer_data_length = length(pos1_accelerometer_data);
pos2_accelerometer_data_length = length(pos2_accelerometer_data);
% remove last 1010 rows (last 5 seconds of recording)
pos1_accelerometer_data(pos1_accelerometer_data_length-1010:pos1_accelerometer_data_length,:) = [];
pos2_accelerometer_data(pos2_accelerometer_data_length-1010:pos2_accelerometer_data_length,:) = [];
% take the time axis
pos1_timeAxis = pos1_accelerometer_data(:,1);
pos2_timeAxis = pos2_accelerometer_data(:,1);
% we want the signal in mG:
resolution = 0.244; % mg/LSB
fprintf('Application resolution: %.3f mg/LSB \n', resolution);
pos1_accelerometer_data= pos1_accelerometer_data * resolution;
pos2_accelerometer_data= pos2_accelerometer_data * resolution;
%% Pre-processing step
% Plotting all the signals collected
fprintf('*Pre-processing step* \n');
figure(1)
signals_plotting(pos1_accelerometer_data, pos1_timeAxis, 'Pos1-accelerometer-data-16sbreathing')
figure(2)
signals_plotting(pos2_accelerometer_data, pos2_timeAxis, 'Pos2-accelerometer-data-16sbreathing')
%% Respiratory frequency detection step
% reference: https://doi.org/10.22489/CinC.2017.137-402
fprintf('*RESPIRATORY FREQUENCY DETECTION* \n');
%% Design a 4th-order lowpass Butterworth filter with a cut-off frequency of 0.5 Hz
fs = 202; % sampling rate at which the application samples the signals
fc = 0.5; % cut frequency fixed at 0.5 Hz
fc_rad = fc/(fs/2); % cut frequency converted into 0.00495 rad/sample
n_order = 4; % order of the Butterworth filter
fprintf('Design of %d order Butterworth filter with %d Hz as sampling rate and %.1f Hz as cut frequency \n', n_order, fs, fc);
[b,a] = butter(n_order, fc_rad, 'low'); % get Transfer function coefficients of the
% 4th-order lowpass Butterworth filter
% plot magnitude and phase
figure(3)
freqz(b, a)
pos1_accelerometer_data_filtered = filter(b, a, pos1_accelerometer_data); % apply the filter
% plot the original data and the filtered data
figure(4)
signals_filtered_plotting(pos1_accelerometer_data, pos1_accelerometer_data_filtered, pos1_timeAxis, 'Pos1-accelerometer-data-16sbreathing')
% plot the y axis of the original and the filtered data
pos1_y = pos1_accelerometer_data(:,3);
pos1_y_filtered = pos1_accelerometer_data_filtered(:,3);
figure(5)
yAxis_filtered_plotting(pos1_y, pos1_y_filtered, pos1_timeAxis, 'Pos1-accelerometer-data-16sbreathing')
%% Time domain analysis: Detection of minima of y axis of the filtered acceleration data
half_breath_duration = 8; % 8 seconds represent half of the entire breath duration
fprintf('*Time-domain analysis* \n');
fprintf('Detection of minima of y axis of the filtered acceleration data \n');
local_minima_indexes = islocalmin(pos1_y_filtered, 'MinSeparation', half_breath_duration, 'SamplePoints', pos1_timeAxis);
local_minima = pos1_y_filtered(local_minima_indexes);
time_local_minima = pos1_timeAxis(local_minima_indexes);
figure(6)
plot(pos1_timeAxis,pos1_y_filtered,'g', time_local_minima, local_minima,'r.', 'MarkerSize',20)
title('Pos1-accelerometer-data-16sbreathing filtered - minima detection')
xlabel('time (s)')
ylabel('Y (mg/LSB)')
%% Time domain analysis: respiratory frequency computation
fprintf('Detection of respiratory intervals \n');
respiratory_intervals = abs(diff(time_local_minima));
respiratory_frequency = 1/mean(respiratory_intervals);
fprintf('Respiratory frequency detected: %.4f Hz \n', respiratory_frequency);
breath_duration = 1/respiratory_frequency;
fprintf('Breath duration detected: %.4f s \n', breath_duration);
%% Frequency domain analysis: computation of PSDls (Lomb-Scargle Power Spectral Density)
fprintf('*Frequency-domain analysis* \n');
figure(7)
plomb(pos1_y_filtered, pos1_timeAxis, fs) % include frequencies up to 202 Hz
df = 1/fs; % fine grid with a spacing of 1/202 (0.00495)s
fvec = 0.05:df:0.5; % input frequencies (only frequencies we want to consider for zooming)
figure(8)
plomb(pos1_y_filtered,pos1_timeAxis,fvec)
df_str = sprintf('df = %.4f',df);
legend(df_str)
hold off
fprintf('Computation of PSDls (Lomb-Scargle Power Spectral Density) \n');
[pxx,f] = plomb(pos1_y_filtered,pos1_timeAxis,fvec);
%% Frequency domain analysis: respiratory frequency computation
fprintf('Detection of the maximum peak of the Lomb-Scargle Periodogram of y axis of the filtered acceleration data \n');
[maximum_peak, maximum_peak_index] = max(plomb(pos1_y_filtered,pos1_timeAxis,fvec));
figure(9)
semilogy(f,pxx,'c', f(maximum_peak_index), maximum_peak, 'r.', 'MarkerSize',20)
title('Lomb-Scargle Power Spectral Density estimate - maximum peak detection')
xlabel('Frequency (Hz)')
ylabel('PSD')
breath_duration = 1/f(maximum_peak_index);
fprintf('Breath duration detected: %.4f s \n', breath_duration);
%% Heart rate detection step
% reference: https://doi.org/10.1109/EMBC.2016.7590755
fprintf('*HEART RATE DETECTION* \n');
%% Design a 4th-order bandpass Butterworth filter with a band of 5-25 Hz for the z component of the accelerometer in position 1
low_fc = 5;
high_fc = 25;
low_fc_rad = low_fc/(fs/2); % lower cut frequency converted into 0.00495 rad/sample
high_fc_rad = high_fc/(fs/2); % higher cut frequency converted into 0.2475 rad/sample
fprintf('Design of %d order Butterworth bandpass filter with %d Hz as sampling rate and a band of %d - %d Hz \n', n_order, fs, low_fc, high_fc);
[b1,a1] = butter(n_order, [low_fc_rad high_fc_rad], 'bandpass'); % get Transfer function coefficients of the
% 4th-order bandpass Butterworth filter
% plot magnitude and phase
figure(10)
freqz(b1, a1)
pos1_accelerometer_data_bandpassed = filter(b1, a1, pos1_accelerometer_data); % apply the filter
% plot the original data and the filtered data
figure(11)
signals_filtered_plotting(pos1_accelerometer_data, pos1_accelerometer_data_bandpassed, pos1_timeAxis, 'Pos1-accelerometer-data-16sbreathing')
% plot the y axis of the original and the filtered data
pos1_z = pos1_accelerometer_data(:,4);
pos1_z_filtered = pos1_accelerometer_data_bandpassed(:,4);
figure(12)
zAxis_filtered_plotting(pos1_z, pos1_z_filtered, pos1_timeAxis, 'Pos1-accelerometer-data-16sbreathing')
%% Design a 4th-order bandpass Butterworth filter with a band of 1-25 Hz for the y component of the accelerometer in position 2
low_fc = 1;
high_fc = 25;
low_fc_rad = low_fc/(fs/2); % lower cut frequency converted into 0.0099 rad/sample
high_fc_rad = high_fc/(fs/2); % higher cut frequency converted into 0.2475 rad/sample
fprintf('Design of %d order Butterworth bandpass filter with %d Hz as sampling rate and a band of %d - %d Hz \n', n_order, fs, low_fc, high_fc);
[b2,a2] = butter(n_order, [low_fc_rad high_fc_rad], 'bandpass'); % get Transfer function coefficients of the
% 4th-order bandpass Butterworth filter
% plot magnitude and phase
figure(13)
freqz(b2, a2)
pos2_accelerometer_data_bandpassed = filter(b2, a2, pos2_accelerometer_data); % apply the filter
% plot the original data and the filtered data
figure(14)
signals_filtered_plotting(pos2_accelerometer_data, pos2_accelerometer_data_bandpassed, pos2_timeAxis, 'Pos2-accelerometer-data-16sbreathing')
% plot the y axis of the original and the filtered data
pos2_y = pos2_accelerometer_data(:,3);
pos2_y_filtered = pos2_accelerometer_data_bandpassed(:,3);
figure(15)
yAxis_filtered_plotting(pos2_y, pos2_y_filtered, pos2_timeAxis, 'Pos2-accelerometer-data-16sbreathing')
%% Plotting the signals filtered to detect the heart rate
figure(16)
signals_heart_rate_plotting_1(pos1_accelerometer_data, pos1_accelerometer_data_bandpassed, pos2_accelerometer_data, pos2_accelerometer_data_bandpassed, pos1_timeAxis, pos2_timeAxis, 'Pos1-accelerometer-data-16sbreathing', 'Pos2-accelerometer-data-16sbreathing')
figure(17)
signals_heart_rate_plotting_2(pos1_z_filtered, pos2_y_filtered, pos1_timeAxis, pos2_timeAxis, 'accelerometer-data-16sbreathing')