用代碼移動桌面圖標(貪吃蛇)

效果圖windows

實例.gif

前言api

記得上高二的時候,閒來無事,上b站搜電腦病毒的視頻看(不要問我爲何會搜這個),看到一個頗有意思的"病毒",其實也不算病毒,它會控制桌面圖標造成一我的形,並跳舞,跳完以後電腦就藍屏了.以後下定決心也要整一個,埋頭研究了兩個星期吧,寫了一個貪吃蛇,此貪吃蛇非彼貪吃蛇,它固然是控制的桌面圖標來玩啦,還寫了個網絡版的,經過手機去控制.網絡

貪吃蛇效果函數

本文章只介紹如何移動圖標,不介紹貪吃蛇實現(源碼太多),能夠評價私信要源碼oop

實例.gif

實現思路code

說到這,真的很後悔之前沒第一個學c語言,反而學了vb,如今被其餘語言搞得沒有精力去學了,並無說vb很差,它也很強大,只是它封裝了不少,讓不少人接觸底層得知識更少了,好比窗體,歸根到底,全部窗體都是用CreateWindow 函數來建立,可是學了vb,誰會傻到在用CreateWindow來建立窗口呢?orm

好了,說原理吧.
首先要了解什麼是句柄,通俗得講它就是一個整數,具體它標識窗口、位圖、畫筆等對象,而且是不變得整數,就像身份證同樣,當人出生後,一定會有一個18位的身份證號,而且是不變得,經過它,能夠知道某人得姓名,地址,年齡等信息,句柄也是如此.視頻

上面所說的CreateWindow,當它成功建立一個窗口後,它的返回值就是一個句柄,你拿到這個句柄後,你就能夠對它作你想作的事了,好比用SetWindowText 給它設置一個標題.對象

明白了句柄,如今就要拿到桌面的句柄,對它進行控制,用spy++能夠分析一下它blog

image.png

這裏有一點不同,在windows7中,它的父窗口類名是Program Manage,可是在windows10中,它的類名是WorkerW,而且有多個WorkerW.下圖是windows 7中

image.png

可是這不影響咱們.從中能夠看到,桌面就是一個ListView,咱們只要拿到它的句柄,就能夠對他進行控制.

獲取桌面句柄

須要用到api:
FindWindow:根據窗口的類名和窗口名稱匹配指定的窗口,而且返回這個窗口的句柄
FindWindowEx:根據所在的父窗口中查找類名和窗口名稱匹配的窗口,而且返回這個窗口的句柄

查找桌面句柄以下.
注:在win7下可就不這麼寫了,

Private Function getDesktopHwnd() As Long
Dim hwndWorkerW As Long, hwndShelldll As Long, hwndDesktop As Long

Do While (hwndDesktop = 0)
    hwndWorkerW = FindWindowEx(0, hwndWorkerW, "WorkerW", vbNullString)
        If (hwndWorkerW <> 0) Then
            hwndShelldll = FindWindowEx(hwndWorkerW, 0, "SHELLDLL_DefView", vbNullString)
   
            hwndDesktop = FindWindowEx(hwndShelldll, 0, "SysListView32", vbNullString)
        End If
Loop
getDesktopHwnd = hwndDesktop
End Function

移動圖標

知道了桌面的句柄,而後就是移動圖標了
既然桌面就是一個ListView,那麼只須要給ListView發送一個LVM_SETITEMPOSITION消息來設置item的位置.
若是你對windows機制不太瞭解的話,可能不明白爲什麼要這樣作,簡單說一下,windows是根據消息來驅動程序運行的,例如,單擊鼠標、改變窗口尺寸、按下鍵盤上的一個鍵都會使Windows發送一個消息給應用程序,應用程序會在一個叫窗口過程函數的地方處理消息,windows定義了大量的消息,列入:WM_CLOSE,對這個窗口發送WM_CLOSE,若是應用程序不作特殊處理的話,窗口就會關閉,

