diff --git a/src/FFT.cpp b/src/FFT.cpp index 76e2557..2171da9 100644 --- a/src/FFT.cpp +++ b/src/FFT.cpp @@ -18,6 +18,7 @@ inline double WindowRectangle(unsigned int k, unsigned int offset, unsigned int inline double WindowVonHann(unsigned int k, unsigned int offset, unsigned int width); inline double WindowGauss(unsigned int k, unsigned int offset, unsigned int width); inline double WindowTriangle(unsigned int k, unsigned int offset, unsigned int width); +inline double WindowBlackman(unsigned int k, unsigned int offset, unsigned int width); std::vector> radix2dit( @@ -103,6 +104,7 @@ void SetWindowFunction(WindowFunctions func, unsigned int width) case WindowFunctions::VON_HANN: window = std::bind(WindowVonHann, std::placeholders::_1, 0, width); break; case WindowFunctions::GAUSS: window = std::bind(WindowGauss, std::placeholders::_1, 0, width); break; case WindowFunctions::TRIANGLE: window = std::bind(WindowTriangle, std::placeholders::_1, 0, width); break; + case WindowFunctions::BLACKMAN: window = std::bind(WindowBlackman, std::placeholders::_1, 0, width); break; } } @@ -126,3 +128,8 @@ inline double WindowTriangle(unsigned int k, unsigned int offset, unsigned int w { return 1.0f - std::abs(((double)k - ((double)width / 2.0f)) / ((double)width / 2.0f)); } + +inline double WindowBlackman(unsigned int k, unsigned int offset, unsigned int width) +{ + return 0.5f * (1.0f - 0.16f) - 0.5f * cos(2.0f * M_PI * k / (width - 1)) + 0.5f * 0.16f * cos(4.0f * M_PI * k / (width - 1)); +} diff --git a/src/FFT.hpp b/src/FFT.hpp index f7d8ad3..0a04842 100644 --- a/src/FFT.hpp +++ b/src/FFT.hpp @@ -6,7 +6,8 @@ enum class WindowFunctions { RECTANGLE, GAUSS, VON_HANN, - TRIANGLE + TRIANGLE, + BLACKMAN }; extern std::vector> FFT(const std::vector::const_iterator& begin, diff --git a/src/main.cpp b/src/main.cpp index 17ddd2a..8daff5a 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -15,7 +15,8 @@ const std::map FUNCTIONS { {"rectangle", WindowFunctions::RECTANGLE}, {"von-hann", WindowFunctions::VON_HANN}, {"gauss", WindowFunctions::GAUSS}, - {"triangle", WindowFunctions::TRIANGLE} + {"triangle", WindowFunctions::TRIANGLE}, + {"blackman", WindowFunctions::BLACKMAN} }; struct Settings { @@ -138,7 +139,7 @@ Settings Parse(int argc, char** argv) ("i,interval", "Splits audio file into intervals of length i milliseconds and transforms them individually (0 to not split file)", cxxopts::value()) ("f,frequency", "Defines the frequency range of the output spectrum (Default: all the frequencies)", cxxopts::value>()) ("p,pad", "Add extra zero-padding. By default, the program will pad the signals with 0s until the number of samples is a power of 2 (this would be equivalent to -p 1). With this option you can tell the program to instead pad until the power of 2 after the next one (-p 2) etc. This increases frequency resolution", cxxopts::value()) - ("w,window", "Specify the window function used (rectangle (default), von-hann, gauss, triangle)", cxxopts::value()->default_value("rectangle")) + ("w,window", "Specify the window function used (rectangle (default), von-hann, gauss, triangle, blackman (3-term))", cxxopts::value()->default_value("rectangle")) ("m,mono", "Analyze only the given channel", cxxopts::value()->default_value("0")) ("files", "Files to fourier transform", cxxopts::value>()) ("h,help", "Print usage")