【发布时间】:2018-02-16 01:05:31
【问题描述】:
我有一个程序可以切换桌面并在其上启动一个新进程。当进程退出时,父进程恢复原来的桌面。
出于测试目的,我在一个普通的 win32 应用程序中放置了一个按钮来触发开关。它工作,并关闭启动的进程(记事本),我回到原来的桌面。
在同一个程序中,我调用了 WTSRegisterSessionNotification 以在会话解锁 (WTS_SESSION_UNLOCK) 时接收通知。我收到了。
但是当我尝试在 WTS_SESSION_UNLOCK 消息处理程序中切换桌面时,SwitchDesktop 失败并且 GetLastError 为 0。文档说最后一个错误通常不是由 SwitchDesktop 设置的。
很有趣,如果我将切换桌面的调用放在 for 循环中,它会在第 5 次迭代中运行。
简而言之,这是行不通的:
case WM_WTSSESSION_CHANGE:
if(wParam == WTS_SESSION_UNLOCK)
{
SwitchDesktop(a_valid_desktop_handle);
}
break;
但是这个丑陋的黑客是有效的:
case WM_WTSSESSION_CHANGE:
if(wParam == WTS_SESSION_UNLOCK)
{
for(int i=0; i<10; ++i)
{
if(SwitchDesktop(a_valid_desktop_handle))
{
//This will work when i == 5, maybe 6.
break;
}
}
}
break;
设置一个计时器(退出消息循环)也可以,但就这个问题而言,它只是一种更复杂的循环形式。 SwitchDesktop 将在少量 WM_TIMER 消息后运行。它看起来像常数时间,虽然我没有测量它。
MSDN documentation for SwitchDesktop 提到这将因我使用的自定义 Userinit 进程而失败。但是在切换之前获取当前桌面的名称:
wchar_t name[512];
GetUserObjectInformation(GetThreadDesktop(GetCurrentThreadId()), UOI_NAME, name, sizeof(name)/sizeof(*name), 0);
OutputDebugString(name);
一直给我default。由于GetLastError 是 0,而不是 5(拒绝访问),我很确定安全桌面已经消失在我收到 WTS_SESSION_UNLOCK 通知之前。
我知道在屏幕锁定时我无法切换桌面,但在桌面解锁后是否有一个“宽限期”我无法调用 SwitchDesktop?
【问题讨论】: