windows下c++扫描连接wifi_实现用c++在win7下,扫描可用的本地连接-程序员宅基地

技术标签: Qt  qt  windows wifi  

Windows下使用Wifi Native Api在应用程序内部控制wifi,官方文档链接https://docs.microsoft.com/zh-cn/windows/win32/nativewifi/native-wifi-api-sample,主要注意以下几点:

  • 连接wifi使用wlan_connection_mode_profile时,如果被连接的wifi有配置文件,那么连接参数直接使用配置文件名称即可,如果没有配置文件,要先WlanSetProfile成功后才能创建连接,不同的加密方式有不同的配置文件样本,常用的WPA2样本WPA2-Personal Profile Sample
  • 官方的配置文件样本有个坑,官方的xml demo里面命名空间是https,实际使用的是http才能WlanSetProfile成功,否则会报ERROR_BAD_PROFILE错误
  • 连接wifi前需要断开当前已经连接的wifi
  • WlanRegisterNotification的回调函数接收通知是系统全局的,也就是说在系统设置里面操作wifi也能通知回调,并且是异步回调

下面是根据官方文档和demo写的demo,结合Qt
WifiHelper.h

#pragma once

#include <Windows.h>
#include <wlanapi.h>
#include <objbase.h>
#include <wtypes.h>

#pragma comment(lib, "ole32.lib")
#pragma comment(lib, "wlanapi.lib")

#include <qobject.h>

struct EntryInfo {
    
    QString profile;
    QString ssid;
    bool connectable;
    long signalQuality;//信号质量
    int rssi;
    bool securityEnabled;//启动网络安全
    QString authAlgorithm;//认证算法
    QString cipherAlgorithm;//加密算法
    DWORD dwFlags;
    QString status;//当前状态
    //
    DOT11_SSID dot11_ssid;
    DOT11_BSS_TYPE dot11BssType;
    DOT11_AUTH_ALGORITHM dot11DefaultAuthAlgorithm;
    DOT11_CIPHER_ALGORITHM dot11DefaultCipherAlgorithm;
};

struct WifiInfo {
    
    QString description;
    GUID guid;
    QString guidStr;
    QString status;
    QList<EntryInfo> entryList;
};

class WifiHelper : public QObject {
    
    Q_OBJECT

public:
    WifiHelper(QString profileTemplateStr, QObject* parent = nullptr);
    ~WifiHelper();

    void loadWifiInfo(); 
    void scanWifiList(int interfaceIndex);
    void reloadWifiList(int interfaceIndex);
    void connectWifi(int interfaceIndex, int entryInfoIndex, QString passwordIfNeed);
    void disconnectWifi(int interfaceIndex);

signals:
    void getNewInterfaceGuid(QString guidStr);
    void interfaceListRefreshed(const QList<WifiInfo>& wifiList);
    void wifiListRefreshed(const WifiInfo& wifiInfo);
    void printErr(QString title, QString content);

private:
    QList<WifiInfo> wifiList;
    HANDLE hClient = NULL;
    PWLAN_INTERFACE_INFO_LIST pIfList = NULL;
    QString profileTemplateStr;

private:
    void findActiveWireless(WifiInfo& wifiInfo);
    void connectWifi(const GUID& interfaceGuid, const EntryInfo& entryInfo, QString password);
    void disconnectWifi(const GUID& interfaceGuid);
    bool setProfile(const GUID& interfaceGuid, const EntryInfo& entryInfo, QString password);
    QString getProfileStr(const EntryInfo& entryInfo, QString password);
    QString getProfileStr(const GUID& interfaceGuid, QString profileName);

private slots:
    void reloadWifiListInMainThread(QString guidStr);
};

WifiHelper.cpp

#include "WifiHelper.h"
#include <qdebug.h>

