检测窗口恢复操作即将开始

Dem*_*ion 2 directx winapi multithreading event-handling

WM_GETMINMAXINFO当最大化操作即将开始和WM_SIZE最大化操作完成时生成。

WM_SIZE恢复操作完成时也会生成。

但如何检测窗口恢复操作即将开始呢?


我需要检测窗口即将恢复的确切时刻,而不是已经恢复的时刻。我正在开发多线程DirectX应用程序。我在专用的辅助线程中渲染。当窗口即将开始最大化或恢复时,我需要更改渲染后缓冲区大小(DirectX Device Reset)。我只能从主线程调整后台缓冲区的大小,因此我用来Critical Sections与渲染线程同步。问题是我无法中断Present渲染线程中的操作,并且当最大化或恢复操作即将开始时,我会等到当前Present操作完成,然后才开始调整大小(最大化/恢复)。如果您太晚更改后台缓冲区大小(当最大化/恢复操作完成时(WM_SIZE消息),您会注意到旧帧以错误的大小绘制(图像被拉伸)。

Iva*_*rop 5

嗯,很高兴看到有人仍在思考这些问题。这些小事情,例如调整大小时图像拉伸一秒,将专业外观的应用程序与卧室编码的应用程序区分开来。

在我的旧代码中,我发现了您可能会感兴趣的技巧。首先,您可以检查WM_SYSCOMMAND

        case WM_SYSCOMMAND:
           {
               switch (wParam)
               {
               case SC_MAXIMIZE:
                   std::cout << "Going to MAXIMIZE: " << std::endl;
                   break;

               case SC_MINIMIZE:
                   std::cout << "Going to MINIMIZE: " << std::endl;

                   break;

               case SC_RESTORE:
                   std::cout << "Going to RESTORE: " << std::endl;
                   break;


               default:
                   break;
               }
               return DefWindowProc(m_hWnd, msg, wParam, lParam);
           }
Run Code Online (Sandbox Code Playgroud)

但问题是,当用户双击 titlebar时,它不会捕获最大化/恢复事件。

所以我在解析时发现了一个小技巧WM_WINDOWPOSCHANGING

首先让我们做一些有趣的研究。我们在这里这里阅读了文档,我们发现:

WM_WINDOWPOSCHANGING 消息发送到其大小、位置或 Z 顺序中的位置即将更改的窗口

因为在通用调试器中调试事件几乎是不可能的(如果调试器经常更改窗口的 z 顺序,您将如何测试?),因此出于测试目的,我们将使用 制作一个小控制台应用程序,在其中我们像往常int main()一样初始化窗口(HINSTANCE我们可以从GetModuleHandle(0);) 得到。所以我们同时有一个控制台和一个窗口。在窗口过程中,我们捕获WM_WINDOWPOSCHANGING并打印信息,它会告诉我们,控制台:

        case WM_WINDOWPOSCHANGING:
        {
            WINDOWPOS* wp = ((WINDOWPOS*)lParam);

            // We checking current state which is saved in member (or global) bools
            // Set them checking WM_SIZE before
            if (m_bMaximized)
            {
                std::cout << "Currently MAXIMIZED: ";
            }
            else if (m_bMinimized)
            {
                std::cout << "Currently MINIMIZED: ";
            }
            else
            {
                std::cout << "Currently NORMAL: ";
            }

            dbgPrintPositionCurrent();

            std::cout << "Going to change to: ";
            dbgPrintPosition(wp->x, wp->y, wp->cx, wp->cy, wp->flags);
            std::cout << std::endl << std::endl;

            return DefWindowProc(m_hWnd, msg, wParam, lParam);
        }
Run Code Online (Sandbox Code Playgroud)

请参阅此处的实用函数。

当我们玩够了之后,我们现在就可以让它变得有用了:

                bool bFrameChanged = ((wp->flags) & SWP_FRAMECHANGED) > 0;
            bool bNoCopyBits = ((wp->flags) & SWP_NOCOPYBITS) > 0;
            bool bNormal = (!m_bMaximized) && (!m_bMinimized);

            // from maximized
            if(m_bMaximized && bFrameChanged && !bNoCopyBits)
            {
                std::cout << " MAXIMIZED -> NORMAL " << std::endl;
            }
            if (m_bMaximized && bFrameChanged && bNoCopyBits)
            {
                std::cout << " MAXIMIZED -> MINIMIZED " << std::endl;
            }

            // from normal states
            if (bNormal && bFrameChanged && !bNoCopyBits)
            {
                std::cout << " NORMAL -> MAXIMIZED " << std::endl;
            }
            if (bNormal && bFrameChanged && bNoCopyBits)
            {
                std::cout << " NORMAL -> MINIMIZED"  << std::endl;
            }

            // from minimized
            if(m_bMinimized && bFrameChanged)
            {
                std::cout << " MINIMIZED -> MAXIMIZED " << std::endl;
            }
            if(m_bMinimized && m_bMaximized && bFrameChanged)
            {
                std::cout << " MINIMIZED -> MAXIMIZED " << std::endl;
            }


            return DefWindowProc(m_hWnd, msg, wParam, lParam);
Run Code Online (Sandbox Code Playgroud)

我真的不确定这是最简单的方法还是正确的方法。但它现在有效 =) 另外,我认为你的应用程序并不关心到底发生了什么:最大化、恢复或调整大小。它只关心大小是否发生变化,因此您需要调整缓冲区大小/重新创建交换链等......

希望能帮助到你!快乐编码!