首页
社区
课程
招聘
关于重启explorer.exe进程跟SetParent的问题求助
发表于: 2015-5-8 11:44 5349

关于重启explorer.exe进程跟SetParent的问题求助

2015-5-8 11:44
5349
遇到个问题是这样的:我创建了一个窗口,setparent设置为桌面子窗口,这样WIN+D就不会最小化了,这个时候我结束explorer.exe进程再重启explorer进程,然后再重新获取桌面句柄(demo里面用的定时器),然后setparent,这个时候窗口就不显示了,进程仍旧存在,请问各路大侠要怎样才能继续让窗口显示且依然还是桌面子窗口依然win+d不最小化?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
// ParentWindowTest.cpp : 定义应用程序的入口点。
//
 
#include "stdafx.h"
#include "ParentWindowTest.h"
 
#define MAX_LOADSTRING 100
 
// 全局变量:
HINSTANCE hInst;                                // 当前实例
TCHAR szTitle[MAX_LOADSTRING];                  // 标题栏文本
TCHAR szWindowClass[MAX_LOADSTRING];            // 主窗口类名
HWND g_hParentWnd;              //父窗口句柄                        
 
// 此代码模块中包含的函数的前向声明:
ATOM                MyRegisterClass(HINSTANCE hInstance);
BOOL                InitInstance(HINSTANCE, int);
LRESULT CALLBACK    WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK    About(HWND, UINT, WPARAM, LPARAM);
 
int APIENTRY _tWinMain(HINSTANCE hInstance,
    HINSTANCE hPrevInstance,
    LPTSTR    lpCmdLine,
    int       nCmdShow)
{
    UNREFERENCED_PARAMETER(hPrevInstance);
    UNREFERENCED_PARAMETER(lpCmdLine);
 
    // TODO: 在此放置代码。
    MSG msg;
    HACCEL hAccelTable;
 
    // 初始化全局字符串
    LoadString(hInstance, IDS_APP_TITLE, szTitle, MAX_LOADSTRING);
    LoadString(hInstance, IDC_PARENTWINDOWTEST, szWindowClass, MAX_LOADSTRING);
    MyRegisterClass(hInstance);
 
    // 执行应用程序初始化:
    if (!InitInstance (hInstance, nCmdShow))
    {
        return FALSE;
    }
 
    hAccelTable = LoadAccelerators(hInstance, MAKEINTRESOURCE(IDC_PARENTWINDOWTEST));
 
    // 主消息循环:
    while (GetMessage(&msg, NULL, 0, 0))
    {
        if (!TranslateAccelerator(msg.hwnd, hAccelTable, &msg))
        {
            TranslateMessage(&msg);
            DispatchMessage(&msg);
        }
    }
 
    return (int) msg.wParam;
}
 
 
 
//
//  函数: MyRegisterClass()
//
//  目的: 注册窗口类。
//
//  注释:
//
//    仅当希望
//    此代码与添加到 Windows 95 中的“RegisterClassEx”
//    函数之前的 Win32 系统兼容时,才需要此函数及其用法。调用此函数十分重要,
//    这样应用程序就可以获得关联的
//    “格式正确的”小图标。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
    WNDCLASSEX wcex;
 
    wcex.cbSize = sizeof(WNDCLASSEX);
 
    wcex.style          = CS_HREDRAW | CS_VREDRAW;
    wcex.lpfnWndProc    = WndProc;
    wcex.cbClsExtra     = 0;
    wcex.cbWndExtra     = 0;
    wcex.hInstance      = hInstance;
    wcex.hIcon          = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_PARENTWINDOWTEST));
    wcex.hCursor        = LoadCursor(NULL, IDC_ARROW);
    wcex.hbrBackground  = (HBRUSH)(COLOR_WINDOW+1);
    wcex.lpszMenuName   = MAKEINTRESOURCE(IDC_PARENTWINDOWTEST);
    wcex.lpszClassName  = szWindowClass;
    wcex.hIconSm        = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_SMALL));
 
    return RegisterClassEx(&wcex);
}
 
//
//   函数: InitInstance(HINSTANCE, int)
//
//   目的: 保存实例句柄并创建主窗口
//
//   注释:
//
//        在此函数中,我们在全局变量中保存实例句柄并
//        创建和显示主程序窗口。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
    HWND hWnd;
 
    hInst = hInstance; // 将实例句柄存储在全局变量中
 
 
    hWnd = CreateWindow(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
        CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, NULL, NULL, hInstance, NULL);
 
 
    g_hParentWnd = ::FindWindow(_T("Progman"), NULL);
    if(NULL == g_hParentWnd)
    {
        return false;
    }
    g_hParentWnd = ::FindWindowEx(g_hParentWnd, NULL, _T("SHELLDLL_DefView"), NULL);
    if(NULL == g_hParentWnd)
    {
        return false;
    }
    g_hParentWnd = ::FindWindowEx(g_hParentWnd, NULL, _T("SysListView32"), NULL);
    if(NULL == g_hParentWnd)
    {
        return false;
    }
 