void WlanNotificationCallback(PWLAN_NOTIFICATION_DATA Arg1, PVOID Arg2) {
    
    if (Arg1 != NULL) {
    
        switch (Arg1->NotificationSource) {
    
            case WLAN_NOTIFICATION_SOURCE_ACM:
                const auto wifi_list_refresh = [&] {
    
                    auto obj = static_cast<WifiHelper*>(Arg2);
                    if (obj != nullptr) {
    
                        WCHAR GuidString[39] = {
     0 };
                        int iRet = StringFromGUID2(Arg1->InterfaceGuid, (LPOLESTR)&GuidString, sizeof(GuidString) / sizeof(*GuidString));
                        obj->getNewInterfaceGuid(QString::fromStdWString(GuidString));
                    }
                };
                switch (Arg1->NotificationCode) {
    
                    case wlan_notification_acm_connection_complete:
                        {
    
                            if (Arg1->dwDataSize < sizeof(WLAN_CONNECTION_NOTIFICATION_DATA)) {
    
                                break;
                            }
                            auto data = (PWLAN_CONNECTION_NOTIFICATION_DATA)Arg1->pData;
                            QString ssid = QByteArray((char*)data->dot11Ssid.ucSSID, data->dot11Ssid.uSSIDLength);
                            if (data->wlanReasonCode == WLAN_REASON_CODE_SUCCESS) {
    
                                qDebug() << QString("%1 connection successed!").arg(ssid);
                                wifi_list_refresh();
                            } else {
    
                                wchar_t reasonCodeStr[1024] = {
     0 };
                                WlanReasonCodeToString(data->wlanReasonCode, 1024, reasonCodeStr, NULL);
                                auto obj = static_cast<WifiHelper*>(Arg2);
                                if (obj != nullptr) {
    
                                    obj->printErr(QString("%1 connection failed!").arg(ssid), QString::fromWCharArray(reasonCodeStr));
                                }
                            }
                        }
                        break;
                    case wlan_notification_acm_scan_complete:
                    case wlan_notification_acm_disconnected:
                        wifi_list_refresh();
                        break;
                    default:
                        break;
                }
                break;
        }
    }
    qDebug() << "notifycode = " << Arg1->NotificationCode;
}

WifiHelper::WifiHelper(QString profileTemplateStr, QObject* parent)
    : QObject(parent) 
    , profileTemplateStr(profileTemplateStr)
    , pIfList(NULL)
{
    
    DWORD dwMaxClient = 2;
    DWORD dwCurVersion = 0;
    DWORD dwResult = WlanOpenHandle(dwMaxClient, NULL, &dwCurVersion, &hClient);
    Q_ASSERT(dwResult == ERROR_SUCCESS);

    DWORD dwNotifSourcePre;
    dwResult = WlanRegisterNotification(hClient, WLAN_NOTIFICATION_SOURCE_ACM, TRUE, (WLAN_NOTIFICATION_CALLBACK)WlanNotificationCallback, this, NULL, &dwNotifSourcePre);

    connect(this, &WifiHelper::getNewInterfaceGuid, this, &WifiHelper::reloadWifiListInMainThread);
}


WifiHelper::~WifiHelper() {
    
    if (pIfList != NULL) {
    
        WlanFreeMemory(pIfList);
        pIfList = NULL;
    }
    WlanCloseHandle(hClient, NULL);
}

