function [cr,repos,Lc]=filterbanksynchrosqueeze(c,tgrad,var)
%FILTERBANKSYNCHROSQUEEZE Synchrosqueeze filterbank spectrogram
% Usage: cr = filterbanksynchrosqueeze(c,tgrad,cfreq);
% cr = filterbanksynchrosqueeze(c,tgrad,g);
% [cr,repos,Lc] = filterbanksynchrosqueeze(...);
%
% Input parameters:
% c : Coefficients to be synchrosqueezed.
% tgrad : Instantaneous frequency relative to original position.
% cfreq : Vector of relative center frequencies in ]-1,1].
% g : Set of filters.
% Output parameters:
% cr : Synchrosqueezed filterbank coefficients.
% repos : Reassigned positions.
% Lc : Subband lengths.
%
% FILTERBANKSYNCHROSQUEEZE(c,tgrad,cfreq) will reassign the values of
% the filterbank coefficients c according to instantaneous frequency
% tgrad. The frequency center frequencies of filters are given by cfreq.
% The filterbank coefficients c are assumed to be obtained from a
% non-subsampled filterbank (a=1).
%
% FILTERBANKSYNCHROSQUEEZE(s,tgrad,g) will do the same thing except
% the center frequencies are estimated from a set of filters g.
%
% [sr,repos,Lc]=FILTERBANKSYNCHROSQUEEZE(...) does the same thing, but
% in addition returns a vector of subband lengths Lc (Lc = cellfun(@numel,s))
% and cell array repos with sum(Lc) elements. Each element corresponds
% to a single coefficient obtained by cell2mat(sr) and it is a vector
% of indices identifying coefficients from cell2mat(s) assigned to
% the particular time-frequency position.
%
% The arguments s, tgrad must be cell-arrays of vectors
% of the same lengths. Arguments cfreq or g must have the
% same number of elements as the cell arrays with coefficients.
%
% Examples:
% ---------
%
% This example shows how to synchrosqueeze a ERB filterbank spectrogram:
%
% % Genrate 3 chirps half a second long
% L = 22050; fs = 44100; l = 0:L-1;
%
% f = sin(2*pi*(l/35+(l/300).^2)) + ...
% sin(2*pi*(l/10+(l/300).^2)) + ...
% sin(2*pi*(l/5-(l/450).^2));
% f = 0.7*f';
%
% % Create ERB filterbank
% [g,~,fc]=erbfilters(fs,L,'uniform','spacing',1/12,'warped');
%
% % Compute phase gradient
% [tgrad,~,~,c]=filterbankphasegrad(f,g,1);
% % Do the reassignment
% sr=filterbanksynchrosqueeze(c,tgrad,cent_freqs(fs,fc));
% figure(1); subplot(211);
% plotfilterbank(c,1,fc,fs,60);
% title('ERBlet spectrogram of 3 chirps');
% subplot(212);
% plotfilterbank(sr,1,fc,fs,60);
% title('Synchrosqueezed ERBlet spectrogram of 3 chirps');
%
% See also: filterbankphasegrad, gabreassign
%
% References:
% N. Holighaus, Z. Průša, and P. L. Søndergaard. Reassignment and
% synchrosqueezing for general time-frequency filter banks, subsampling
% and processing. Signal Processing, 125:1--8, 2016. [1]http ]
%
% References
%
% 1. http://www.sciencedirect.com/science/article/pii/S0165168416000141
%
%
% Url: http://ltfat.github.io/doc/filterbank/filterbanksynchrosqueeze.html
% Copyright (C) 2005-2023 Peter L. Soendergaard <peter@sonderport.dk> and others.
% This file is part of LTFAT version 2.6.0
%
% 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, either version 3 of the License, or
% (at your option) any later version.
%
% This program is distributed in the hope that it will be useful,
% but WITHOUT ANY WARRANTY; without even the implied warranty of
% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
% GNU General Public License for more details.
%
% You should have received a copy of the GNU General Public License
% along with this program. If not, see <http://www.gnu.org/licenses/>.
% AUTHOR : Nicki Holighaus.
% Sanity checks
complainif_notenoughargs(nargin,3,'FILTERBANKSYNCHROSQUEEZE');
if isempty(c) || ~iscell(c)
error('%s: s should be a nonempty cell array.',upper(mfilename));
end
if isempty(tgrad) || ~iscell(tgrad) || any(~cellfun(@isreal,tgrad))
error('%s: tgrad should be a nonempty cell array.',upper(mfilename));
end
if any(cellfun(@(sEl,tEl) ~isvector(sEl) || ~isvector(tEl) ...
, c,tgrad))
error('%s: s, tgrad, must be cell arrays of numeric vectors.',...
upper(mfilename));
end
if ~isequal(size(c),size(tgrad)) || ...
any(cellfun(@(sEl,tEl) ~isequal(size(sEl),size(tEl)), ...
c,tgrad))
error('%s: s, tgrad does not have the same format.',upper(mfilename));
end
W = cellfun(@(sEl)size(sEl,2),c);
if any(W>1)
error('%s: Only one-channel signals are supported.',upper(mfilename));
end
% Number of channels
M = numel(c);
% Number of elements in channels
Lc = cellfun(@(sEl)size(sEl,1),c);
% Check if a comply with subband lengths
L = Lc;
if any(abs(L-L(1))>1e-6)
error(['%s: Subsampling factors and subband lengths do not ',...
'comply.'],upper(mfilename));
end
L = L(1);
% Determine center frequencies
if isempty(var) || numel(var)~=M || ~isvector(var) && ~iscell(var)
error(['%s: cfreq must be length-M numeric vector or a cell-array ',...
'containg M filters.'],upper(mfilename));
else
if iscell(var)
cfreq = cent_freqs(var,L);
else
cfreq = var;
end
end
% Dummy fgrad
fgrad = tgrad;
for m=1:numel(fgrad);
fgrad{m} = zeros(L,1);
end
a = ones(M,1);
% Do the computations
if nargout>1
[cr,repos] = comp_filterbankreassign(c,tgrad,fgrad,a,cfreq);
else
cr = comp_filterbankreassign(c,tgrad,fgrad,a,cfreq);
end