How can I calculate a mean value of peak amplitude and area under the curve?
    7 visualizzazioni (ultimi 30 giorni)
  
       Mostra commenti meno recenti
    
    Erik Näslund
 il 14 Feb 2023
  
    
    
    
    
    Commentato: Star Strider
      
      
 il 14 Giu 2023
            Hello,
I hope someone can help me with my problem. As a healthcare professional, I lack the basic understanding of Data programming and mathematics. 
I have a recording of, how the absorption varies, from an infrared photoplethysmography signal obtained in vivo. The data is imported in Matlab as column vectors. I have managed to apply a low pass and a bandpass filter to the raw signal, separating the different components I have to analyse further. From the band pass filtered signal (picture below), I need to – for a number of given time points (illustrated with a green arrow):
-  Determine the peak amplitude (local minima to the peak). 
-  Determine the area under the curve (AUC) – from minima to the peak. 
-  The amplitude and AUC should be estimated with some mean value, e.g. +/- 5 seconds from the given time point or +/- 5 peaks. I don’t know which of them is the best method? 
Does anyone have any suggestions for possible solutions? 
I appreciate any help you can provide.

Y-axis: light absorption in arbitrary units, X-axis: time, based on sampling rate (100 Hz) – every second = 100 units
2 Commenti
  Dyuman Joshi
      
      
 il 15 Feb 2023
				Do you have to find the amplitude and AUC for every peak? or just corresponding to the global maximum peak?
Risposta accettata
  Star Strider
      
      
 il 15 Feb 2023
        It always helps to have a signal to work with, so I found one from a previous pulse-plethysmograph post and am using it here.  (All the pre-processing steps may not be necessary for your signal, however they were for this one.  They also demonstrate how to pre-process a physiological signal in general.)  
Try something like this — 
PPG = readmatrix('https://www.mathworks.com/matlabcentral/answers/uploaded_files/103608/p3_sit.csv');   % Use PPG Signal From Previoous Answer
PPG = PPG(PPG>=1E+6);                                                                                   % Remove Artefacts (This Signal Only)
L = size(PPG,1)
Fs = 960;
Fn = Fs/2;
t = linspace(0, L-1, L)/Fs;                                                                             % Time Vector
t = t(:);                                                                                               % Force Column Vector
NFFT = 2^nextpow2(L)
FT_PPG = fft((PPG-mean(PPG)).*hann(L),NFFT);
Fv = linspace(0,1,NFFT/2+1)*Fn;
Iv = 1:numel(Fv);
figure
plot(Fv, abs(FT_PPG(Iv)*2))
grid
xlim([0 10])
PPG = lowpass(PPG, 5, Fs, 'ImpulseResponse','iir');                                                     % Remove High-Frequency Noise (This Signal Only)
PPGmax = max(PPG);
PPGmin = min(PPG);
[pks,plocs] = findpeaks(PPG, 'MinPeakProminence',(PPGmax-PPGmin)*0.05);                                 % Peaks
[vys,vlocs] = findpeaks(-PPG, 'MinPeakProminence',(PPGmax-PPGmin)*0.05);                                % Valleys
for k = 1:numel(vlocs)-1
    idxrng = vlocs(k) : vlocs(k+1);                                                                     % Index Range For This Pulse
    idxlim = [vlocs(k); vlocs(k+1)];                                                                    % Inmdex Limits
    tv = t(idxrng);                                                                                     % Pulse 't' Vector
    sv = PPG(idxrng);                                                                                   % Pulse 'PPG' Vector
    ts(k,:) = t(vlocs(k));                                                                              % Pulse Start Time
    te(k,:) = t(vlocs(k+1));                                                                            % Pul;se End Time
    kv(k,:) = k;                                                                                        % Peak Counter
    B = [t(idxlim) ones(2,1)] \ PPG(idxlim);                                                            % Linear Detrending Regression Parameters
    dtsv = [t(idxrng) ones(size(t(idxrng)))] * B;                                                       % Linear Detrending Vector
    svc = sv - dtsv;                                                                                    % Detrended Signal
    p2p(k,:) = max(svc);                                                                                % Peak-To-Peak Amplitude
    AUC(k,:) = trapz(tv,svc);                                                                           % Pulse Area
end
PPG_Statistics = table(kv,ts,te,p2p,AUC, 'VariableNames',{'Peak','t_start','t_end','PeakToPeak','AUC'})
figure
hp{1} = plot(t, PPG, 'DisplayName','PPG Data');
hold on
hp{2} = plot(t(plocs), pks, '^r', 'DisplayName','Peaks');
hp{3} = plot(t(vlocs), -vys, 'vr', 'DisplayName','Valleys');
hp{4} = plot((t(plocs(1:numel(p2p)))*[1 1]).', [pks(1:numel(p2p)) pks(1:numel(p2p))-p2p].', '-k', 'DisplayName','Peak-Peak');
hp{5} = plot(t(vlocs), -vys, '-g', 'DisplayName','Inter-Valley Baseline');
hold off
grid
legend([hp{1}(1),hp{2}(1),hp{3}(1),hp{4}(1),hp{5}(1)],'Location','best')
xlim([0 10])                                                                                            % Zoom To See Details
I have no idea what the original units were for this signal, however the values for the derived statistics appear to be correct.  
Try my code with your signal.  If there are any problems, I will help adapt my code to it.  
.
10 Commenti
Più risposte (1)
  Aman
    
 il 15 Feb 2023
        Hi,
I understand that you have a function, and you want to find out its global maxima. Then you want to find the nearest local minima on the either side of the global maxima and then calculate the area between the two local minima.
For finding the global maxima you can use the “max” function. Upon getting the global maxima you can use the “islocal” function to get all the local minima, upon getting all the local minima you can find the nearest local minima on the either side of the global maxima. Once you get the local minima of the either side you can find the subsection between them and then use the ”trapz” function to get the area between them. You can refer to the following example:
x = 1:100;
A = (1-cos(2*pi*0.01*x)).*sin(2*pi*0.15*x); % objective function
[maxVal,indx] = max(A); % global maxima
TF = islocalmin(A); % all local minima
for i=indx:-1:1
    if(TF(i))
        localMinLeft = i;
        localMinLeftVal = A(i);    % local minima to the left of the global maxima
        break;
    end
end
for i=indx:100
    if(TF(i))
        localMinRight = i;
        localMinRightVal = A(i);    % local minima to the right of the global maxima
        break;
    end
end
subSection = A(localMinLeft:localMinRight);     % subsection between the local minimas
plot(x,A);
hold on;
plot(localMinRight,localMinRightVal,'r*');
plot(localMinLeft,localMinLeftVal,'r*');
plot(x(indx),A(indx),'bo');
plot(localMinLeft:localMinRight,subSection);
hold off;
disp('Area under the curve is ' + trapz(subSection));    % area under the subsection selected
Hope this helps!
Vedere anche
Community Treasure Hunt
Find the treasures in MATLAB Central and discover how the community can help you!
Start Hunting!








