Пытаюсь вывести имя нового USB-девайса, когда его подключают. Проблема в том, что при вставке флешки messagebox появляется, но на нем ничего нет. Где-то давно находил в интернете: "Когда приходит WM_DEVICECHANGE, смотрим, чтобы wParam был равен DBT_DEVICEARRIVAL. В lParam тогда будет указатель на DEV_BROADCAST_DEVICEINTERFACE. Если в его dbcc_devicetype будет значение DBT_DEVTYP_DEVICEINTERFACE, тогда в dbcc_name будет имя устройства." Я пытался что-то написать и как-то получить этот указатель из lParam (см. код ниже), но этот указатель не содержит поля-имени. Вообщем, я понятия не имею что нужно написать после проверки wParam и просто написал примерный код. Прошу помощи с решением этой задачи.
Глобальная часть:
#include <windows.h>
//DEV_BROADCAST_DEVICEINTERFACE struct
#include "dbt.h"
//contains initguid.h and usbiodef.h
#include "GUIDheader.h"
HDEVNOTIFY NotificationHande;
DEV_BROADCAST_DEVICEINTERFACE DeviceStruct;
Оконная процедура:
LRESULT CALLBACK WndProc(HWND hWnd,UINT uMsg,WPARAM wParam,LPARAM lParam)
{
switch (uMsg)
{
case(WM_CREATE):
DeviceStruct.dbcc_size = sizeof(DeviceStruct);
DeviceStruct.dbcc_classguid = GUID_DEVINTERFACE_USB_DEVICE;
DeviceStruct.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE;
NotificationHande = RegisterDeviceNotification(hWnd, &DeviceStruct, DEVICE_NOTIFY_WINDOW_HANDLE | DEVICE_NOTIFY_ALL_INTERFACE_CLASSES);
break;
case (WM_DESTROY):
PostQuitMessage(0);
break;
case(WM_DEVICECHANGE):
if (wParam == DBT_DEVICEARRIVAL)
{
DEV_BROADCAST_HDR* NewDevice = (DEV_BROADCAST_HDR*)lParam;
if (NewDevice->dbch_devicetype == DBT_DEVTYP_DEVICEINTERFACE)
{
MessageBox(0,(LPCSTR)DeviceStruct.dbcc_name, (LPCSTR)"WOW", MB_OK);
}
}
break;
default:
return DefWindowProc(hWnd, uMsg, wParam, lParam);
}
return 0;
}
Основная функция:
int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
// Main window handle
HWND hMainWnd;
WNDCLASSEX wc;
wc.cbSize = sizeof(wc);
wc.style = CS_HREDRAW | CS_VREDRAW;
wc.lpfnWndProc = WndProc;
wc.lpszMenuName = NULL;
wc.lpszClassName = "WindowClass";
wc.cbWndExtra = NULL;
wc.cbClsExtra = NULL;
wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
wc.hInstance = hInstance;
wc.hIcon = LoadIcon(NULL, IDI_WINLOGO);
wc.hIconSm = LoadIcon(NULL, IDI_WINLOGO);
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
if (!RegisterClassEx(&wc))
{
MessageBox(0, "Cannot register class.", "Error", MB_OK);
return 0;
}
hMainWnd = CreateWindowEx(0, "WindowClass", "", WS_SYSMENU, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, 0, 0, hInstance, 0);
if (hMainWnd == 0)
{
MessageBox(0, "Cannot create window.", "Error", MB_OK);
return 0;
}
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
MSG message;
while (GetMessage(&message, NULL, 0, 0))
{
TranslateMessage(&message);
DispatchMessage(&message);
}
DestroyWindow(hMainWnd);
UnregisterClass(wc.lpszClassName, hInstance);
return 0;
}
У вас не из той структуры значение берется. Да и значение в структуре на самом деле не имя устройства, а путь к нему. Имя устройства еще нужно искать (проще всего, через реестр). Как-то так:
#include <Windows.h>
#include "dbt.h"
#include "initguid.h"
#include "usbiodef.h"
//...
HDEVNOTIFY NotificationHande;
DEV_BROADCAST_DEVICEINTERFACE DeviceStruct;
//получение имени устройства
bool GetDeviceName(const wchar_t* symlink, wchar_t* name, int cname) {
if(symlink == NULL) return false;
if (name == NULL) return false;
if (wcslen(symlink) < 10) return false;
//распарсим путь к устройству на составляющие
const wchar_t* pEnum = symlink + 3;
const wchar_t* pDeviceID = wcsstr(symlink, L"#");
if(pDeviceID == NULL) return false;
int lenEnum = pDeviceID - pEnum;
pDeviceID++;
const wchar_t* pInstID = wcsstr(pDeviceID, L"#");
if (pInstID == NULL) return false;
int lenDeviceID = pInstID - pDeviceID;
pInstID++;
const wchar_t* pEnd = wcsstr(pInstID, L"#");
if (pEnd == NULL) return false;
int lenInstID = pEnd - pInstID;
//соберем из них путь к ключу реестра
wchar_t path[1024] = L"SYSTEM\\CurrentControlSet\\Enum";
wcsncat_s(path, 1024, pEnum, lenEnum);
wcsncat_s(path, 1024, L"\\", 1);
wcsncat_s(path, 1024, pDeviceID, lenDeviceID);
wcsncat_s(path, 1024, L"\\", 1);
wcsncat_s(path, 1024, pInstID, lenInstID);
//попытаемся получить имя или описание устройства из реестра
HKEY hKey = NULL;
LONG lRes = RegOpenKeyExW(HKEY_LOCAL_MACHINE, path, 0, KEY_READ, &hKey);
if (ERROR_SUCCESS != lRes)
{
return false;
}
WCHAR szBuffer[1024] = L"";
DWORD dwBufferSize = sizeof(szBuffer);
lRes = RegQueryValueExW(hKey, L"FriendlyName", 0, NULL, (LPBYTE)szBuffer, &dwBufferSize);
if (ERROR_SUCCESS != lRes)
{
dwBufferSize = sizeof(szBuffer);
lRes = RegQueryValueExW(hKey, L"DeviceDesc", 0, NULL, (LPBYTE)szBuffer, &dwBufferSize);
}
if (ERROR_SUCCESS != lRes)
{
RegCloseKey(hKey);
return false;
}
wcscpy_s(name, cname, szBuffer);
return true;
}
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_CREATE:
{
DeviceStruct.dbcc_size = sizeof(DeviceStruct);
DeviceStruct.dbcc_classguid = GUID_DEVINTERFACE_USB_DEVICE;
DeviceStruct.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE;
NotificationHande = RegisterDeviceNotification(hWnd, &DeviceStruct, DEVICE_NOTIFY_WINDOW_HANDLE | DEVICE_NOTIFY_ALL_INTERFACE_CLASSES);
}
break;
case WM_DEVICECHANGE:
if (wParam == DBT_DEVICEARRIVAL)
{
DEV_BROADCAST_HDR* NewDevice = (DEV_BROADCAST_HDR*)lParam;
if (NewDevice->dbch_devicetype == DBT_DEVTYP_DEVICEINTERFACE)
{
DEV_BROADCAST_DEVICEINTERFACE* pDev = (DEV_BROADCAST_DEVICEINTERFACE*)lParam;
WCHAR name[1024] = L"";
if(GetDeviceName(pDev->dbcc_name, name, 1024) != false)
MessageBoxW(hWnd, name, L"Device", MB_OK);
else
MessageBoxW(hWnd, pDev->dbcc_name, L"Device", MB_OK);
}
}
break;
case WM_DESTROY:
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}
Неправильно отображаются символы в сообщениях об ошибках в терминале CLionСреди руководств в инете полезной информации не нашел, поэтому надеюсь...
заинтересовала одна вещь и возможность ее реализацииХочу создать меню выбора в Командной строке на C++, это должно выглядеть примерно так:
По условию задачи можно задать любой массив, так что я его рандомизировалСейчас только начал составлять код, при компиляции выдает кучу ошибок,...
В файле содержатся несколько прокси ip:portНе могу по циклу проверить каждый прокси на доступность к сайту