mirror of
https://github.com/AlexandreRouma/SDRPlusPlus.git
synced 2025-01-07 08:47:12 +01:00
177 lines
6.7 KiB
C++
177 lines
6.7 KiB
C++
#include <waterfall.h>
|
|
#include <algorithm>
|
|
|
|
#define MAP_VAL(aMin, aMax, bMin, bMax, val) ( ( ( ((val) - (aMin)) / ((aMax) - (aMin)) ) * ((bMax) - (bMin)) ) + bMin)
|
|
|
|
const float COLOR_MAP[][3] = {
|
|
{0x4A, 0x00, 0x00},
|
|
{0x75, 0x00, 0x00},
|
|
{0x9F, 0x00, 0x00},
|
|
{0xC6, 0x00, 0x00},
|
|
{0xFF, 0x00, 0x00},
|
|
{0xFE, 0x6D, 0x16},
|
|
{0xFF, 0xFF, 0x00},
|
|
{0xFF, 0xFF, 0xFF},
|
|
{0x1E, 0x90, 0xFF},
|
|
{0x00, 0x00, 0x91},
|
|
{0x00, 0x00, 0x50},
|
|
{0x00, 0x00, 0x30},
|
|
{0x00, 0x00, 0x20}
|
|
};
|
|
|
|
namespace ImGui {
|
|
WaterFall::WaterFall() {
|
|
std::vector<float> base;
|
|
for (int i = 0; i < 1024; i++) {
|
|
base.push_back(-100.0f);
|
|
}
|
|
fftBuffer.push_back(base);
|
|
newSamples = false;
|
|
glGenTextures(1, &textureId);
|
|
}
|
|
|
|
void drawFFT(ImGuiWindow* window, int width, int height, ImVec2 pos, std::vector<float>& data) {
|
|
float lineHeight = (float)(height - 20 - 30) / 7.0f;
|
|
char buf[100];
|
|
|
|
for (int i = 0; i < 8; i++) {
|
|
sprintf(buf, "%d", -i * 10);
|
|
window->DrawList->AddText(ImVec2(pos.x + 7, pos.y + (i * lineHeight) + 2), IM_COL32( 255, 255, 255, 255 ), buf);
|
|
if (i == 7) {
|
|
window->DrawList->AddLine(ImVec2(pos.x + 40, pos.y + (i * lineHeight) + 10),
|
|
ImVec2(pos.x + width - 10, pos.y + (i * lineHeight) + 10),
|
|
IM_COL32( 255, 255, 255, 255 ), 1.0f);
|
|
break;
|
|
}
|
|
window->DrawList->AddLine(ImVec2(pos.x + 40, pos.y + (i * lineHeight) + 10),
|
|
ImVec2(pos.x + width - 10, pos.y + (i * lineHeight) + 10),
|
|
IM_COL32( 70, 70, 70, 255 ), 1.0f);
|
|
}
|
|
|
|
int fftWidth = width - 50;
|
|
|
|
int dataCount = data.size();
|
|
float multiplier = (float)dataCount / (float)fftWidth;
|
|
for (int i = 1; i < fftWidth; i++) {
|
|
float a = (data[(int)((float)(i - 1) * multiplier)] / 10.0f) * lineHeight;
|
|
float b = (data[(int)((float)i * multiplier)] / 10.0f) * lineHeight;
|
|
window->DrawList->AddLine(ImVec2(pos.x + i + 39, pos.y - a),
|
|
ImVec2(pos.x + i + 40, pos.y - b),
|
|
IM_COL32( 0, 255, 255, 255 ), 1.0f);
|
|
|
|
window->DrawList->AddLine(ImVec2(pos.x + i + 39, pos.y - a),
|
|
ImVec2(pos.x + i + 39, pos.y + (7.0f * lineHeight) + 9),
|
|
IM_COL32( 0, 255, 255, 50 ), 1.0f);
|
|
}
|
|
|
|
// window->DrawList->AddLine(ImVec2(pos.x + ((i - 1) * spacing) + 40, pos.y - a),
|
|
// ImVec2(pos.x + (i * spacing) + 40, pos.y - b),
|
|
// IM_COL32( 0, 255, 255, 255 ), 1.0f);
|
|
|
|
window->DrawList->AddLine(ImVec2(pos.x + 40, pos.y + 10), ImVec2(pos.x + 40, pos.y + (7 * lineHeight) + 10), IM_COL32( 255, 255, 255, 255 ), 1.0f);
|
|
}
|
|
|
|
uint32_t mapColor(float val) {
|
|
float mapped = MAP_VAL(-50.0f, 0.0f, 0, 12, val);
|
|
mapped = std::max<float>(mapped, 0.0f);
|
|
mapped = std::min<float>(mapped, 12.0f);
|
|
int floored = floorf(mapped);
|
|
float ratio = mapped - (float)floored;
|
|
|
|
float r = ((COLOR_MAP[floored][2] * (1.0f - ratio)) + (COLOR_MAP[floored + 1][2] * ratio));
|
|
float g = ((COLOR_MAP[floored][1] * (1.0f - ratio)) + (COLOR_MAP[floored + 1][1] * ratio));
|
|
float b = ((COLOR_MAP[floored][0] * (1.0f - ratio)) + (COLOR_MAP[floored + 1][0] * ratio));
|
|
|
|
//printf("%f %f %f\n", r, g, b);
|
|
|
|
return ((uint32_t)255 << 24) | ((uint32_t)b << 16) | ((uint32_t)g << 8) | (uint32_t)r;
|
|
}
|
|
|
|
uint32_t* img = NULL;
|
|
int lastW = 0;
|
|
int lastH = 0;
|
|
|
|
void WaterFall::drawWaterfall(ImGuiWindow* window, int width, int height, ImVec2 pos) {
|
|
int w = width - 10;
|
|
int h = height;
|
|
int count = fftBuffer.size();
|
|
float factor = (float)count / (float)w;
|
|
bool newSize = false;
|
|
|
|
if (lastW != w || lastH != h) {
|
|
newSize = true;
|
|
lastW = w;
|
|
lastH = h;
|
|
if (img != NULL) {
|
|
free(img);
|
|
}
|
|
printf("Allocating new buffer");
|
|
img = (uint32_t*)malloc(w * h * sizeof(uint32_t));
|
|
newSamples = true;
|
|
}
|
|
|
|
if (newSamples || newSize) {
|
|
newSamples = false;
|
|
float factor;
|
|
|
|
if (newSize) {
|
|
for (int y = 0; y < count; y++) {
|
|
factor = (float)fftBuffer[y].size() / (float)w;
|
|
for (int x = 0; x < w; x++) {
|
|
img[(y * w) + x] = mapColor(fftBuffer[y][(int)((float)x * factor)]);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
factor = (float)fftBuffer[0].size() / (float)w;
|
|
memcpy(&img[w], img, (h - 1) * w * sizeof(uint32_t));
|
|
for (int x = 0; x < w; x++) {
|
|
img[x] = mapColor(fftBuffer[0][(int)((float)x * factor)]);
|
|
}
|
|
}
|
|
|
|
|
|
glBindTexture(GL_TEXTURE_2D, textureId);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
glPixelStorei(GL_UNPACK_ROW_LENGTH, 0);
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, w, h, 0, GL_RGBA, GL_UNSIGNED_BYTE, img);
|
|
}
|
|
|
|
|
|
|
|
window->DrawList->AddImage((void*)(intptr_t)textureId, ImVec2(pos.x + 40, pos.y), ImVec2(pos.x + w, pos.y + h));
|
|
}
|
|
|
|
void WaterFall::draw() {
|
|
ImGuiWindow* window = GetCurrentWindow();
|
|
ImVec2 vMin = ImGui::GetWindowContentRegionMin();
|
|
ImVec2 vMax = ImGui::GetWindowContentRegionMax();
|
|
vMin.x += ImGui::GetWindowPos().x;
|
|
vMin.y += ImGui::GetWindowPos().y;
|
|
vMax.x += ImGui::GetWindowPos().x;
|
|
vMax.y += ImGui::GetWindowPos().y;
|
|
int width = vMax.x - vMin.x;
|
|
int height = vMax.y - vMin.y;
|
|
window->DrawList->AddRect( vMin, vMax, IM_COL32( 50, 50, 50, 255 ) );
|
|
|
|
window->DrawList->AddLine(ImVec2(vMin.x, vMin.y + 300), ImVec2(vMin.x + width, vMin.y + 300), IM_COL32( 50, 50, 50, 255 ), 1.0f);
|
|
|
|
buf_mtx.lock();
|
|
if (fftBuffer.size() > height - 302) {
|
|
fftBuffer.resize(height - 302);
|
|
}
|
|
drawFFT(window, width, 300, vMin, fftBuffer[0]);
|
|
drawWaterfall(window, width - 2, height - 302, ImVec2(vMin.x + 1, vMin.y + 301));
|
|
buf_mtx.unlock();
|
|
}
|
|
|
|
void WaterFall::pushFFT(std::vector<float> data, int n) {
|
|
buf_mtx.lock();
|
|
fftBuffer.insert(fftBuffer.begin(), data);
|
|
newSamples = true;
|
|
buf_mtx.unlock();
|
|
}
|
|
};
|
|
|