void WifiHelper::loadWifiInfo() {
    
    wifiList.clear();

    if (pIfList != NULL) {
    
        WlanFreeMemory(pIfList);
        pIfList = NULL;
    }

    DWORD dwResult = WlanEnumInterfaces(hClient, NULL, &pIfList);
    Q_ASSERT(dwResult == ERROR_SUCCESS);

    for (int i = 0; i < (int)pIfList->dwNumberOfItems; i++) {
    
        PWLAN_INTERFACE_INFO pIfInfo = (WLAN_INTERFACE_INFO*)&pIfList->InterfaceInfo[i];

        WifiInfo wifiInfo;
        wifiInfo.description = QString::fromStdWString(pIfInfo->strInterfaceDescription);

        WCHAR GuidString[39] = {
     0 };
        int iRet = StringFromGUID2(pIfInfo->InterfaceGuid, (LPOLESTR)&GuidString, sizeof(GuidString) / sizeof(*GuidString));
        wifiInfo.guid = pIfInfo->InterfaceGuid;
        wifiInfo.guidStr = iRet == 0 ? "StringFromGUID2 failed" : QString::fromStdWString(GuidString);

        switch (pIfInfo->isState) {
    
            case wlan_interface_state_not_ready:
                wifiInfo.status = "Not ready";
                break;
            case wlan_interface_state_connected:
                wifiInfo.status = "Connected";
                break;
            case wlan_interface_state_ad_hoc_network_formed:
                wifiInfo.status = "First node in a ad hoc network";
                break;
            case wlan_interface_state_disconnecting:
                wifiInfo.status = "Disconnecting";
                break;
            case wlan_interface_state_disconnected:
                wifiInfo.status = "Not connected";
                break;
            case wlan_interface_state_associating:
                wifiInfo.status = "Attempting to associate with a network";
                break;
            case wlan_interface_state_discovering:
                wifiInfo.status = "Auto configuration is discovering settings for the network";
                break;
            case wlan_interface_state_authenticating:
                wifiInfo.status = "In process of authenticating";
                break;
            default:
                wifiInfo.status = "Unknown state " + QString::number(pIfInfo->isState);
                break;
        }
        wifiList << wifiInfo;
    }

    findActiveWireless(wifiList[0]);
    wifiListRefreshed(wifiList.at(0));
    interfaceListRefreshed(wifiList);
}

void WifiHelper::scanWifiList(int interfaceIndex) {
    
    WLAN_RAW_DATA wlanRawData = {
     0 };
    DWORD dwResult = WlanScan(hClient, &wifiList[interfaceIndex].guid, NULL, &wlanRawData, NULL);
    Q_ASSERT(dwResult == ERROR_SUCCESS);
}

void WifiHelper::reloadWifiList(int interfaceIndex) {
    
    findActiveWireless(wifiList[interfaceIndex]);
    wifiListRefreshed(wifiList.at(interfaceIndex));
}

void WifiHelper::connectWifi(int interfaceIndex, int entryInfoIndex, QString passwordIfNeed) {
    
    connectWifi(wifiList.at(interfaceIndex).guid,
                wifiList.at(interfaceIndex).entryList.at(entryInfoIndex),
                passwordIfNeed
    );
}

void WifiHelper::disconnectWifi(int interfaceIndex) {
    
    disconnectWifi(wifiList.at(interfaceIndex).guid);
}

void WifiHelper::reloadWifiListInMainThread(QString guidStr) {
    
    for (int i = 0; i < wifiList.size(); i++) {
    
        if (wifiList.at(i).guidStr == guidStr) {
    
            findActiveWireless(wifiList[i]);
            wifiListRefreshed(wifiList.at(i));
            break;
        }
    }
}

void WifiHelper::connectWifi(const GUID& interfaceGuid, const EntryInfo& entryInfo, QString password) {
    
    if (entryInfo.dwFlags & WLAN_AVAILABLE_NETWORK_CONNECTED) {
    
        return;
    }

    disconnectWifi(interfaceGuid);

    wchar_t profile[WLAN_MAX_NAME_LENGTH] = {
    };
    if (!(entryInfo.dwFlags & WLAN_AVAILABLE_NETWORK_HAS_PROFILE)) {
    
        setProfile(interfaceGuid, entryInfo, password);
    }
    entryInfo.ssid.toWCharArray(profile);
    auto fstr = getProfileStr(interfaceGuid, entryInfo.ssid);

    WLAN_CONNECTION_PARAMETERS parameter;
    parameter.strProfile = profile;
    parameter.pDot11Ssid = NULL;
    parameter.pDesiredBssidList = NULL;
    parameter.wlanConnectionMode = wlan_connection_mode_profile;
    parameter.dot11BssType = entryInfo.dot11BssType;
    parameter.dwFlags = WLAN_CONNECTION_HIDDEN_NETWORK;

    DWORD dwResult = WlanConnect(hClient, &interfaceGuid, &parameter, NULL);
    if (dwResult != ERROR_SUCCESS) {
    
        if (dwResult == ERROR_INVALID_PARAMETER) {
    
            printErr(u8"无法连接", u8"不支持的连接!修改profile模板!");
        } else {
    
            Q_ASSERT(dwResult == ERROR_SUCCESS);
        }
    }
}

