per-channel oscilloscope, part 9
work in progress!
This commit is contained in:
parent
269f07f597
commit
2c9bad3b3d
|
@ -209,11 +209,13 @@ struct DivRegWrite {
|
||||||
struct DivDispatchOscBuffer {
|
struct DivDispatchOscBuffer {
|
||||||
unsigned int rate;
|
unsigned int rate;
|
||||||
unsigned short needle;
|
unsigned short needle;
|
||||||
|
unsigned short readNeedle;
|
||||||
short data[65536];
|
short data[65536];
|
||||||
|
|
||||||
DivDispatchOscBuffer():
|
DivDispatchOscBuffer():
|
||||||
rate(65536),
|
rate(65536),
|
||||||
needle(0) {
|
needle(0),
|
||||||
|
readNeedle(0) {
|
||||||
memset(data,0,65536*sizeof(short));
|
memset(data,0,65536*sizeof(short));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -30,15 +30,38 @@ void FurnaceGUI::drawChanOsc() {
|
||||||
if (!chanOscOpen) return;
|
if (!chanOscOpen) return;
|
||||||
ImGui::SetNextWindowSizeConstraints(ImVec2(64.0f*dpiScale,32.0f*dpiScale),ImVec2(scrW*dpiScale,scrH*dpiScale));
|
ImGui::SetNextWindowSizeConstraints(ImVec2(64.0f*dpiScale,32.0f*dpiScale),ImVec2(scrW*dpiScale,scrH*dpiScale));
|
||||||
if (ImGui::Begin("Oscilloscope (per-channel)",&chanOscOpen)) {
|
if (ImGui::Begin("Oscilloscope (per-channel)",&chanOscOpen)) {
|
||||||
if (ImGui::InputInt("Columns",&chanOscCols,1,1)) {
|
if (ImGui::BeginTable("ChanOscSettings",3)) {
|
||||||
if (chanOscCols<1) chanOscCols=1;
|
ImGui::TableNextRow();
|
||||||
if (chanOscCols>64) chanOscCols=64;
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Columns");
|
||||||
|
ImGui::SameLine();
|
||||||
|
ImGui::SetNextItemWidth(ImGui::GetContentRegionAvail().x);
|
||||||
|
if (ImGui::InputInt("##COSColumns",&chanOscCols,1,1)) {
|
||||||
|
if (chanOscCols<1) chanOscCols=1;
|
||||||
|
if (chanOscCols>64) chanOscCols=64;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Size (ms)");
|
||||||
|
ImGui::SameLine();
|
||||||
|
ImGui::SetNextItemWidth(ImGui::GetContentRegionAvail().x);
|
||||||
|
if (ImGui::InputFloat("##COSWinSize",&chanOscWindowSize,1.0f,1.0f)) {
|
||||||
|
if (chanOscWindowSize<1.0f) chanOscWindowSize=1.0f;
|
||||||
|
if (chanOscWindowSize>50.0f) chanOscWindowSize=50.0f;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Checkbox("Center waveform",&chanOscWaveCorr);
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
ImGui::PushStyleVar(ImGuiStyleVar_CellPadding,ImVec2(0.0f,0.0f));
|
ImGui::PushStyleVar(ImGuiStyleVar_CellPadding,ImVec2(0.0f,0.0f));
|
||||||
float availY=ImGui::GetContentRegionAvail().y;
|
float availY=ImGui::GetContentRegionAvail().y;
|
||||||
if (ImGui::BeginTable("ChanOsc",chanOscCols,ImGuiTableFlags_Borders)) {
|
if (ImGui::BeginTable("ChanOsc",chanOscCols,ImGuiTableFlags_Borders)) {
|
||||||
std::vector<DivDispatchOscBuffer*> oscBufs;
|
std::vector<DivDispatchOscBuffer*> oscBufs;
|
||||||
|
std::vector<int> oscChans;
|
||||||
int chans=e->getTotalChannelCount();
|
int chans=e->getTotalChannelCount();
|
||||||
ImDrawList* dl=ImGui::GetWindowDrawList();
|
ImDrawList* dl=ImGui::GetWindowDrawList();
|
||||||
ImGuiWindow* window=ImGui::GetCurrentWindow();
|
ImGuiWindow* window=ImGui::GetCurrentWindow();
|
||||||
|
@ -49,7 +72,10 @@ void FurnaceGUI::drawChanOsc() {
|
||||||
|
|
||||||
for (int i=0; i<chans; i++) {
|
for (int i=0; i<chans; i++) {
|
||||||
DivDispatchOscBuffer* buf=e->getOscBuffer(i);
|
DivDispatchOscBuffer* buf=e->getOscBuffer(i);
|
||||||
if (buf!=NULL) oscBufs.push_back(buf);
|
if (buf!=NULL) {
|
||||||
|
oscBufs.push_back(buf);
|
||||||
|
oscChans.push_back(i);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
int rows=(oscBufs.size()+(chanOscCols-1))/chanOscCols;
|
int rows=(oscBufs.size()+(chanOscCols-1))/chanOscCols;
|
||||||
|
|
||||||
|
@ -58,13 +84,14 @@ void FurnaceGUI::drawChanOsc() {
|
||||||
ImGui::TableNextColumn();
|
ImGui::TableNextColumn();
|
||||||
|
|
||||||
DivDispatchOscBuffer* buf=oscBufs[i];
|
DivDispatchOscBuffer* buf=oscBufs[i];
|
||||||
|
int ch=oscChans[i];
|
||||||
if (buf==NULL) {
|
if (buf==NULL) {
|
||||||
ImGui::Text("Error!");
|
ImGui::Text("Error!");
|
||||||
} else {
|
} else {
|
||||||
ImVec2 size=ImGui::GetContentRegionAvail();
|
ImVec2 size=ImGui::GetContentRegionAvail();
|
||||||
size.y=availY/rows;
|
size.y=availY/rows;
|
||||||
|
|
||||||
int displaySize=(buf->rate)/30;
|
int displaySize=(float)(buf->rate)*(chanOscWindowSize/1000.0f);
|
||||||
|
|
||||||
ImVec2 minArea=window->DC.CursorPos;
|
ImVec2 minArea=window->DC.CursorPos;
|
||||||
ImVec2 maxArea=ImVec2(
|
ImVec2 maxArea=ImVec2(
|
||||||
|
@ -85,7 +112,26 @@ void FurnaceGUI::drawChanOsc() {
|
||||||
waveform[i]=ImLerp(inRect.Min,inRect.Max,ImVec2(x,0.5f));
|
waveform[i]=ImLerp(inRect.Min,inRect.Max,ImVec2(x,0.5f));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
unsigned short needlePos=buf->needle-displaySize;
|
unsigned short needlePos=buf->needle;
|
||||||
|
if (chanOscWaveCorr) {
|
||||||
|
float cutoff=0.01f;
|
||||||
|
while (buf->readNeedle!=needlePos) {
|
||||||
|
//float old=chanOscLP1[ch];
|
||||||
|
chanOscLP0[ch]+=cutoff*((float)buf->data[buf->readNeedle]-chanOscLP0[ch]);
|
||||||
|
chanOscLP1[ch]+=cutoff*(chanOscLP0[ch]-chanOscLP1[ch]);
|
||||||
|
if (chanOscLP1[ch]>=20) {
|
||||||
|
lastCorrPos[ch]=buf->readNeedle;
|
||||||
|
}
|
||||||
|
buf->readNeedle++;
|
||||||
|
}
|
||||||
|
needlePos=lastCorrPos[ch];
|
||||||
|
/*
|
||||||
|
for (unsigned short i=0; i<displaySize; i++) {
|
||||||
|
short old=buf->data[needlePos--];
|
||||||
|
if (buf->data[needlePos]>old) break;
|
||||||
|
}*/
|
||||||
|
}
|
||||||
|
needlePos-=displaySize;
|
||||||
for (unsigned short i=0; i<512; i++) {
|
for (unsigned short i=0; i<512; i++) {
|
||||||
float x=(float)i/512.0f;
|
float x=(float)i/512.0f;
|
||||||
float y=(float)buf->data[(unsigned short)(needlePos+(i*displaySize/512))]/65536.0f;
|
float y=(float)buf->data[(unsigned short)(needlePos+(i*displaySize/512))]/65536.0f;
|
||||||
|
|
|
@ -4000,6 +4000,8 @@ FurnaceGUI::FurnaceGUI():
|
||||||
oscZoom(0.5f),
|
oscZoom(0.5f),
|
||||||
oscZoomSlider(false),
|
oscZoomSlider(false),
|
||||||
chanOscCols(3),
|
chanOscCols(3),
|
||||||
|
chanOscWindowSize(20.0f),
|
||||||
|
chanOscWaveCorr(true),
|
||||||
followLog(true),
|
followLog(true),
|
||||||
pianoOctaves(7),
|
pianoOctaves(7),
|
||||||
pianoOptions(false),
|
pianoOptions(false),
|
||||||
|
@ -4057,4 +4059,8 @@ FurnaceGUI::FurnaceGUI():
|
||||||
memset(patChanSlideY,0,sizeof(float)*(DIV_MAX_CHANS+1));
|
memset(patChanSlideY,0,sizeof(float)*(DIV_MAX_CHANS+1));
|
||||||
memset(lastIns,-1,sizeof(int)*DIV_MAX_CHANS);
|
memset(lastIns,-1,sizeof(int)*DIV_MAX_CHANS);
|
||||||
memset(oscValues,0,sizeof(float)*512);
|
memset(oscValues,0,sizeof(float)*512);
|
||||||
|
|
||||||
|
memset(chanOscLP0,0,sizeof(float)*DIV_MAX_CHANS);
|
||||||
|
memset(chanOscLP1,0,sizeof(float)*DIV_MAX_CHANS);
|
||||||
|
memset(lastCorrPos,0,sizeof(short)*DIV_MAX_CHANS);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1101,6 +1101,12 @@ class FurnaceGUI {
|
||||||
|
|
||||||
// per-channel oscilloscope
|
// per-channel oscilloscope
|
||||||
int chanOscCols;
|
int chanOscCols;
|
||||||
|
float chanOscWindowSize;
|
||||||
|
bool chanOscWaveCorr;
|
||||||
|
float chanOscLP0[DIV_MAX_CHANS];
|
||||||
|
float chanOscLP1[DIV_MAX_CHANS];
|
||||||
|
unsigned short lastNeedlePos[DIV_MAX_CHANS];
|
||||||
|
unsigned short lastCorrPos[DIV_MAX_CHANS];
|
||||||
|
|
||||||
// visualizer
|
// visualizer
|
||||||
float keyHit[DIV_MAX_CHANS];
|
float keyHit[DIV_MAX_CHANS];
|
||||||
|
|
Loading…
Reference in a new issue