使用WMI更改亮度

时间:2017-11-16 15:15:45

标签: c++ wmi monitor brightness

我尝试过此Windows示例Getting WMI Data from a Remote Computer和其他Calling a Provider Method,并且都在我的计算机中正常运行。但是我尝试使用WMI来改变亮度,我在步骤6中遇到错误,在这部分:

    //Get the Next Object from the collection  
    hres = pEnum->Next(WBEM_INFINITE, //Timeout  
    1, //No of objects requested  
    &pObj, //Returned Object  
    &ulReturned /*No of object returned*/);

pEnum是负值。

重要提示:在我写这个问题时,我尝试使用不同的计算机,但我在所有计算机上都出错了,除非我使用笔记本电脑。因此,如何更改显示器的亮度?

我也意识到如果我去wbemtest.exe,在我的所有计算机中都存在类WmiMonitorBrightnessMethods和方法WmiSetBrightness,但只有在笔记本电脑中才有它的实例。事实上,在我点击类的视图对象的计算机中,我收到此消息(见图)

No instances of WmiMonitorBrightnessMethodsClass

这是我的代码:

#define _WIN32_DCOM

#include <iostream>
using namespace std;
#include <comdef.h>
#include <Wbemidl.h>

#pragma comment(lib, "wbemuuid.lib")

int main(int iArgCnt, char ** argv)
{
    HRESULT hres;

    // Step 1: --------------------------------------------------
    // Initialize COM. ------------------------------------------

    hres =  CoInitializeEx(0, COINIT_MULTITHREADED); 
    if (FAILED(hres))
    {
        cout << "Failed to initialize COM library. Error code = 0x" 
             << hex << hres << endl;
        return 1;                  // Program has failed.
    }

    // Step 2: --------------------------------------------------
    // Set general COM security levels --------------------------

    hres =  CoInitializeSecurity(
        NULL, 
        -1,                          // COM negotiates service
        NULL,                        // Authentication services
        NULL,                        // Reserved
        RPC_C_AUTHN_LEVEL_DEFAULT,   // Default authentication 
        RPC_C_IMP_LEVEL_IMPERSONATE, // Default Impersonation
        NULL,                        // Authentication info
        EOAC_NONE,                   // Additional capabilities 
        NULL                         // Reserved
        );


    if (FAILED(hres))
    {
        cout << "Failed to initialize security. Error code = 0x" 
             << hex << hres << endl;
        CoUninitialize();
        return 1;                      // Program has failed.
    }

    // Step 3: ---------------------------------------------------
    // Obtain the initial locator to WMI -------------------------

    IWbemLocator *pLoc = NULL;

    hres = CoCreateInstance(
        CLSID_WbemLocator,
        0, 
        CLSCTX_INPROC_SERVER, 
        IID_IWbemLocator, (LPVOID *) &pLoc);

    if (FAILED(hres))
    {
        cout << "Failed to create IWbemLocator object. "
             << "Err code = 0x"
             << hex << hres << endl;
        CoUninitialize();
        return 1;                 // Program has failed.
    }

    // Step 4: ---------------------------------------------------
    // Connect to WMI through the IWbemLocator::ConnectServer method

    IWbemServices *pSvc = NULL;

    // Connect to the local root\wminamespace
    // and obtain pointer pSvc to make IWbemServices calls.
    hres = pLoc->ConnectServer(
        _bstr_t(L"ROOT\\WMI"), 
        NULL,
        NULL, 
        0, 
        NULL, 
        0, 
        0, 
        &pSvc
    );

    if (FAILED(hres))
    {
        cout << "Could not connect. Error code = 0x" 
             << hex << hres << endl;
        pLoc->Release();
        CoUninitialize();
        return 1;                // Program has failed.
    }

    cout << "Connected to ROOT\\WMI namespace" << endl;


    // Step 5: --------------------------------------------------
    // Set security levels for the proxy ------------------------

    hres = CoSetProxyBlanket(
        pSvc,                        // Indicates the proxy to set
        RPC_C_AUTHN_WINNT,           // RPC_C_AUTHN_xxx 
        RPC_C_AUTHZ_NONE,            // RPC_C_AUTHZ_xxx 
        NULL,                        // Server principal name 
        RPC_C_AUTHN_LEVEL_CALL,      // RPC_C_AUTHN_LEVEL_xxx 
        RPC_C_IMP_LEVEL_IMPERSONATE, // RPC_C_IMP_LEVEL_xxx
        NULL,                        // client identity
        EOAC_NONE                    // proxy capabilities 
    );

    if (FAILED(hres))
    {
        cout << "Could not set proxy blanket. Error code = 0x" 
             << hex << hres << endl;
        pSvc->Release();
        pLoc->Release();
        CoUninitialize();
        return 1;               // Program has failed.
    }

    // Step 6: --------------------------------------------------
    // Call WmiSetBrightness method -----------------------------

    // set up to call the Win32_Process::Create method
    BSTR ClassName = SysAllocString(L"WmiMonitorBrightnessMethods");
    BSTR MethodName = SysAllocString(L"WmiSetBrightness");
    BSTR bstrQuery = SysAllocString(L"Select * from WmiMonitorBrightnessMethods");
    IEnumWbemClassObject *pEnum = NULL;

    hres = pSvc->ExecQuery(_bstr_t(L"WQL"), //Query Language  
    bstrQuery, //Query to Execute  
    WBEM_FLAG_FORWARD_ONLY | WBEM_FLAG_RETURN_IMMEDIATELY, //Make a semi-synchronous call  
    NULL, //Context  
    &pEnum /*Enumeration Interface*/);

    hres = WBEM_S_NO_ERROR;

    ULONG ulReturned;
    IWbemClassObject *pObj;
    DWORD retVal = 0;

    //Get the Next Object from the collection  
    hres = pEnum->Next(WBEM_INFINITE, //Timeout  
    1, //No of objects requested  
    &pObj, //Returned Object  
    &ulReturned /*No of object returned*/);

    IWbemClassObject* pClass = NULL;
    hres = pSvc->GetObject(ClassName, 0, NULL, &pClass, NULL);

    IWbemClassObject* pInParamsDefinition = NULL;
    hres = pClass->GetMethod(MethodName, 0, &pInParamsDefinition, NULL);

    IWbemClassObject* pClassInstance = NULL;
    hres = pInParamsDefinition->SpawnInstance(0, &pClassInstance);

    VARIANT var1;
    VariantInit(&var1);
    BSTR ArgName0 = SysAllocString(L"Timeout");

    V_VT(&var1) = VT_BSTR;
    V_BSTR(&var1) = SysAllocString(L"0");
    hres = pClassInstance->Put(ArgName0,
        0,
        &var1,
        CIM_UINT32); //CIM_UINT64  
    VariantClear(&var1);

    VARIANT var;
    VariantInit(&var);
    BSTR ArgName1 = SysAllocString(L"Brightness");

    V_VT(&var2) = VT_BSTR;
    V_BSTR(&var2) = SysAllocString(L"80"); //Brightness value
    hres = pClassInstance->Put(ArgName1,
        0,
        &var2,
        CIM_UINT8);
    VariantClear(&var2);

    // Call the method  
    VARIANT pathVariable;
    VariantInit(&pathVariable);

    hres = pSvc->ExecMethod(pathVariable.bstrVal,
        MethodName,
        0,
        NULL,
        pClassInstance,
        NULL,
        NULL);
    VariantClear(&pathVariable);


    return 0;
}