void WifiHelper::disconnectWifi(const GUID & interfaceGuid) {
    
    DWORD dwResult = WlanDisconnect(hClient, &interfaceGuid, NULL);
    Q_ASSERT(dwResult == ERROR_SUCCESS);
}

bool WifiHelper::setProfile(const GUID& interfaceGuid, const EntryInfo& entryInfo, QString password) {
    
    wchar_t profile[2048] = {
     0 };
    QString profileStr = getProfileStr(entryInfo, password);
    profileStr.toWCharArray(profile);
    DWORD dwReasonCode;
    wchar_t reasonCodeStr[1024] = {
     0 };
    DWORD dwResult = WlanSetProfile(hClient, &interfaceGuid, 0, profile, NULL, TRUE, NULL, &dwReasonCode);
    if (dwResult != ERROR_SUCCESS) {
    
        WlanReasonCodeToString(dwReasonCode, 1024, reasonCodeStr, NULL);
    }
    return dwResult == ERROR_SUCCESS;
}

QString WifiHelper::getProfileStr(const EntryInfo& entryInfo, QString password) {
    
    QString templateContent = profileTemplateStr;
    templateContent.replace("{ssid}", entryInfo.ssid);
    templateContent.replace("{password}", password);
    switch (entryInfo.dot11DefaultAuthAlgorithm) {
    
        case DOT11_AUTH_ALGO_80211_OPEN:
            templateContent.replace("{authentication}", "open");
            break;
        case DOT11_AUTH_ALGO_80211_SHARED_KEY:
            templateContent.replace("{authentication}", "shared");
            break;
        case DOT11_AUTH_ALGO_WPA:
            templateContent.replace("{authentication}", "WPA");
            break;
        case DOT11_AUTH_ALGO_WPA_PSK:
            templateContent.replace("{authentication}", "WPAPSK");
            break;
        case DOT11_AUTH_ALGO_WPA_NONE:
            templateContent.replace("{authentication}", "none");
            break;
        case DOT11_AUTH_ALGO_RSNA:
            templateContent.replace("{authentication}", "WPA2");
            break;
        case DOT11_AUTH_ALGO_RSNA_PSK:
            templateContent.replace("{authentication}", "WPA2PSK");
            break;
        default:
            break;
    }
    switch (entryInfo.dot11DefaultCipherAlgorithm) {
    
        case DOT11_CIPHER_ALGO_NONE:
            templateContent.replace("{encryption}", "none");
            break;
        case DOT11_CIPHER_ALGO_WEP40:
            templateContent.replace("{encryption}", "WEP");
            break;
        case DOT11_CIPHER_ALGO_TKIP:
            templateContent.replace("{encryption}", "TKIP");
            break;
        case DOT11_CIPHER_ALGO_CCMP:
            templateContent.replace("{encryption}", "AES");
            break;
        case DOT11_CIPHER_ALGO_WEP104:
            templateContent.replace("{encryption}", "WEP");
            break;
        case DOT11_CIPHER_ALGO_WEP:
            templateContent.replace("{encryption}", "WEP");
            break;
        default:
            break;
    }
    return templateContent;
}

QString WifiHelper::getProfileStr(const GUID & interfaceGuid, QString profileName) {
    
    wchar_t profile[WLAN_MAX_NAME_LENGTH] = {
    };
    profileName.toWCharArray(profile);
    DWORD dwFlags, dwGrantedAccess;
    LPWSTR pProfileXml = NULL;
    DWORD dwResult = WlanGetProfile(hClient, &interfaceGuid, profile, NULL, &pProfileXml, &dwFlags, &dwGrantedAccess);
    if (dwResult == ERROR_SUCCESS) {
    
        return QString::fromWCharArray(pProfileXml);
    }
    return QString();
}