若是對ListView操做的話,就須要發送以LVM開頭的消息,WM開頭的是窗口消息,還有對列表框的消息LB_xxx

發送消息能夠用SendMessage函數或PostMessage,區別在於使用PostMessage是當即返回,SendMessage須要等待應用程序處理完以後返回,在這裏咱們也不須要等待程序返回,因此用PostMessage.

看一下PostMeeage的定義

image.png
參數hwnd:消息接收的窗口句柄
參數msg:具體消息
參數wparam和lparam:其餘特定於消息的信息。

在看一下LVM_SETITEMPOSITION消息的定義

image.png

也就是說,使用PostMeeage投遞LVM_SETITEMPOSITION消息的時候,參數wparam是某個圖標的索引,lparam是位置.
可是如何用一個整數標識一個座標呢,在c中能夠用MAKELPARAM來獲取,vb中可沒有這樣的方法,那就本身寫一個

Public Function MAKELPARAM(ByVal l As Integer, ByVal h As Integer) As Long
Dim ll As String
Dim lh As String
Dim r As String
ll = Format(Hex(l), "@@@@")
lh = Format(Hex(h), "@@@@")
Dim result As Long

result = CLng("&h" & Replace(lh & ll, " ", "0"))
 MAKELPARAM = result
 
End Function

所有代碼

Private Const LVM_FIRST As Long = &H1000

Private Declare Function GetDesktopWindow Lib "user32.dll" () As Long

Private Const LVM_SETITEMPOSITION32 As Long = (LVM_FIRST + 49)
Private Declare Function PostMessage Lib "user32.dll" Alias "PostMessageA" (ByVal hwnd As Long, ByVal wMsg As Long, ByVal wParam As Long, ByVal lParam As Long) As Long

Private Declare Function EnumWindows Lib "user32.dll" (ByVal lpEnumFunc As Long, ByVal lParam As Long) As Long

Private Declare Function FindWindow Lib "user32.dll" Alias "FindWindowA" (ByVal lpClassName As String, ByVal lpWindowName As String) As Long
Private Declare Function FindWindowEx Lib "user32.dll" Alias "FindWindowExA" (ByVal hWnd1 As Long, ByVal hWnd2 As Long, ByVal lpsz1 As String, ByVal lpsz2 As String) As Long

Private Const LVM_SETITEMPOSITION As Long = (LVM_FIRST + 15)

Private Type POINTAPI
    x As Long
    y As Long
End Type

Dim curPoint As POINTAPI

Dim listViewhwnd As Long

Public Function MAKELPARAM(ByVal l As Integer, ByVal h As Integer) As Long
Dim ll As String
Dim lh As String
Dim r As String
ll = Format(Hex(l), "@@@@")
lh = Format(Hex(h), "@@@@")
Dim result As Long

result = CLng("&h" & Replace(lh & ll, " ", "0"))
 MAKELPARAM = result
 
End Function


Private Function getDesktopHwnd() As Long
Dim hwndWorkerW As Long, hwndShelldll As Long, hwndDesktop As Long

Do While (hwndDesktop = 0)
    hwndWorkerW = FindWindowEx(0, hwndWorkerW, "WorkerW", vbNullString)
        If (hwndWorkerW <> 0) Then
            hwndShelldll = FindWindowEx(hwndWorkerW, 0, "SHELLDLL_DefView", vbNullString)
   
            hwndDesktop = FindWindowEx(hwndShelldll, 0, "SysListView32", vbNullString)
        End If
Loop
getDesktopHwnd = hwndDesktop
End Function
Private Sub Form_Load()
listViewhwnd = getDesktopHwnd()

End Sub

Private Sub Timer1_Timer()
curPoint.x = curPoint.x + 10

PostMessage listViewhwnd, LVM_SETITEMPOSITION, 10, MAKELPARAM(curPoint.x, 110)
End Sub

注:右擊桌面--->查看---->自動排列圖標 、將圖標和網格對齊 須要取消勾選

相關文章
相關標籤/搜索