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 ...
随机推荐
- BFS-八数码问题与状态图搜索
在一个3*3的棋盘上放置编号为1~8的八个方块,每个占一格,另外还有一个空格.与空格相邻的数字方块可以移动到空格里.任务1:指定的初始棋局和目标棋局,计算出最少的移动步数:任务2:数出数码的移动序列. ...
- iloc与loc的区别
pandas.DataFrame.iloc iloc基于位置进行索引,主要是整数位置,也可以用布尔数组 iloc的输入可以是:单个整数.整数列表或数组.整数切片.布尔数组 pandas.DataFr ...
- Django---Django初始
现在测试行业慢慢的也需要存在代码能力了,从以前的点点点到通过工具进行测试接口,到目前的自动化测试,需要测试会代码的越来越多了.慢慢的测试开发这一行业出现在我们的面前,测试不仅仅会点点,还能自己帮助开发 ...
- 巨杉数据库入选年度Gartner Peer Insights报告,获得市场高度评价
Gartner Peer Insights 年度评选结果于近日出炉,在数据库管理系统市场报告中,巨杉数据库获得了总平均分4.7(满分5分)的成绩,在众多国际厂商中位居第三,是国内唯一一家入选的数据库厂 ...
- CSS的字体样式
CSS的字体样式 1. span标签(约定俗成:重要的东西用它括起来) 首选介绍一个约定俗成的东西:span标签.一般将想要突出的东西,比较重要的东西,用span标签括起来. 比如,”学习Java“这 ...
- [CF467C] George and Job - DP,前缀和
简单dp + 前缀和 你谷这乱标难度的风气真是-- #include <bits/stdc++.h> using namespace std; #define int long long ...
- socket 简单实现HTTP服务器
# -*- coding: utf-8 -*- # @Time : 2019-07-17 1:39 # @File : 网络socket实现http服务器.py # @Software: PyChar ...
- JavaScript 运算,流程控制和循环
算数运算符 算术运算符 描叙 运算符 实例 加 + 10 + 20 = 30 减 - 10 – 20 = -10 乘 * 10 * 20 = 600 除 / 10 / 20 = 0.5 取余数 % 返 ...
- CRC碰撞
循环冗余效验(Cyclic Redundancy Check, CRC) 是一种根据网络数据包或电脑文件等数据产生简短固定位数校验码的一种散列函数,主要用来检测或校验数据传输或者保存后可能出现的错误. ...
- Educational Codeforces Round 76 (Rated for Div. 2) D. Yet Another Monster Killing Problem
You play a computer game. In this game, you lead a party of mm heroes, and you have to clear a dunge ...