void WifiHelper::findActiveWireless(WifiInfo& wifiInfo) {
    

    PWLAN_AVAILABLE_NETWORK_LIST pBssList = NULL;
    PWLAN_AVAILABLE_NETWORK pBssEntry = NULL;

    wifiInfo.entryList.clear();
    DWORD dwResult = WlanGetAvailableNetworkList(hClient, &wifiInfo.guid, 0, NULL, &pBssList);
    if (dwResult == ERROR_SUCCESS) {
    
        QHash<QString, EntryInfo> entryList;
        for (int j = 0; j < pBssList->dwNumberOfItems; j++) {
    
            pBssEntry = (WLAN_AVAILABLE_NETWORK *)& pBssList->Network[j];

            EntryInfo entryInfo;
            entryInfo.profile = QString::fromWCharArray(pBssEntry->strProfileName);
            entryInfo.dot11_ssid = pBssEntry->dot11Ssid;
            entryInfo.ssid = QByteArray((char*)pBssEntry->dot11Ssid.ucSSID, pBssEntry->dot11Ssid.uSSIDLength);
            entryInfo.connectable = pBssEntry->bNetworkConnectable;
            entryInfo.signalQuality = pBssEntry->wlanSignalQuality;

            int iRSSI = 0;
            if (pBssEntry->wlanSignalQuality == 0)
                iRSSI = -100;
            else if (pBssEntry->wlanSignalQuality == 100)
                iRSSI = -50;
            else
                iRSSI = -100 + (pBssEntry->wlanSignalQuality / 2);

            entryInfo.rssi = iRSSI;
            entryInfo.dot11BssType = pBssEntry->dot11BssType;
            entryInfo.securityEnabled = pBssEntry->bSecurityEnabled;

            entryInfo.dot11DefaultAuthAlgorithm = pBssEntry->dot11DefaultAuthAlgorithm;
            switch (pBssEntry->dot11DefaultAuthAlgorithm) {
    
                case DOT11_AUTH_ALGO_80211_OPEN:
                    entryInfo.authAlgorithm = "802.11 Open";
                    break;
                case DOT11_AUTH_ALGO_80211_SHARED_KEY:
                    entryInfo.authAlgorithm = "802.11 Shared";
                    break;
                case DOT11_AUTH_ALGO_WPA:
                    entryInfo.authAlgorithm = "WPA";
                    break;
                case DOT11_AUTH_ALGO_WPA_PSK:
                    entryInfo.authAlgorithm = "WPA-PSK";
                    break;
                case DOT11_AUTH_ALGO_WPA_NONE:
                    entryInfo.authAlgorithm = "WPA-None";
                    break;
                case DOT11_AUTH_ALGO_RSNA:
                    entryInfo.authAlgorithm = "RSNA";
                    break;
                case DOT11_AUTH_ALGO_RSNA_PSK:
                    entryInfo.authAlgorithm = "RSNA with PSK";
                    break;
                default:
                    entryInfo.authAlgorithm = "Other";
                    break;
            }

            entryInfo.dot11DefaultCipherAlgorithm = pBssEntry->dot11DefaultCipherAlgorithm;
            switch (pBssEntry->dot11DefaultCipherAlgorithm) {
    
                case DOT11_CIPHER_ALGO_NONE:
                    entryInfo.cipherAlgorithm = "None";
                    break;
                case DOT11_CIPHER_ALGO_WEP40:
                    entryInfo.cipherAlgorithm = "WEP-40";
                    break;
                case DOT11_CIPHER_ALGO_TKIP:
                    entryInfo.cipherAlgorithm = "TKIP";
                    break;
                case DOT11_CIPHER_ALGO_CCMP:
                    entryInfo.cipherAlgorithm = "CCMP";
                    break;
                case DOT11_CIPHER_ALGO_WEP104:
                    entryInfo.cipherAlgorithm = "WEP-104";
                    break;
                case DOT11_CIPHER_ALGO_WEP:
                    entryInfo.cipherAlgorithm = "WEP";
                    break;
                default:
                    entryInfo.cipherAlgorithm = "Other";
                    break;
            }
            entryInfo.dwFlags = pBssEntry->dwFlags;
            if (pBssEntry->dwFlags & WLAN_AVAILABLE_NETWORK_CONNECTED) {
    
                entryInfo.status = u8"已连接";
            } else if (pBssEntry->dwFlags & WLAN_AVAILABLE_NETWORK_HAS_PROFILE) {
    
                entryInfo.status = u8"已保存";
            } else {
    
                entryInfo.status = u8"未连接";
                if (entryList.contains(entryInfo.ssid)) {
    
                    continue;
                }
            }
            entryList.insert(entryInfo.ssid, entryInfo);
        }
        for (EntryInfo info : entryList.values()) {
    
            wifiInfo.entryList << info;
        }

    } else {
    
        qDebug() << "WlanGetAvailableNetworkList failed with error: " << dwResult;
    }

    if (pBssList != NULL) {
    
        WlanFreeMemory(pBssList);
        pBssList = NULL;
    }
}

