Win32:UpdateLayeredWindow有时失败(错误317)

时间:2014-05-15 19:04:27

标签: mfc png

我这里有一个奇怪的问题。我正在使用分层窗口显示半透明的启动画面(.png文件)。它适用于某些机器但不适用于其他机器。在它不起作用的机器上,GetLastError返回317(这不是很有用)。有谁之前经历过这个吗?这是我的相关功能。 CreateAsPNG的传入参数是WS_VISIBLE | WS_POPUP(dwStyle),0(dwExStyle),以及父级隐藏工具窗口的句柄,因此不会创建任务栏条目。我已经验证我可以将嵌入的PNG资源加载到CImage中,并且图像的大小是正确的。

提前感谢您的帮助!

BOOL MySplashWnd::CreateAsPNG( DWORD dwStyle, DWORD dwExStyle, const CString& sTitle, HWND hWndParent )
{
    ATL::CImage img;
    CreateStreamOnResource( m_nBitmapID, img );

    m_nWndWidth = img.GetWidth();
    m_nWndHeight = img.GetHeight();

    int nTop = 0;
    int nLeft = 0;
    GetTopLeft( nTop, nLeft );

    dwExStyle |= WS_EX_LAYERED;

    // Create the Splash Window
    BOOL bRetVal = CWnd::CreateEx( dwExStyle, AfxRegisterWndClass( CS_CLASSDC ), sTitle,
                                  dwStyle, nLeft, nTop, m_nWndWidth, m_nWndHeight, hWndParent, NULL );

    //Couldn't create the window for some unknown reason...
    X_ASSERT( bRetVal != FALSE );

    if ( bRetVal )
    {
        HDC hScreenDC = ::GetDC( m_hWnd );
        HDC hDC = ::CreateCompatibleDC( hScreenDC );
        HBITMAP hBmp = ::CreateCompatibleBitmap( hScreenDC, m_nWndWidth, m_nWndHeight );
        HBITMAP hBmpOld = ( HBITMAP ) ::SelectObject( hDC, hBmp );

        img.Draw( hDC, 0, 0, m_nWndWidth, m_nWndHeight, 0, 0, m_nWndWidth, m_nWndHeight );

        BLENDFUNCTION blend = { 0 };
        blend.BlendOp = AC_SRC_OVER;
        blend.BlendFlags = 0;
        blend.SourceConstantAlpha = 255;
        blend.AlphaFormat = AC_SRC_ALPHA;

        POINT ptPos = { nLeft, nTop };
        SIZE sizeWnd = { m_nWndWidth, m_nWndHeight };
        POINT ptSource = { 0, 0 };

        if ( ::UpdateLayeredWindow( m_hWnd, hScreenDC, &ptPos, &sizeWnd, hDC, &ptSource, 0, &blend, ULW_ALPHA ) )
        {
        }
        else
        {
            // The last error value is 317 on some Win7 machines.
            TRACE( _T( "*** Last error: %d\n" ), ::GetLastError() );
        }

        ::SelectObject( hDC, hBmpOld );
        ::DeleteObject( hBmp );
        ::DeleteDC( hDC );
        ::ReleaseDC( NULL, hScreenDC );
    }

    return bRetVal;
}

void MySplashWnd::CreateStreamOnResource( UINT nIDRes, ATL::CImage& img )
{
    HINSTANCE hInstance = ::GetMUIResourceInstance();
    if ( hInstance == NULL )
    {
        return;
    }

    HRSRC hResource = ::FindResource( hInstance, MAKEINTRESOURCE( nIDRes ), "PNG" );
    if ( hResource == NULL )
    {
        return;
    }

    DWORD dwResourceSize = ::SizeofResource( hInstance, hResource );
    if ( dwResourceSize == 0 )
    {
        return;
    }

    HGLOBAL hImage = ::LoadResource( hInstance, hResource );
    if ( hImage == NULL )
    {
        return;
    }

    LPVOID pvImageResourceData = ::LockResource( hImage );
    if ( pvImageResourceData == nullptr )
    {
        return;
    }

    HGLOBAL hImageData = ::GlobalAlloc( GMEM_MOVEABLE, dwResourceSize );
    if ( hImageData == NULL )
    {
        return;
    }

    LPVOID pvImageBuffer = ::GlobalLock( hImageData );
    if ( pvImageBuffer != nullptr )
    {
        ::CopyMemory( pvImageBuffer, pvImageResourceData, dwResourceSize );
        ::GlobalUnlock( hImageData );

        IStream* pStream = nullptr;

        if ( SUCCEEDED( ::CreateStreamOnHGlobal( hImageData, TRUE, &pStream ) ) )
        {
            img.Load( pStream );
            pStream->Release();
        }

        ::GlobalUnlock( hImageData );
    }

    ::GlobalFree( hImageData );
} // CTTSplashWnd::CreateStreamOnResource

更新:我发现即使在同一台机器上,有时UpdateLayeredWindow会成功,有时会失败(但如果失败则始终使用代码317)。另一条信息是这个启动是在一个单独的UI线程上运行的。它总是可以在我的机器上工作......

2 个答案:

答案 0 :(得分:2)

我遇到同样的问题而我无法找到任何信息。使用SetWindowAttributes方法可以工作,但我想使用SetLayeredWindow方式。我将要通过整个Windows API进行调试,以了解发生了什么,因为msdn提供了有关此错误消息的插孔信息。唯一的区别是我看到的使用此方法的OpenGL分层窗口演示示例使用CreateDIBSection而不是CreateCompatibleBitmap,它似乎可以在我的PC上运行。

答案 1 :(得分:1)

我已经在Windows 7 x86_64计算机上完美地运行UpdateLayeredWindow(),因为当我禁用桌面组合时它才会启动失败。原来那个

UpdateLayeredWindow (window_handle, NULL,
                     &position, &size,
                     buffer_hdc, &buffer_offset,
                     0, &blend_options, ULW_ALPHA);

有效,而

UpdateLayeredWindow (window_handle, NULL,
                     &position, NULL,
                     buffer_hdc, &buffer_offset,
                     0, &blend_options, ULW_ALPHA);

无效,因错误317而失败。

我一直试图跳过一些参数,因为我不需要改变窗口大小或内容,只是为了移动它。禁用桌面组合后,事实证明这是不可能的。

不确定这与您的原始问题有什么关系,因为您还提供屏幕DC,而我不提供。