2 个答案:

答案 0 :(得分:1)

正如Amit Shakya所说。通过WMI改变亮度只能在可以动态设置亮度的系统中使用(笔记本电脑和一些一体化设备)。

但是,有一个Microsoft功能允许您更改外部显示器的亮度SetMonitorBrightness

请参阅Microsoft library

我附上了一个如何做的简单示例:

// Includes
#include "PhysicalMonitorEnumerationAPI.h"
#include "HighLevelMonitorConfigurationAPI.h"

(...)

// Prepare variables
HMONITOR hMonitor = NULL;
HMONITOR hMonitorTest = NULL;
DWORD cPhysicalMonitors;
LPPHYSICAL_MONITOR pPhysicalMonitors = NULL;

// Get the screen
HWND hWnd = GetDesktopWindow();
hMonitor = MonitorFromWindow(hWnd, MONITOR_DEFAULTTOPRIMARY);

_BOOL success = GetNumberOfPhysicalMonitorsFromHMONITOR(hMonitor, &cPhysicalMonitors);
if(success)
{
    pPhysicalMonitors = (LPPHYSICAL_MONITOR)malloc(cPhysicalMonitors* sizeof(PHYSICAL_MONITOR));        
    if(pPhysicalMonitors != NULL)
    {
        success = GetPhysicalMonitorsFromHMONITOR(hMonitor,cPhysicalMonitors, pPhysicalMonitors);
        HANDLE hPhysicalMonitor = pPhysicalMonitors[0].hPhysicalMonitor;

        // Set brightness to 50%
        DWORD dwNewBrightness = 50;
        success = SetMonitorBrightness(hPhysicalMonitor, dwNewBrightness);

        // Free resources
        free(pPhysicalMonitors);
    }
}

答案 1 :(得分:0)

主要是因为您的电脑不支持通过软件进行亮度控制。例如,在笔记本电脑中,有可用于控制亮度的热键,如果存在,您将获得一个实例并通过WMI使用它。

可以肯定的是,您可以在powershell中运行下面提到的命令。如果它返回不支持,则您无法将WMI用于这些控件。

Get-CimInstance -Namespace root/WMI -ClassName WmiMonitorBrightnessMethods

注意:热键通过移动中心等其他软件工作。