ixe*_*013 6 windows winapi wtsapi32
我有一个程序可以切换桌面并在其上启动一个新进程.进程退出时,父进程将还原原始桌面.
出于测试目的,我在一个触发切换的普通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;
Run Code Online (Sandbox Code Playgroud)
但这个丑陋的黑客有效:
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;
Run Code Online (Sandbox Code Playgroud)
设置一个计时器(退出消息循环)也有效,但对于这个问题,它只是一个更复杂的循环形式.SwitchDesktop将在完成一系列WM_TIMER消息后继续工作.看起来像是不变的时间,虽然我没有测量它.
SwitchDesktop的MSDN文档提到我将使用自定义Userinit进程失败.但是在切换之前获取当前桌面的名称:
wchar_t name[512];
GetUserObjectInformation(GetThreadDesktop(GetCurrentThreadId()), UOI_NAME, name, sizeof(name)/sizeof(*name), 0);
OutputDebugString(name);
Run Code Online (Sandbox Code Playgroud)
给我default所有的时间.并且因为GetLastError是0 而不是5(访问被拒绝)我非常确定在收到WTS_SESSION_UNLOCK通知之前安全桌面已经消失.
我知道在屏幕被锁定时我无法切换桌面,但桌面解锁后是否有"宽限期",我无法调用SwitchDesktop?
我现在无法测试它,但我会调用SwitchDesktopnot onWTS_SESSION_UNLOCK但 on WTS_CONSOLE_CONNECT。从我收集的情况来看,WTS_SESSION_UNLOCK首先发生,然后你得到的结果WTS_CONSOLE_CONNECT将与你在“恒定时间”中看到的内容相对应......