[COLOR="red"]   SetTimer(hWnd,58,10000,NULL);
    ::SetParent(hWnd, g_hParentWnd);[/COLOR]
    if (!hWnd)
    {
        return FALSE;
    }
 
    ShowWindow(hWnd, nCmdShow);
    UpdateWindow(hWnd);
 
    return TRUE;
}
 
//
//  函数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
//  目的: 处理主窗口的消息。
//
//  WM_COMMAND  - 处理应用程序菜单
//  WM_PAINT    - 绘制主窗口
//  WM_DESTROY  - 发送退出消息并返回
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    int wmId, wmEvent;
    PAINTSTRUCT ps;
    HDC hdc;
 
    switch (message)
    {
    case WM_COMMAND:
        wmId    = LOWORD(wParam);
        wmEvent = HIWORD(wParam);
        // 分析菜单选择:
        switch (wmId)
        {
        case IDM_ABOUT:
            DialogBox(hInst, MAKEINTRESOURCE(IDD_ABOUTBOX), hWnd, About);
            break;
        case IDM_EXIT:
            DestroyWindow(hWnd);
            break;
        default:
            return DefWindowProc(hWnd, message, wParam, lParam);
        }
        break;
    case WM_PAINT:
        hdc = BeginPaint(hWnd, &ps);
        // TODO: 在此添加任意绘图代码...
        EndPaint(hWnd, &ps);
        break;
    [COLOR="Red"]case WM_TIMER:[/COLOR]     {
            //MessageBox(NULL,L"10s",L"时间到",0);     ////---测试发现重启explorer.exe之后再也接受不到定时器消息了
            HWND hParentWnd = ::FindWindow(_T("Progman"), NULL);
            if(NULL == hParentWnd)
            {
            return false;
            }
            hParentWnd = ::FindWindowEx(hParentWnd, NULL, _T("SHELLDLL_DefView"), NULL);
            if(NULL == hParentWnd)
            {
            return false;
            }
            hParentWnd = ::FindWindowEx(hParentWnd, NULL, _T("SysListView32"), NULL);
            if(NULL == hParentWnd)
            {
            return false;
            }
            if(hParentWnd!=g_hParentWnd)
            {
            ::SetParent(hWnd, g_hParentWnd);
            ShowWindow(hWnd,SW_SHOW);
            }
        }
        break;
    case WM_DESTROY:
        PostQuitMessage(0);
        break;
    default:
        return DefWindowProc(hWnd, message, wParam, lParam);
    }
    return 0;
}
 
// “关于”框的消息处理程序。
INT_PTR CALLBACK About(HWND hDlg, UINT message, WPARAM wParam, LPARAM lParam)
{
    UNREFERENCED_PARAMETER(lParam);
    switch (message)
    {
    case WM_INITDIALOG:
        return (INT_PTR)TRUE;
 
    case WM_COMMAND:
        if (LOWORD(wParam) == IDOK || LOWORD(wParam) == IDCANCEL)
        {
            EndDialog(hDlg, LOWORD(wParam));
            return (INT_PTR)TRUE;
        }
        break;
    }
    return (INT_PTR)FALSE;
}

[培训]内核驱动高级班,冲击BAT一流互联网大厂工作,每周日13:00-18:00直播授课

收藏
免费
支持
分享
最新回复 (2)
雪    币: 69
活跃值: (86)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
2
不懂。
不会是窗口和explorer一起被销毁了吧。
2015-5-8 12:14
0
雪    币: 1178
活跃值: (3903)
能力值: ( LV7,RANK:140 )
在线值:
发帖
回帖
粉丝
3
在子窗口的 WM_NCDESTROY 消息响应函数中,把父窗口设置为空或者别的什么。最好设置一个变量判断是由于父窗口销毁导致的WM_NCDESTROY还是你确实要把子窗口销毁,如果是由于父窗口销毁的话,就按上面那样,设置父窗口为空,然后隐藏窗口而不是销毁该子窗口;如果是你自己要销毁的,就调用CDialog::OnDestroy()正常销毁。

看你的代码有处理WM_DESTROY,我没测试这个消息,我用的是MFC的窗体草草测试的,我对于windows的消息机制也不太了解,知其然而不知其所以然。你可以自己测试一下WM_DESTROY消息是否也可以。
2015-5-10 14:13
0
游客
登录 | 注册 方可回帖
返回

账号登录
验证码登录

忘记密码?
没有账号?立即免费注册