曾幾何時,服務(wù)里面彈出一個窗口到桌面上不再是那么地容易了:以前只要把服務(wù)設(shè)置為允許和桌面交互就可以直接在服務(wù)啟動的時候,把一個窗口彈給用戶。但是現(xiàn)在在vista(其它的OS 下沒有測試,未知)下要彈出這樣的窗體,首先會彈出一個提示框提示是否接否一個服務(wù)彈出來的消息,點接受后,才會在一個全灰的桌面里面彈出這個窗口。不用想,這樣的用戶體驗,肯定是會被直接PASS。原因很簡單,因為不同的用戶間的桌面是不一樣的,服務(wù)用的是System的權(quán)限,在vista里面是Session0,而用戶的帳戶不是這個(肯定大于0)。
看來,想彈出一個窗體,需要另一個程序來作輔助了。解決方案有兩種:
- 開發(fā)一個程序A在啟動的時候,隨系統(tǒng)啟動,并監(jiān)控指定文件M,服務(wù)S有消息的時候,放在文件M里,A 感受了文件變化了,就去讀這個文件里的內(nèi)容,根據(jù)規(guī)則來作對應(yīng)的動作。壞處很明顯,當(dāng)用戶為了讓系統(tǒng)跑得快的時候,這種自啟動文件很容易被砍掉,導(dǎo)致了有些功能莫名奇妙地不可用。
- 同樣地,也是開發(fā)一個程序A,用CreateProcessAsUser這個API來創(chuàng)建這個A ,并且發(fā)送到用戶的桌面上去。好處是可以把這個A和服務(wù)S 放在同一個程序文件里面,根據(jù)不同的參數(shù)來啟動不同的功能。這樣用戶除非是刪除整個服務(wù),否則不會有部分功能能用,又有部分功能不能用的問題。
現(xiàn)在來看看第二種方案,要想用CreateProcessAsUser這個API ,有這樣幾個步驟:
1.取得用戶的令牌(Token)
2.指定好虛擬桌面
3.調(diào)用API創(chuàng)建這個用戶進(jìn)程。
在取得用戶令牌的時候,又有幾個方法:
1.從用戶的進(jìn)程上去剝
2.先用WTSGetActiveConsoleSessionId得到用戶會話ID,再用WTSQueryUserToken這個API去取。
不過在vista下面,服務(wù)里面的WTSGetActiveConsoleSessionId這個API得到的總是0,也就是Session0,用這個創(chuàng)建出來的進(jìn)程,還是屬于一個服務(wù)進(jìn)程。(而且這個API是XP以及以后的系統(tǒng)才會提供的,在早點的系統(tǒng)上就會調(diào)用失敗)所以我們只好從用戶的進(jìn)程上去找,用戶登錄的時候,一定會有的進(jìn)程就是:explorer.exe,這樣可以遍歷所有的進(jìn)程去找到這個(如果是多人同時登錄這個系統(tǒng)里,我也不知道該怎么辦了,不知道有沒有高人指點一下)。
這樣創(chuàng)建出來的用戶進(jìn)程,在進(jìn)程列表里面可以看到是活動用戶的。但是如果在這里面使用一下文件選擇框,或是去取一些系統(tǒng)目錄,比如說用戶的桌面,用戶的收藏夾等,都會得到空。這時候因為沒有指定用戶環(huán)境造成的。使用CreateEnvironmentBlock這個API就可以搞定了。