win api 音频可视化
暂时记录,改天有时间再完善。。。其实写好好久了,但以前的代码丢了,重新写一遍。。
原理和 python 的一样,获取输入设备,然后把数据读取到 buffer 中,在绘制出来。
这里要注意两点:
1. waveformat 结构的参数都要填写正确才能打开设备,wavehdr结构必须先初始化才能调用准备函数,官方文档里都有解释。
2. 读取出来的数据是无符号字符类型(0~255),以及 window 坐标是以左上角为基准,所以,要正确展示波形需要注意下。
做了一些修改:
#include "framework.h"
#include "audio_analysis.h" #define MAX_LOADSTRING 100
#define RATE 44100
#define AUBUFF 2048
#define PIPE 2 HINSTANCE hInst; // 当前实例
WCHAR szTitle[MAX_LOADSTRING]; // 标题栏文本
WCHAR szWindowClass[MAX_LOADSTRING]; // 主窗口类名
WAVEFORMATEX waveformat;
WAVEHDR wavehdr;
HWAVEOUT hWaveOut;
HWAVEIN hWaveIn;
HWND hBtn;
WCHAR szFilter[] = L"ALL Files (*.*)\0*.*\0\0";
WCHAR szButton[] = L"播放";
BYTE waveBuffer[RATE * PIPE];
signed char audioData[AUBUFF];
BOOL playing = TRUE;
INT audioSum = 0; ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM); int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
_In_opt_ HINSTANCE hPrevInstance,
_In_ LPWSTR lpCmdLine,
_In_ int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine); // TODO: 在此处放置代码。 // 初始化全局字符串
LoadStringW(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
LoadStringW(hInstance, IDC_AUDIOANALYSIS, szWindowClass, MAX_LOADSTRING);
MyRegisterClass(hInstance); // 执行应用程序初始化:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
} HACCEL hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_AUDIOANALYSIS)); MSG msg; // 主消息循环:
while (GetMessage(&msg, nullptr, 0, 0))
{
if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
} return (int) msg.wParam;
} ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEXW wcex; wcex.cbSize = sizeof(WNDCLASSEX); wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_AUDIOANALYSIS));
wcex.hCursor = LoadCursor(nullptr, IDC_ARROW);
wcex.hbrBackground = CreateSolidBrush(RGB(0, 0, 0));
wcex.lpszMenuName = MAKEINTRESOURCEW(IDC_AUDIOANALYSIS);
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL)); return RegisterClassExW(&wcex);
} BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
hInst = hInstance; // 将实例句柄存储在全局变量中 HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr); if (!hWnd)
{
return FALSE;
} ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd); return TRUE;
} LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_CREATE:
{
HDC hdc = GetDC(hWnd);
RECT rect;
TEXTMETRIC textMetric; GetTextMetrics(hdc, &textMetric);
GetClientRect(hWnd, &rect); int rWidth = rect.right - rect.left,
cxChar = textMetric.tmAveCharWidth,
cyChar = textMetric.tmHeight + textMetric.tmExternalLeading,
bWidth = 10 * cxChar,
bHeight = 8 * cyChar / 4; hBtn = CreateWindow(TEXT("button"), szButton,
WS_CHILD | WS_VISIBLE | BS_PUSHBUTTON,
rWidth / 2 - bWidth / 2, cyChar, bWidth, bHeight,
hWnd, (HMENU)MAKEINTRESOURCE(IDM_BUTTON),
((LPCREATESTRUCT)lParam)->hInstance, NULL); waveformat.cbSize = 0;
waveformat.nAvgBytesPerSec = RATE * 4;
waveformat.nBlockAlign = 4;
waveformat.nChannels = PIPE;
waveformat.nSamplesPerSec = RATE;
waveformat.wBitsPerSample = 16;
waveformat.wFormatTag = WAVE_FORMAT_PCM;
UINT rst = waveInOpen(&hWaveIn, WAVE_MAPPER, &waveformat, (DWORD)hWnd, 0, CALLBACK_WINDOW);
if (rst != MMSYSERR_NOERROR)
{
MessageBox(hWnd, L"Could not find suitable input device",
L"Error Message", MB_ICONSTOP | MB_OK);
DestroyWindow(hWnd);
break;
}
}
break;
case WM_COMMAND:
{
int wmId = LOWORD(wParam);
switch (wmId)
{
case IDM_OPEN:
{
OPENFILENAME ofn;
WCHAR szFile[MAX_PATH]; ZeroMemory(&ofn, sizeof(ofn));
ofn.hwndOwner = hWnd;
ofn.lStructSize = sizeof(ofn);
ofn.lpstrFilter = szFilter;
ofn.nMaxFile = _MAX_PATH;
ofn.lpstrFile = szFile;
ofn.lpstrFile[0] = '\0';
ofn.nFilterIndex = 1;
ofn.lpstrFileTitle = NULL;
ofn.nMaxFileTitle = 0;
ofn.lpstrInitialDir = NULL;
ofn.Flags = OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST;
if (GetOpenFileName(&ofn) == TRUE)
{
WCHAR wComd[MAX_PATH];
_stprintf_s(wComd, TEXT("open cdaudio!%s"), szFile);
mciSendString(wComd, NULL, 0, 0);
_stprintf_s(wComd, TEXT("play cdaudio!%s"), szFile);
mciSendString(wComd, NULL, 0, 0);
}
else
{
MessageBoxW(NULL, TEXT("打开文件失败"), TEXT("错误"), MB_OK);
}
}
break;
case IDM_BUTTON:
{
if (playing == TRUE)
{
playing = FALSE;
waveInStop(hWaveIn);
SetWindowText(hBtn, TEXT("播放"));
}
else
{
playing = TRUE;
waveInStart(hWaveIn);
SetWindowText(hBtn, TEXT("暂停"));
}
}
break;
case IDM_ABOUT:
DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
break;
case IDM_EXIT:
DestroyWindow(hWnd);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
}
break;
case WM_PAINT:
{
PAINTSTRUCT ps;
HDC hdc = BeginPaint(hWnd, &ps);
RECT rect;
TEXTMETRIC textMetric; GetTextMetrics(hdc, &textMetric);
GetClientRect(hWnd, &rect); int rWidth = rect.right - rect.left,
rHeight = rect.bottom - rect.top,
cxChar = textMetric.tmAveCharWidth,
cyChar = textMetric.tmHeight + textMetric.tmExternalLeading,
bWidth = 10 * cxChar,
bHeight = 8 * cyChar / 4,
waveAxis = rHeight / 2; SetWindowPos(hBtn, HWND_TOP, rWidth / 2 - bWidth / 2,
cyChar, bWidth, bHeight, SWP_SHOWWINDOW | SWP_NOREDRAW);
HGDIOBJ hPen = SelectObject(hdc, GetStockObject(DC_PEN));
SetDCPenColor(hdc, RGB(0, 245, 0));
for (int i = 0; i < rWidth - 1; i++)
{
MoveToEx(hdc, 1 + 4 * i, audioData[i] + waveAxis, NULL);
LineTo(hdc, 1 + 4 * (i + 1), audioData[i + 1] + waveAxis);
}
SelectObject(hdc, hPen);
DeleteObject(hPen);
EndPaint(hWnd, &ps);
}
break;
case MM_WIM_OPEN:
{
ZeroMemory(&wavehdr, sizeof(wavehdr));
ZeroMemory(waveBuffer, sizeof(waveBuffer));
wavehdr.lpData = (LPSTR)waveBuffer;
wavehdr.dwBufferLength = sizeof(waveBuffer);
wavehdr.dwFlags = 0;
wavehdr.dwLoops = 0;
wavehdr.dwUser = 0;
if (waveInPrepareHeader(hWaveIn, &wavehdr, sizeof(wavehdr)) != MMSYSERR_NOERROR)
{
MessageBox(hWnd, L"Could not prepare wave header",
L"Warning Message", MB_ICONWARNING | MB_OK);
waveInClose(hWaveIn);
break;
}
if (waveInStart(hWaveIn) != MMSYSERR_NOERROR)
{
MessageBox(hWnd, L"Could not start input device",
L"Warning Message", MB_ICONWARNING | MB_OK);
waveInClose(hWaveIn);
break;
}
if (waveInAddBuffer(hWaveIn, &wavehdr, sizeof(wavehdr)) != MMSYSERR_NOERROR)
{
MessageBox(hWnd, L"Could not add buffer",
L"Warning Message", MB_ICONWARNING | MB_OK);
waveInClose(hWaveIn);
break;
}
SetWindowText(hBtn, TEXT("暂停"));
}
break;
case MM_WIM_DATA:
{
RECT rect;
GetClientRect(hWnd, &rect);
int width = rect.right - rect.left; for (int i = 0; i < width; i++)
audioData[i] = wavehdr.lpData[i + audioSum];
audioSum += 4; RedrawWindow(hWnd, NULL, NULL, RDW_INVALIDATE);
HBRUSH wb = (HBRUSH)GetStockObject(BLACK_BRUSH);
RECT r = rect;
HDC hdc = GetDC(hWnd);
r.top += 56;
FillRect(hdc, &r, wb);
ReleaseDC(hWnd, hdc);
if (audioSum > RATE * PIPE)
{
audioSum = 0;
ZeroMemory(waveBuffer, sizeof(waveBuffer));
}
waveInAddBuffer(hWaveIn, &wavehdr, sizeof(wavehdr));
}
break;
case MM_WIM_CLOSE:
{
playing = FALSE;
}
break;
case WM_DESTROY:
{
if (playing)
{
waveInStop(hWaveIn);
waveInClose(hWaveIn);
}
PostQuitMessage(0);
}
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
} INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
UNREFERENCED_PARAMETER(lParam);
switch (message)
{
case WM_INITDIALOG:
return (INT_PTR)TRUE; case WM_COMMAND:
if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
{
EndDialog(hDlg, LOWORD(wParam));
return (INT_PTR)TRUE;
}
break;
}
return (INT_PTR)FALSE;
}
截图:

原截图:

win api 音频可视化的更多相关文章
- 黑客编程教程(二)Win API编程简介
第二节 Win API编程简介 下面介绍一下WIN API. 我们需要自己编写一个工具时,必然会用到很多操作windows和控制windows的函数,这些函数就是windows API. API是Ap ...
- HTML5 ——web audio API 音乐可视化(二)
上一篇 web audio API 音乐可视化(一)介绍了一些基本的API,以及如何简单的播放一个音频,本篇介绍一下怎么对获取到的音频进行分析,并将分析后的数据绘制成图像. 最终效果请戳这里; 完整版 ...
- HTML5 ——web audio API 音乐可视化(一)
使用Web Audio API可以对音频进行分析和操作,最终实现一个音频可视化程序. 最终效果请戳这里; 完整版代码请戳这里,如果还看得过眼,请给一个start⭐ 一.API AudioContext ...
- HTML5音频可视化频谱跳动代码
今天学习到用canvas来写 HTML5音频可视化频谱跳动代码 将代码在此做一总结: <!DOCTYPE html> <html lang="en"> ...
- 用webAudio和canvas实现音频可视化
前两天遇到了要显示音频波形图的需求,因为时间紧,就直接用了wavesufer.js,这两天有空,就研究了一下怎么用webAudio实现音频的可视化. 大致流程是对音源进行解析,解析得到的数据是个频谱数 ...
- C#用WebBrowser与WIN API辅助模拟获取网站完整Cookie
网上找到的可以完整获取Cookie的方法,转载一下希望能帮助更多人. 亲测可用 在Winform中使用WebBrowser控件获取网站的Cookie有时候是不完整的,默认调用Document.Cook ...
- 文件操作(CRT、C++、WIN API、MFC)
一.使用CRT函数文件操作 二.使用标准C++库 std::fstream std::string 1)std::string对象内部存储了一个C的字符串,以'\0'结尾的. 2)std::strin ...
- SwaggerUI+SpringMVC——构建RestFul API的可视化界面
今天给大家介绍一款工具,这个工具眼下可预见的优点是:自己主动维护最新的接口文档. 我们都知道,接口文档是非常重要的,可是随着代码的不断更新,文档却非常难持续跟着更新,今天要介绍的工具,完美的攻克了这个 ...
- C# 调用win api获取chrome浏览器中地址
//FindWindow 查找窗口 //FindWindowEx查找子窗口 //EnumWindows列举屏幕上的所有顶层窗口,如果回调函数成功则返回非零,失败则返回零 //GetWindowText ...
随机推荐
- 最长公共子串(LCS) lg SP1811
后缀自动机的一大用处就是求最长公共子串了 这道题的话题意就是给你两个字符串,求最长公共子串 做法的话是先使用一个字符串建立SAM,然后让另一个串在上面进行匹配 匹配的策略是优先匹配当前节点的下一个字符 ...
- vue学习指南:第十三篇(详细) - Vue的 路由 第三篇 ( 路由的缓存 )
路由的缓存 路由缓存是 Vue组件优化的一个重要方法 为什么实现路由缓存? 为了 组件间 相互切换不会重复加载数据,影响用户体验,我们通常需要将组件的数组实现缓存,当我们点过来,在点的时候会再次发送 ...
- 16day 路径信息系列
../ 上一级目录 ./ 当前路径 ~ 返回到家目录 - 两个目录之间进行快速切换 An argument of - is equivalent to $OLDPWD(环境变量) 补充说明: [roo ...
- New Skateboard
Max wants to buy a new skateboard. He has calculated the amount of money that is needed to buy a new ...
- pytest学习1-安装和入门
一.安装pytest: 在命令行窗口下执行: pip install -U pytest 检查安装的pytest版本: pytest -v 二.运行第一个测试用例: import pytest def ...
- 计算a除b的第一位小数 in C++.
my codes: #include<iostream> #include<cstdio> using namespace std; int main() { int a,b; ...
- .Net Core初体验
对于C#语言支持(由C#1.0-C#7.1): 编码可以使用跨平台的IDE选择,就如同VS+Resharper一样方便: 运行效果:
- MySQL条件查询
语法: ①SELECT 查询列表(可以包括:字段.表达式.常量值.几个拼在一起的,构成的表) ②FROM 表名(原始表) ③WHERE (理解为当...筛选条件=TRUE或筛选条件=FALSE) 筛选 ...
- Git的基本使用 -- 分支管理
查看分支 git branch 前面带 * 的为当前所在分支 创建分支 git branch 分支名 切换分支 git checkout 分支名 创建并切换到此分支 git checkout -b 分 ...
- jdk之java.lang.Integer源码理解
基本数据类型的包装类java.lang.Integer是我们频繁使用的一个系统类,那么通过一个示例反应出的几个问题来深入理解一下此类的源码. 需求:实现Integer类型的两个数值交换. packag ...