/** * Furnace Tracker - multi-system chiptune tracker * Copyright (C) 2021-2022 tildearrow and contributors * * 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 2 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, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "gui.h" #include "imgui_internal.h" // TODO: // - potentially move oscilloscope seek position to the end, and read the last samples // - this allows for setting up the window size void FurnaceGUI::readOsc() { int writePos=e->oscWritePos; int readPos=e->oscReadPos; int avail=0; int total=0; if (firstFrame) { readPos=writePos; } if (writePos>=readPos) { avail=writePos-readPos; } else { avail=writePos-readPos+32768; } if (oscTotal==0) { oscTotal=ImGui::GetIO().DeltaTime*e->getAudioDescGot().rate; } else { oscTotal=(oscTotal+(int)round(ImGui::GetIO().DeltaTime*e->getAudioDescGot().rate))>>1; } int bias=avail-oscTotal-e->getAudioDescGot().bufsize; if (bias<0) bias=0; total=oscTotal+(bias>>6); if (total>avail) total=avail; //printf("total: %d. avail: %d bias: %d\n",total,avail,bias); for (int i=0; i<512; i++) { int pos=(readPos+(i*total/512))&0x7fff; oscValues[i]=(e->oscBuf[0][pos]+e->oscBuf[1][pos])*0.5f; if (oscValues[i]>0.001f || oscValues[i]<-0.001f) { WAKE_UP; } } float peakDecay=0.05f*60.0f*ImGui::GetIO().DeltaTime; for (int i=0; i<2; i++) { peak[i]*=1.0-peakDecay; if (peak[i]<0.0001) { peak[i]=0.0; } else { WAKE_UP; } float newPeak=peak[i]; for (int j=0; joscBuf[i][pos])>newPeak) { newPeak=fabs(e->oscBuf[i][pos]); } } peak[i]+=(newPeak-peak[i])*0.9; } readPos=(readPos+total)&0x7fff; e->oscReadPos=readPos; } void FurnaceGUI::drawOsc() { if (nextWindow==GUI_WINDOW_OSCILLOSCOPE) { oscOpen=true; ImGui::SetNextWindowFocus(); nextWindow=GUI_WINDOW_NOTHING; } if (!oscOpen) return; ImGui::SetNextWindowSizeConstraints(ImVec2(64.0f*dpiScale,32.0f*dpiScale),ImVec2(scrW*dpiScale,scrH*dpiScale)); if (settings.oscTakesEntireWindow) { ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding,ImVec2(0,0)); ImGui::PushStyleVar(ImGuiStyleVar_ItemSpacing,ImVec2(0,0)); ImGui::PushStyleVar(ImGuiStyleVar_ItemInnerSpacing,ImVec2(0,0)); } if (ImGui::Begin("Oscilloscope",&oscOpen)) { if (oscZoomSlider) { if (ImGui::VSliderFloat("##OscZoom",ImVec2(20.0f*dpiScale,ImGui::GetContentRegionAvail().y),&oscZoom,0.5,2.0)) { if (oscZoom<0.5) oscZoom=0.5; if (oscZoom>2.0) oscZoom=2.0; } ImGui::SameLine(); } ImDrawList* dl=ImGui::GetWindowDrawList(); ImGuiWindow* window=ImGui::GetCurrentWindow(); ImVec2 waveform[512]; ImVec2 size=ImGui::GetContentRegionAvail(); ImVec2 minArea=window->DC.CursorPos; ImVec2 maxArea=ImVec2( minArea.x+size.x, minArea.y+size.y ); ImRect rect=ImRect(minArea,maxArea); ImRect inRect=rect; inRect.Min.x+=dpiScale; inRect.Min.y+=dpiScale; inRect.Max.x-=dpiScale; inRect.Max.y-=dpiScale; ImGuiStyle& style=ImGui::GetStyle(); ImU32 color=ImGui::GetColorU32(isClipping?uiColors[GUI_COLOR_OSC_WAVE_PEAK]:uiColors[GUI_COLOR_OSC_WAVE]); ImU32 borderColor=ImGui::GetColorU32(uiColors[GUI_COLOR_OSC_BORDER]); ImU32 refColor=ImGui::GetColorU32(uiColors[GUI_COLOR_OSC_REF]); ImU32 guideColor=ImGui::GetColorU32(uiColors[GUI_COLOR_OSC_GUIDE]); ImGui::ItemSize(size,style.FramePadding.y); if (ImGui::ItemAdd(rect,ImGui::GetID("wsDisplay"))) { // https://github.com/ocornut/imgui/issues/3710 const int v0 = dl->VtxBuffer.Size; dl->AddRectFilled(inRect.Min,inRect.Max,0xffffffff,settings.oscRoundedCorners?(8.0f*dpiScale):0.0f); const int v1 = dl->VtxBuffer.Size; for (int i=v0; iVtxBuffer.Data[i]; ImVec4 col0=uiColors[GUI_COLOR_OSC_BG1]; ImVec4 col1=uiColors[GUI_COLOR_OSC_BG3]; ImVec4 col2=uiColors[GUI_COLOR_OSC_BG2]; ImVec4 col3=uiColors[GUI_COLOR_OSC_BG4]; float shadeX=(v->pos.x-rect.Min.x)/(rect.Max.x-rect.Min.x); float shadeY=(v->pos.y-rect.Min.y)/(rect.Max.y-rect.Min.y); if (shadeX<0.0f) shadeX=0.0f; if (shadeX>1.0f) shadeX=1.0f; if (shadeY<0.0f) shadeY=0.0f; if (shadeY>1.0f) shadeY=1.0f; col0.x+=(col2.x-col0.x)*shadeX; col0.y+=(col2.y-col0.y)*shadeX; col0.z+=(col2.z-col0.z)*shadeX; col0.w+=(col2.w-col0.w)*shadeX; col1.x+=(col3.x-col1.x)*shadeX; col1.y+=(col3.y-col1.y)*shadeX; col1.z+=(col3.z-col1.z)*shadeX; col1.w+=(col3.w-col1.w)*shadeX; col0.x+=(col1.x-col0.x)*shadeY; col0.y+=(col1.y-col0.y)*shadeY; col0.z+=(col1.z-col0.z)*shadeY; col0.w+=(col1.w-col0.w)*shadeY; ImVec4 conv=ImGui::ColorConvertU32ToFloat4(v->col); col0.x*=conv.x; col0.y*=conv.y; col0.z*=conv.z; col0.w*=conv.w; v->col=ImGui::ColorConvertFloat4ToU32(col0); } dl->AddLine( ImLerp(rect.Min,rect.Max,ImVec2(0.0f,0.5f)), ImLerp(rect.Min,rect.Max,ImVec2(1.0f,0.5f)), refColor, dpiScale ); dl->AddLine( ImLerp(rect.Min,rect.Max,ImVec2(0.48f,0.125f)), ImLerp(rect.Min,rect.Max,ImVec2(0.52f,0.125f)), guideColor, dpiScale ); dl->AddLine( ImLerp(rect.Min,rect.Max,ImVec2(0.47f,0.25f)), ImLerp(rect.Min,rect.Max,ImVec2(0.53f,0.25f)), guideColor, dpiScale ); dl->AddLine( ImLerp(rect.Min,rect.Max,ImVec2(0.45f,0.375f)), ImLerp(rect.Min,rect.Max,ImVec2(0.55f,0.375f)), guideColor, dpiScale ); dl->AddLine( ImLerp(rect.Min,rect.Max,ImVec2(0.45f,0.625f)), ImLerp(rect.Min,rect.Max,ImVec2(0.55f,0.625f)), guideColor, dpiScale ); dl->AddLine( ImLerp(rect.Min,rect.Max,ImVec2(0.47f,0.75f)), ImLerp(rect.Min,rect.Max,ImVec2(0.53f,0.75f)), guideColor, dpiScale ); dl->AddLine( ImLerp(rect.Min,rect.Max,ImVec2(0.48f,0.875f)), ImLerp(rect.Min,rect.Max,ImVec2(0.52f,0.875f)), guideColor, dpiScale ); dl->AddLine( ImLerp(rect.Min,rect.Max,ImVec2(0.5f,0.08f)), ImLerp(rect.Min,rect.Max,ImVec2(0.5f,0.92f)), guideColor, dpiScale ); for (size_t i=0; i<512; i++) { float x=(float)i/512.0f; float y=oscValues[i]*oscZoom; if (y<-0.5f) y=-0.5f; if (y>0.5f) y=0.5f; waveform[i]=ImLerp(inRect.Min,inRect.Max,ImVec2(x,0.5f-y)); } dl->AddPolyline(waveform,512,color,ImDrawFlags_None,dpiScale); if (settings.oscBorder) { dl->AddRect(inRect.Min,inRect.Max,borderColor,settings.oscRoundedCorners?(8.0f*dpiScale):0.0f,0,1.5f*dpiScale); } } if (ImGui::IsItemClicked(ImGuiMouseButton_Right)) { oscZoomSlider=!oscZoomSlider; } } if (settings.oscTakesEntireWindow) { ImGui::PopStyleVar(3); } if (ImGui::IsWindowFocused(ImGuiFocusedFlags_ChildWindows)) curWindow=GUI_WINDOW_OSCILLOSCOPE; ImGui::End(); }