2023-12-05 10:49:51 +01:00
|
|
|
#include <hex/helpers/utils.hpp>
|
|
|
|
|
2023-12-23 21:09:41 +01:00
|
|
|
#include <content/visualizer_helpers.hpp>
|
2023-12-05 10:49:51 +01:00
|
|
|
|
|
|
|
#include <implot.h>
|
|
|
|
#include <imgui.h>
|
|
|
|
#include <miniaudio.h>
|
|
|
|
#include <fonts/codicons_font.h>
|
|
|
|
#include <hex/api/task_manager.hpp>
|
|
|
|
|
|
|
|
#include <hex/ui/imgui_imhex_extensions.h>
|
|
|
|
|
2023-12-23 21:09:41 +01:00
|
|
|
namespace hex::plugin::visualizers {
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-07-10 20:50:58 +02:00
|
|
|
void drawSoundVisualizer(pl::ptrn::Pattern &, bool shouldReset, std::span<const pl::core::Token::Literal> arguments) {
|
2023-12-05 10:49:51 +01:00
|
|
|
auto wavePattern = arguments[0].toPattern();
|
|
|
|
auto channels = arguments[1].toUnsigned();
|
|
|
|
auto sampleRate = arguments[2].toUnsigned();
|
2024-11-29 09:22:22 -07:00
|
|
|
u32 downSampling = wavePattern->getSize() / 300_scaled / 8 / channels;
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
static std::vector<i16> waveData;
|
|
|
|
static std::vector<std::vector<i16>> sampledData;
|
|
|
|
sampledData.resize(channels);
|
2023-12-05 10:49:51 +01:00
|
|
|
static ma_device audioDevice;
|
|
|
|
static ma_device_config deviceConfig;
|
|
|
|
static bool shouldStop = false;
|
|
|
|
static u64 index = 0;
|
|
|
|
static TaskHolder resetTask;
|
|
|
|
|
|
|
|
if (sampleRate == 0)
|
|
|
|
throw std::logic_error(hex::format("Invalid sample rate: {}", sampleRate));
|
|
|
|
else if (channels == 0)
|
|
|
|
throw std::logic_error(hex::format("Invalid channel count: {}", channels));
|
2024-11-29 09:22:22 -07:00
|
|
|
u64 sampledIndex;
|
2023-12-05 10:49:51 +01:00
|
|
|
if (shouldReset) {
|
|
|
|
waveData.clear();
|
|
|
|
|
2024-07-27 16:29:06 +02:00
|
|
|
resetTask = TaskManager::createTask("hex.visualizers.pl_visualizer.task.visualizing"_lang, TaskManager::NoProgress, [=](Task &) {
|
2023-12-05 10:49:51 +01:00
|
|
|
ma_device_stop(&audioDevice);
|
|
|
|
waveData = patternToArray<i16>(wavePattern.get());
|
2024-11-29 09:22:22 -07:00
|
|
|
if (waveData.empty())
|
|
|
|
return;
|
|
|
|
sampledData = sampleChannels(waveData, 300_scaled * 4, channels);
|
2023-12-05 10:49:51 +01:00
|
|
|
index = 0;
|
|
|
|
|
|
|
|
deviceConfig = ma_device_config_init(ma_device_type_playback);
|
|
|
|
deviceConfig.playback.format = ma_format_s16;
|
|
|
|
deviceConfig.playback.channels = channels;
|
|
|
|
deviceConfig.sampleRate = sampleRate;
|
|
|
|
deviceConfig.pUserData = &waveData;
|
|
|
|
deviceConfig.dataCallback = [](ma_device *device, void *pOutput, const void *, ma_uint32 frameCount) {
|
|
|
|
if (index >= waveData.size()) {
|
|
|
|
index = 0;
|
|
|
|
shouldStop = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ma_copy_pcm_frames(pOutput, waveData.data() + index, frameCount, device->playback.format, device->playback.channels);
|
2024-11-29 09:22:22 -07:00
|
|
|
index += frameCount * device->playback.channels;
|
2023-12-05 10:49:51 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
ma_device_init(nullptr, &deviceConfig, &audioDevice);
|
|
|
|
});
|
|
|
|
}
|
2024-11-29 09:22:22 -07:00
|
|
|
sampledIndex = index / downSampling;
|
2023-12-05 10:49:51 +01:00
|
|
|
ImGui::BeginDisabled(resetTask.isRunning());
|
2024-11-29 09:22:22 -07:00
|
|
|
u32 waveDataSize = waveData.size();
|
|
|
|
u32 sampledDataSize = sampledData[0].size();
|
|
|
|
auto subplotFlags = ImPlotSubplotFlags_LinkAllX | ImPlotSubplotFlags_LinkCols | ImPlotSubplotFlags_NoResize;
|
|
|
|
auto plotFlags = ImPlotFlags_CanvasOnly | ImPlotFlags_NoFrame | ImPlotFlags_NoInputs;
|
|
|
|
auto axisFlags = ImPlotAxisFlags_NoDecorations | ImPlotAxisFlags_NoMenus | ImPlotAxisFlags_AutoFit;
|
2023-12-05 10:49:51 +01:00
|
|
|
ImPlot::PushStyleVar(ImPlotStyleVar_PlotPadding, ImVec2(0, 0));
|
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
if (ImPlot::BeginSubplots("##AxisLinking", channels, 1, scaled(ImVec2(300, 80 * channels)), subplotFlags)) {
|
|
|
|
for (u32 i = 0; i < channels; i++) {
|
|
|
|
if (ImPlot::BeginPlot("##amplitude_plot", scaled(ImVec2(300, 80)), plotFlags)) {
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
ImPlot::SetupAxes("##time", "##amplitude", axisFlags, axisFlags);
|
|
|
|
double dragPos = sampledIndex;
|
|
|
|
if (ImPlot::DragLineX(1, &dragPos, ImGui::GetStyleColorVec4(ImGuiCol_Text))) {
|
|
|
|
if (dragPos < 0) dragPos = 0;
|
|
|
|
if (dragPos >= sampledDataSize) dragPos = sampledDataSize - 1;
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
sampledIndex = dragPos;
|
|
|
|
}
|
|
|
|
ImPlot::PlotLine("##audio", sampledData[i].data(), sampledDataSize);
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
ImPlot::EndPlot();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ImPlot::PopStyleVar();
|
|
|
|
|
|
|
|
index = sampledIndex * downSampling;
|
|
|
|
{
|
|
|
|
const u64 min = 0, max = sampledDataSize-1;
|
|
|
|
ImGui::PushItemWidth(300_scaled);
|
|
|
|
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(0, 0));
|
|
|
|
ImGui::SliderScalar("##index", ImGuiDataType_U64, &sampledIndex, &min, &max, "");
|
|
|
|
ImGui::PopStyleVar();
|
|
|
|
ImGui::PopItemWidth();
|
|
|
|
}
|
|
|
|
index = sampledIndex * downSampling;
|
2023-12-05 10:49:51 +01:00
|
|
|
|
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
if (shouldStop) {
|
|
|
|
shouldStop = false;
|
2023-12-05 10:49:51 +01:00
|
|
|
ma_device_stop(&audioDevice);
|
2024-11-29 09:22:22 -07:00
|
|
|
}
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
bool playing = ma_device_is_started(&audioDevice);
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
if (ImGuiExt::IconButton(playing ? ICON_VS_DEBUG_PAUSE : ICON_VS_PLAY, ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_ToolbarGreen))) {
|
|
|
|
if (playing)
|
|
|
|
ma_device_stop(&audioDevice);
|
|
|
|
else
|
|
|
|
ma_device_start(&audioDevice);
|
|
|
|
}
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
ImGui::SameLine();
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
if (ImGuiExt::IconButton(ICON_VS_DEBUG_STOP, ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_ToolbarRed))) {
|
|
|
|
index = 0;
|
|
|
|
sampledIndex = 0;
|
|
|
|
ma_device_stop(&audioDevice);
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::EndDisabled();
|
|
|
|
|
|
|
|
ImGui::SameLine();
|
|
|
|
index = sampledIndex * downSampling;
|
2023-12-05 10:49:51 +01:00
|
|
|
|
2024-11-29 09:22:22 -07:00
|
|
|
if (resetTask.isRunning())
|
|
|
|
ImGuiExt::TextSpinner("");
|
|
|
|
else
|
|
|
|
ImGuiExt::TextFormatted("{:02d}:{:02d}:{:03d} / {:02d}:{:02d}:{:03d}",
|
|
|
|
(index / sampleRate / channels) / 60, (index / sampleRate / channels) % 60, (index * 1000 / sampleRate / channels ) % 1000,
|
|
|
|
((waveDataSize-1) / sampleRate / channels) / 60, ((waveDataSize-1) / sampleRate / channels) % 60, ((waveDataSize-1) * 1000 / sampleRate / channels) % 1000);
|
|
|
|
ImPlot::EndSubplots();
|
|
|
|
}
|
2023-12-05 10:49:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|