added crude audio loading
This commit is contained in:
parent
4a0e5548dd
commit
63aeae6e21
3
.gitmodules
vendored
3
.gitmodules
vendored
|
@ -7,3 +7,6 @@
|
||||||
[submodule "vendor/lol"]
|
[submodule "vendor/lol"]
|
||||||
path = vendor/lol
|
path = vendor/lol
|
||||||
url = git@github.com:Lauchmelder23/lol.git
|
url = git@github.com:Lauchmelder23/lol.git
|
||||||
|
[submodule "vendor/sdl"]
|
||||||
|
path = vendor/sdl
|
||||||
|
url = git@github.com:libsdl-org/SDL.git
|
||||||
|
|
|
@ -18,6 +18,14 @@ if(NOT GLFW3_FOUND)
|
||||||
set(GLFW3_LIBRARIES glfw)
|
set(GLFW3_LIBRARIES glfw)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
find_package(SDL2)
|
||||||
|
if(NOT SDL2_FOUND)
|
||||||
|
message(STATUS "Could not find SDL binaries on system, building from source instead")
|
||||||
|
add_subdirectory("vendor/sdl")
|
||||||
|
|
||||||
|
set(SDL2_INCLUDE_DIRS sdl_static)
|
||||||
|
set(SDL2_LIBRARIES sdl_static)
|
||||||
|
endif()
|
||||||
|
|
||||||
# Include sub-projects.
|
# Include sub-projects.
|
||||||
add_subdirectory ("vendor/lol")
|
add_subdirectory ("vendor/lol")
|
||||||
|
|
BIN
res/payday.wav
Normal file
BIN
res/payday.wav
Normal file
Binary file not shown.
|
@ -1,14 +1,17 @@
|
||||||
#include "Application.hpp"
|
#include "Application.hpp"
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
#include <bitset>
|
||||||
#include <stdexcept>
|
#include <stdexcept>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
#include <glad/glad.h>
|
#include <glad/glad.h>
|
||||||
#include <GLFW/glfw3.h>
|
#include <GLFW/glfw3.h>
|
||||||
|
#include <SDL2/SDL.h>
|
||||||
#include "imgui.h"
|
#include "imgui.h"
|
||||||
#include "backends/imgui_impl_glfw.h"
|
#include "backends/imgui_impl_glfw.h"
|
||||||
#include "backends/imgui_impl_opengl3.h"
|
#include "backends/imgui_impl_opengl3.h"
|
||||||
|
|
||||||
#include "Colormaps.hpp"
|
#include "Colormaps.hpp"
|
||||||
|
|
||||||
#ifdef NDEBUG
|
#ifdef NDEBUG
|
||||||
|
@ -29,6 +32,7 @@ void Application::Quit()
|
||||||
|
|
||||||
if (window != nullptr)
|
if (window != nullptr)
|
||||||
{
|
{
|
||||||
|
delete file;
|
||||||
delete topology;
|
delete topology;
|
||||||
|
|
||||||
manager.Clear();
|
manager.Clear();
|
||||||
|
@ -37,6 +41,7 @@ void Application::Quit()
|
||||||
window = nullptr;
|
window = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SDL_Quit();
|
||||||
glfwTerminate();
|
glfwTerminate();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,7 +49,10 @@ void Application::Init(int width, int height, const std::string& title)
|
||||||
{
|
{
|
||||||
// Initialize GLFW
|
// Initialize GLFW
|
||||||
if (window == nullptr)
|
if (window == nullptr)
|
||||||
|
{
|
||||||
glfwInit();
|
glfwInit();
|
||||||
|
SDL_Init(SDL_INIT_AUDIO);
|
||||||
|
}
|
||||||
|
|
||||||
int windowWidth = width, windowHeight = height;
|
int windowWidth = width, windowHeight = height;
|
||||||
GLFWmonitor* monitor = NULL;
|
GLFWmonitor* monitor = NULL;
|
||||||
|
@ -62,6 +70,9 @@ void Application::Init(int width, int height, const std::string& title)
|
||||||
windowHeight = mode->height;
|
windowHeight = mode->height;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
glfwWindowHint(GLFW_SAMPLES, 4);
|
||||||
|
glfwWindowHint(GLFW_DOUBLEBUFFER, GLFW_TRUE);
|
||||||
|
|
||||||
// Create GLFW window
|
// Create GLFW window
|
||||||
window = glfwCreateWindow(windowWidth, windowHeight, title.c_str(), monitor, NULL);
|
window = glfwCreateWindow(windowWidth, windowHeight, title.c_str(), monitor, NULL);
|
||||||
if (window == nullptr)
|
if (window == nullptr)
|
||||||
|
@ -164,16 +175,32 @@ void Application::Init(int width, int height, const std::string& title)
|
||||||
colormap = 3;
|
colormap = 3;
|
||||||
topology->SetColormap(colormaps[colormap]);
|
topology->SetColormap(colormaps[colormap]);
|
||||||
|
|
||||||
glfwWindowHint(GLFW_SAMPLES, 4);
|
file = new AudioFile("res/payday.wav");
|
||||||
|
SDL_AudioSpec specs = file->GetAudioSpec();
|
||||||
|
std::cout << "Channels: " << (int)specs.channels << std::endl;
|
||||||
|
std::cout << "Format: " << std::bitset<16>{specs.format} << std::endl;
|
||||||
|
std::cout << "Samples: " << specs.samples << std::endl;
|
||||||
|
std::cout << "Frequency: " << specs.freq << std::endl;
|
||||||
|
file->Normalize();
|
||||||
|
|
||||||
|
std::cout << "First 50 samples: " << std::endl;
|
||||||
|
for(auto it = file->begin(); it != file->begin() + 50; it++)
|
||||||
|
std::cout << *it << std::endl;
|
||||||
|
|
||||||
// glEnable(GL_CULL_FACE);
|
// glEnable(GL_CULL_FACE);
|
||||||
glEnable(GL_MULTISAMPLE);
|
glEnable(GL_MULTISAMPLE);
|
||||||
|
|
||||||
|
frameTimerStart = std::chrono::system_clock::now();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Application::Launch()
|
void Application::Launch()
|
||||||
{
|
{
|
||||||
while (!glfwWindowShouldClose(window))
|
while (!glfwWindowShouldClose(window))
|
||||||
{
|
{
|
||||||
|
unsigned int frametime = std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::system_clock::now() - frameTimerStart).count();
|
||||||
|
frameTimerStart = std::chrono::system_clock::now();
|
||||||
|
float fps = 1000.0f / frametime;
|
||||||
|
|
||||||
glfwPollEvents();
|
glfwPollEvents();
|
||||||
|
|
||||||
camera.SetPosition(pitch, yaw, distance);
|
camera.SetPosition(pitch, yaw, distance);
|
||||||
|
@ -199,6 +226,8 @@ void Application::Launch()
|
||||||
|
|
||||||
ImGui::Begin("Debug");
|
ImGui::Begin("Debug");
|
||||||
|
|
||||||
|
ImGui::Text("FPS: %f", fps);
|
||||||
|
|
||||||
if (ImGui::CollapsingHeader("Camera"))
|
if (ImGui::CollapsingHeader("Camera"))
|
||||||
{
|
{
|
||||||
ImGui::SliderFloat("Yaw", &yaw, 0.0f, 360.0f);
|
ImGui::SliderFloat("Yaw", &yaw, 0.0f, 360.0f);
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <chrono>
|
||||||
|
#include "Util.hpp"
|
||||||
#include "OrbitingCamera.hpp"
|
#include "OrbitingCamera.hpp"
|
||||||
#include "ScrollingPlot.hpp"
|
#include "ScrollingPlot.hpp"
|
||||||
#include "Util.hpp"
|
#include "AudioFile.hpp"
|
||||||
|
|
||||||
struct GLFWwindow;
|
struct GLFWwindow;
|
||||||
|
|
||||||
|
@ -44,6 +46,7 @@ private:
|
||||||
GLFWwindow* window = nullptr;
|
GLFWwindow* window = nullptr;
|
||||||
WindowData data;
|
WindowData data;
|
||||||
lol::ObjectManager manager;
|
lol::ObjectManager manager;
|
||||||
|
std::chrono::system_clock::time_point frameTimerStart;
|
||||||
|
|
||||||
OrbitingCamera camera;
|
OrbitingCamera camera;
|
||||||
float pitch, yaw, distance;
|
float pitch, yaw, distance;
|
||||||
|
@ -57,4 +60,5 @@ private:
|
||||||
int colormap = 0;
|
int colormap = 0;
|
||||||
|
|
||||||
ScrollingPlot* topology;
|
ScrollingPlot* topology;
|
||||||
|
AudioFile* file;
|
||||||
};
|
};
|
45
src/AudioFile.cpp
Normal file
45
src/AudioFile.cpp
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
#include "AudioFile.hpp"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
AudioFile::AudioFile(const std::string& path)
|
||||||
|
{
|
||||||
|
// Load the file from disk
|
||||||
|
uint8_t* tempBuf = nullptr;
|
||||||
|
if(SDL_LoadWAV(path.c_str(), &spec, &tempBuf, &length) == NULL)
|
||||||
|
{
|
||||||
|
std::cerr << "Failed to load audio file \"" << path << "\": " << SDL_GetError() << std::endl;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy raw audio buffer over into vector
|
||||||
|
// Perform conversion to float if necessary
|
||||||
|
unsigned int sampleSize = SDL_AUDIO_BITSIZE(spec.format);
|
||||||
|
bool isFloat = SDL_AUDIO_ISFLOAT(spec.format);
|
||||||
|
for(uint8_t* ptr = tempBuf; ptr < tempBuf + length; ptr += sampleSize)
|
||||||
|
{
|
||||||
|
if(isFloat)
|
||||||
|
buffer.push_back(*((float*)ptr));
|
||||||
|
else
|
||||||
|
buffer.push_back((float)*ptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Free the audio buffer
|
||||||
|
SDL_FreeWAV(tempBuf);
|
||||||
|
}
|
||||||
|
|
||||||
|
AudioFile::~AudioFile()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void AudioFile::Normalize()
|
||||||
|
{
|
||||||
|
float largestVal = 0.0f;
|
||||||
|
for(const float& sample : buffer)
|
||||||
|
largestVal = std::max(largestVal, std::abs(sample));
|
||||||
|
|
||||||
|
float normFactor = 1.0f / largestVal;
|
||||||
|
for(float& sample : buffer)
|
||||||
|
sample *= normFactor;
|
||||||
|
}
|
27
src/AudioFile.hpp
Normal file
27
src/AudioFile.hpp
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <cstdint>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include <SDL2/SDL_audio.h>
|
||||||
|
|
||||||
|
class AudioFile
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
AudioFile(const std::string& path);
|
||||||
|
~AudioFile();
|
||||||
|
|
||||||
|
inline std::vector<float>::const_iterator begin() const { return buffer.begin(); }
|
||||||
|
inline std::vector<float>::const_iterator end() const { return buffer.end(); }
|
||||||
|
|
||||||
|
inline const SDL_AudioSpec& GetAudioSpec() { return spec; }
|
||||||
|
inline uint32_t GetLength() { return length; }
|
||||||
|
|
||||||
|
void Normalize();
|
||||||
|
|
||||||
|
private:
|
||||||
|
SDL_AudioSpec spec;
|
||||||
|
uint32_t length;
|
||||||
|
std::vector<float> buffer;
|
||||||
|
};
|
|
@ -1,9 +1,11 @@
|
||||||
add_executable(visualizer
|
add_executable(visualizer
|
||||||
"main.cpp" "Application.cpp"
|
"main.cpp"
|
||||||
|
"Application.cpp"
|
||||||
"OrbitingCamera.cpp"
|
"OrbitingCamera.cpp"
|
||||||
"Topology.cpp"
|
"Topology.cpp"
|
||||||
"Colormaps.cpp"
|
"Colormaps.cpp"
|
||||||
"ScrollingPlot.cpp"
|
"ScrollingPlot.cpp"
|
||||||
|
"AudioFile.cpp"
|
||||||
)
|
)
|
||||||
|
|
||||||
target_sources(visualizer PUBLIC
|
target_sources(visualizer PUBLIC
|
||||||
|
@ -18,11 +20,17 @@ target_sources(visualizer PUBLIC
|
||||||
|
|
||||||
target_include_directories(visualizer PRIVATE
|
target_include_directories(visualizer PRIVATE
|
||||||
${GLFW3_INCLUDE_DIRS}
|
${GLFW3_INCLUDE_DIRS}
|
||||||
|
${SDL2_INCLUDE_DIRS}
|
||||||
${VENDOR_DIR}/imgui
|
${VENDOR_DIR}/imgui
|
||||||
lol
|
lol
|
||||||
)
|
)
|
||||||
|
|
||||||
target_link_libraries(visualizer PRIVATE
|
target_link_libraries(visualizer PRIVATE
|
||||||
${GLFW3_LIBRARIES}
|
${GLFW3_LIBRARIES}
|
||||||
|
${SDL2_LIBRARIES}
|
||||||
lol
|
lol
|
||||||
|
)
|
||||||
|
|
||||||
|
add_custom_command(TARGET visualizer POST_BUILD
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_directory ${CMAKE_SOURCE_DIR}/res $<TARGET_FILE_DIR:visualizer>/res
|
||||||
)
|
)
|
1
vendor/sdl
vendored
Submodule
1
vendor/sdl
vendored
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 2363ddc330efa939d2d43a6c809794970f401120
|
Loading…
Reference in a new issue