上面是工具类,下面使用一个widget显示和交互

//WifiConnectTest.h
#pragma once

#include <QtWidgets/QWidget>
#include "ui_WifiConnectTest.h"

#include <qstandarditemmodel.h>

#include "WifiHelper.h"

class WifiConnectTest : public QWidget
{
    
    Q_OBJECT

public:
    WifiConnectTest(QWidget *parent = Q_NULLPTR);
    ~WifiConnectTest();

private:
    Ui::WifiConnectTestClass ui; 
    WifiHelper* wifiHelper;

    void addTableHeader(QStandardItemModel* model);

private slots:
    void loadWifiEntryList(const WifiInfo& wifiInfo);
    void loadWifiInterfaceList(const QList<WifiInfo>& wifiList);
    void showErrMessageBox(QString title, QString content);
};

WifiConnectTest.cpp
#include "WifiConnectTest.h"
#include <qfile.h>
#include <qdebug.h>
#include <qmessagebox.h>

WifiConnectTest::WifiConnectTest(QWidget *parent)
    : QWidget(parent)
{
    
    ui.setupUi(this);

    auto model = new QStandardItemModel(this);
    ui.tableView->setModel(model);
    ui.tableView->horizontalHeader()->setSectionResizeMode(QHeaderView::Stretch);
    ui.tableView->setSelectionBehavior(QAbstractItemView::SelectRows);
    ui.tableView->setSelectionMode(QAbstractItemView::SingleSelection);
    addTableHeader(model);

    connect(ui.scan_adapter, &QPushButton::clicked, [&]() {
    
        wifiHelper->loadWifiInfo();
    });
    
    connect(ui.scan_wifi, &QPushButton::clicked, [&]() {
    
        int currentAdapterIndex = ui.interfaceDescription->currentIndex();
        if (currentAdapterIndex >= 0) {
    
            wifiHelper->scanWifiList(currentAdapterIndex);
        }
    });

    connect(ui.interfaceDescription, qOverload<int>(&QComboBox::currentIndexChanged), [&](int index) {
    
        if (index >= 0) {
    
            wifiHelper->reloadWifiList(index);
        }
    });

    connect(ui.connect_tag, &QPushButton::clicked, [&]() {
    
        int interfaceIndex = ui.interfaceDescription->currentIndex();
        if (interfaceIndex >= 0) {
    
            int row = ui.tableView->currentIndex().row();
            if (row >= 0) {
    
                wifiHelper->connectWifi(interfaceIndex, row,
                                        ui.tableView->model()->index(row, ui.tableView->model()->columnCount() - 1).data(Qt::DisplayRole).toString());
            }
        }
    });

    connect(ui.disconnect_tag, &QPushButton::clicked, [&]() {
    
        int interfaceIndex = ui.interfaceDescription->currentIndex();
        if (interfaceIndex >= 0) {
    
            wifiHelper->disconnectWifi(interfaceIndex);
        }
    });

    QFile file(":/WifiConnectTest/Resources/profile_template.xml");
    file.open(QIODevice::ReadOnly);
    QString templateContent = file.readAll();
    file.close();

    wifiHelper = new WifiHelper(templateContent, this);
    connect(wifiHelper, &WifiHelper::interfaceListRefreshed, this, &WifiConnectTest::loadWifiInterfaceList);
    connect(wifiHelper, &WifiHelper::wifiListRefreshed, this, &WifiConnectTest::loadWifiEntryList);
    connect(wifiHelper, &WifiHelper::printErr, this, &WifiConnectTest::showErrMessageBox);

}

