1

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

WM_SIZE还原操作完成时也会生成。

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


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

4

1 回答 1

3

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

在我的旧代码中,我发现了您可能会感兴趣的技巧。第一件事是您可以检查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);
           }

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

所以我发现了一个小技巧,当 parse 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);
        }

请参阅此处的实用程序函数。

当我们玩得足够多时,我们现在可以让它变得有用:

                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);

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

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

于 2013-04-25T08:14:13.510 回答