WifiConnectTest::~WifiConnectTest() {
    
    
}

void WifiConnectTest::addTableHeader(QStandardItemModel * model) {
    
    QStringList titles;
    titles << "ssid" << u8"是否可连接" << u8"信号质量" << "rssi" << u8"启动网络安全" << u8"认证算法" << u8"加密算法" << u8"状态" << u8"密码";
    model->setColumnCount(titles.size());
    for (int i = 0; i < titles.size(); i++) {
    
        model->setHeaderData(i, Qt::Horizontal, titles[i], Qt::DisplayRole);
    }
}

void WifiConnectTest::loadWifiEntryList(const WifiInfo& wifiInfo) {
    
    auto tabModel = static_cast<QStandardItemModel*>(ui.tableView->model());
    tabModel->clear();
    addTableHeader(tabModel);

    ui.interfaceStatus->setText(wifiInfo.status);
    for (int i = 0; i < wifiInfo.entryList.size(); i++) {
    
        auto entryInfo = wifiInfo.entryList.at(i);

        QStandardItem* item;
        int col = 0;

        item = new QStandardItem(entryInfo.ssid);
        tabModel->setItem(i, col++, item);

        item = new QStandardItem(entryInfo.connectable ? "yes" : "no");
        tabModel->setItem(i, col++, item);

        item = new QStandardItem(QString::number(entryInfo.signalQuality));
        tabModel->setItem(i, col++, item);

        item = new QStandardItem(QString("%1 dBm").arg(entryInfo.rssi));
        tabModel->setItem(i, col++, item);

        item = new QStandardItem(entryInfo.securityEnabled ? "yes" : "no");
        tabModel->setItem(i, col++, item);

        item = new QStandardItem(entryInfo.authAlgorithm);
        tabModel->setItem(i, col++, item);

        item = new QStandardItem(entryInfo.cipherAlgorithm);
        tabModel->setItem(i, col++, item);

        item = new QStandardItem(entryInfo.status);
        tabModel->setItem(i, col++, item);

        item = new QStandardItem;
        tabModel->setItem(i, col++, item);
    }
}

void WifiConnectTest::loadWifiInterfaceList(const QList<WifiInfo>& wifiList) {
    
    ui.interfaceDescription->clear();
    for (const auto& i : wifiList) {
    
        ui.interfaceDescription->addItem(i.description);
    }
}

void WifiConnectTest::showErrMessageBox(QString title, QString content) {
    
    QMessageBox::warning(NULL, title, content);
}

界面:
界面

demo工程在这里下载

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/baidu_30570701/article/details/103207045

智能推荐

c# 调用c++ lib静态库_c#调用lib-程序员宅基地

文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib

deepin/ubuntu安装苹方字体-程序员宅基地

文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang

html表单常见操作汇总_html表单的处理程序有那些-程序员宅基地

文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些

PHP设置谷歌验证器(Google Authenticator)实现操作二步验证_php otp 验证器-程序员宅基地

文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器

【Python】matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距-程序员宅基地

文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距

docker — 容器存储_docker 保存容器-程序员宅基地

文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器

随便推点

网络拓扑结构_网络拓扑csdn-程序员宅基地

文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn

JS重写Date函数,兼容IOS系统_date.prototype 将所有 ios-程序员宅基地

文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios

如何将EXCEL表导入plsql数据库中-程序员宅基地

文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql

Git常用命令速查手册-程序员宅基地

文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...

分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120-程序员宅基地

文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120

【C++缺省函数】 空类默认产生的6个类成员函数_空类默认产生哪些类成员函数-程序员宅基地

文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数

推荐文章

热门文章

相关标签