求一个用VB做的屏幕保护程序的源文件

导读:
  --------------使用API------------------
  熟悉Windows操作系统的朋友一定对Windows的屏幕保护程序不陌生吧。如何自己编写Windows屏幕保护程序呢?当你看完下面的讲解后便可以轻易地编写一标准的Windows屏幕保护程序了!
  一个标准的屏保有以下几个特点:
  一:它是以.SCR作为文件的扩展名!
  二:它有三种运行方式。
  (1)运行在预览框中(用于预览屏保的效果。在“显示属性”→“屏幕保护程序”→“小屏幕”)。(见图)
  (2)运行设置程序(用于设置一些相关的样式。在“显示属性”→“屏幕保护程序”→“点击设置按钮”)。
  (3)真正的运行屏保(屏保运行时的效果。在“显示属性”→“屏幕保护程序”→“点击预览”或鼠标、键盘在指定的时间内无动作时)。
  如何让屏保识别当前需要运行哪一种方式呢?答案很简单——分析Windows调用屏保的参数。下面以Windows 98为例向大家分析一下调用屏保的参数。
  当Windows需要屏保显示在“小屏幕”中时会在调用屏保的后面加上两个参数。
  如:myscr.scr /p 7981(参数一:/p 表示让程序显示在“小屏幕”里,参数二:7981表示“小屏幕”的句柄hWnd。这样屏保就会得知Windows要它显示在“小屏幕”中。)
  当Windows需要屏保显示设置对话框时会在调用屏保的后面不加或加上两个参数。
  如:myscr.scr或myscr.scr /C 7987(参数一:/C表示让程序显示设置对话框,参数二:7987表示该属性页的句柄。)
  当Windows需要运行屏保时会在调用屏保的后面加上一个参数。
  如:myscr.scr /S(参数:/S表示让屏保运行。)
  好了,知道了Windows如何让屏保运行的三种方式后,接下来就要讨论如何实现它们了。
  实现原理:Windows通过某种方式调用屏保,屏保知道了它此时要干什么便会在当前环境中搜索是否有相同的实例存在。如果该实例的运行方式与此次要启动的运行方式不同则关闭前个实例,如果该实例的运行方式与此次要启动的运行方式相同则关闭此次运行的实例。
  显然要实现这种方法靠VB的App.PrevInstance是不可行的。因为我们要达到的目的是:侦测到前一个实例后要关闭它然后启动程序。而App.PrevInstance属性只能返回当前是否已启动一个应用程序的实例而不能对前个实例做些什么。(实例 简单地说就是相同的对象集合——同一程序。)在实现此方法之前首先向大家介绍三条API函数:GetClassName、FindWindow和SendMessage。其原型如下:
  Declare Function GetClassName Lib “user32” Alias “GetClassNameA” (ByVal hwnd As Long, ByVal lpClassName As String, ByVal nMaxCount As Long) As Long
  Declare Function FindWindow Lib “user32” Alias “FindWindowA” (ByVal lpClassName As String, ByVal lpWindowName As String) As Long
  Declare Function SendMessage Lib “user32” Alias “SendMessageA” (ByVal hwnd As Long, ByVal wMsg As Long, ByVal wParam As Long, lParam As Any) As Long
  GetClassName用于取得窗体的类名。调用成功后返回类名长度,失败返回零。函数需要三个参数:参数一.窗体的句柄,参数二.存放类名的缓冲,参数三.缓冲的大小。
  FindWindow用于寻找窗体。调用成功后返回窗体的句柄,失败返回零。函数需要两个参数:参数一.窗体的类名,参数二.窗体的标题。
  SendMessage用于向窗体发送一消息。函数需要四个参数:参数一.窗体的句柄,参数二:发送的消息名称,参数三、四.分别表示消息所附带的参数。
  使用了这三个函数便可轻易地实现关闭前有一个已启动的实例从而达到我们的目的。
  其次我们要实现如何让屏幕保护程序显示在预览框中(“小屏幕”)。
  要让屏幕保护程序在预览框中显示必须动态地改变窗口的样式使之成为“小屏幕”的子窗体,这样才能使预览框关闭时得到关闭消息。动态地改变窗口的样式可以使用GetWindowLong、SetWindowLong和SetParent。
  它们的原型如下:
  Public Declare Function GetWindowLong Lib “user32” Alias “GetWindowLongA” (ByVal hwnd As Long, ByVal nIndex As Long) As Long Public Declare Function SetWindowLong Lib “user32” Alias “SetWindowLongA” (ByVal hwnd As Long, ByVal nIndex As Long, ByVal dwNewLong As Long) As Long Declare Function SetParent Lib “user32” (ByVal hWndChild As Long, ByVal hWndNewParent As Long) As Long
  GetWindowLong的作用用于得到窗体的样式。调用成功后返回窗体的样式。函数需要两个参数:参数一.窗体的句柄,参数二.要取得窗体的样式只需使用常数GWL_STYLE。
  SetWindowLong的作用用于设置窗体的样式。函数需要三个参数:参数一.窗体的句柄,参数二.要设置窗体的样式只需用常数GWL_STYLE,参数三.要设置窗体的样式。
  SetParent的作用用于设置子窗体属于哪个父窗体。函数需要两个参数:参数一.子窗体的句柄,参数二.父窗体的句柄。
  知道了以上两点就可编写出标准的屏保。(关于效果就看你自己的了!)纸上谈兵了一阵就要落实到真正的编程上了。为了着重讲解屏保的实现方法故将屏保的效果简单化。
  首先新建一工程再添加一窗口,各属性设置如下:
  窗口 名称 Caption BorderStyle
  Form1 Frm_Setup 无 1 - None
  Form2 Frm_Run 任意 1 - Fixed Single
  其余属性均缺省。再在Frm_Run中添加一Timer控件,将该控件的名称改为Timer_Mov,Interval属性制改为500。
  添加两个模块,将Module1的名称改为Mod_Const,Module2的名称改为Mod_Main,添加以下代码:
  Mod_Const:
  Option Explicit
  Public Const WM_LOOK=“屏保预览(demo)”
  Public Const WM_SET=“屏保设置(demo)”
  Public Const WM_RUN=“屏保运行(demo)”
  Public Const HWND_TOP=0&
  Public Const WS_CHILD=&H40000000
  Public Const GWL_STYLE=(-16)
  Type RECT
  Left As Long
  Top As Long
  Right As Long
  Bottom As Long
  End Type
  Public Const SWP_NOZORDER=&H4
  Public Const SWP_NOACTIVATE=&H10
  Public Const SWP_SHOWWINDOW=&H40
  Public Const WM_CLOSE=&H10
  Declare Function GetClientRect Lib “user32” (ByVal hwnd As Long, lpRect As RECT) As Long
  Declare Function GetClassName Lib “user32” Alias “GetClassNameA” (ByVal hwnd As Long, ByVal lpClassName As String, ByVal nMaxCount As Long) As Long
  Declare Function FindWindow Lib “user32” Alias “FindWindowA” (ByVal lpClassName As String, ByVal lpWindowName As String) As Long
  Declare Function SendMessage Lib “user32” Alias “SendMessageA” (ByVal hwnd As Long, ByVal wMsg As Long, ByVal wParam As Long, lParam As Any) As Long
  Declare Function SetParent Lib “user32” (ByVal hWndChild As Long, ByVal hWndNewParent As Long) As Long
  Public Declare Function GetWindowLong Lib “user32” Alias “GetWindowLongA” (ByVal hwnd As Long, ByVal nIndex As Long) As Long
  Public Declare Function SetWindowLong Lib “user32” Alias “SetWindowLongA” (ByVal hwnd As Long, ByVal nIndex As Long, ByVal dwNewLong As Long) As Long
  Declare Function SetWindowPos Lib “user32” (ByVal hwnd As Long, ByVal hWndInsertAfter As Long, ByVal X As Long, ByVal Y As Long, ByVal cx As Long, ByVal cy As Long, ByVal wFlags As Long) As Long
  Declare Function ShowCursor Lib “user32” (ByVal bShow As Long) As Long
  Mod_Main:
  Option Explicit
  Sub Main() '程序运行入口
  Dim ClassName As String * 64 ’存放窗口的类名
  Dim ExeCmd As String '存放命令行参数
  GetClassName Frm_Setup.hwnd, ClassName, 64 ’取得窗口的类名
  ExeCmd=UCase(Command$) ’将调用的屏保的参数转换成大写后存放在变量ExeCmd里
  If Not (InStr(ExeCmd,“/P”)=0)Then ’检查屏保的调用参数中是否有“/P”参数
  If Not (FindWindow(ClassName, WM_LOOK)=0)Then End ’如果找到已有同一个运行方式的实例存在则程序结束
  ClosePreWindow ClassName, WM_SET ’关闭前面已启动的其他运行方式的实例
  ClosePreWindow ClassName, WM_RUN ’同上
  SCR_Look
  ElseIf Not (InStr(ExeCmd,“/S”)=0)Then
  If Not (FindWindow(ClassName,WM_RUN)=0) Then End
  ClosePreWindow ClassName, WM_LOOK ’同上
  ClosePreWindow ClassName, WM_SET ’同上
  Scr_Run
  Else
  If Not (FindWindow(ClassName, WM_SET)=0) Then End
  ClosePreWindow ClassName, WM_LOOK ’同上
  ClosePreWindow ClassName, WM_RUN ’同上
  Scr_Setup
  End If
  End Sub
  Public Sub ClosePreWindow(ClassName As String, WinCaption As String)
  Dim PreWnd As Long
  PreWnd=FindWindow(ClassName, WinCaption) ’寻找类名为ClassName,标题为WinCaption的窗口
  If Not (PreWnd = 0) Then Call SendMessage(PreWnd, WM_CLOSE, 0, 0) ’如果窗口已找到则关闭它
  End Sub
  Public Sub SCR_Look()
  Dim LookScrWnd As Long
  Dim Style As Long
  Dim LookRect As RECT
  Frm_Run.Caption=WM_LOOK ’赋上具有相应运行方式的标题
  LookScrWnd=Val(Right(Command$, Len(Command$) - 2)) ’取得小屏幕的窗口句柄
  Style=GetWindowLong(Frm_Run.hwnd, GWL_STYLE) ’取得窗口的样式
  Style=Style Or WS_CHILD ’在窗口的样式中加入子窗体常数
  SetWindowLong Frm_Run.hwnd, GWL_STYLE, Style ’改变窗体的样式
  SetParent Frm_Run.hwnd, LookScrWnd ’设置窗体的父窗体
  GetClientRect LookScrWnd, LookRect ’取得小屏幕的大小
  SetWindowPos Frm_Run.hwnd, HWND_TOP, 0, 0, LookRect.Right, LookRect.Bottom, SWP_
  NOZORDER Or SWP_NOACTIVATE Or SWP_SHOWWINDOW
  '显示窗体并将窗体的大小设置为小屏幕的大小以便覆盖小屏幕
  End Sub
  Public Sub Scr_Setup()
  Frm_Run.Caption=WM_SET ’赋上具有相应运行方式的标题
  Frm_Setup.Show
  End Sub
  Public Sub Scr_Run()
  Frm_Run.Caption = WM_RUN ’赋上具有相应运行方式的标题
  ShowCursor False ’隐藏鼠标
  Frm_Run.Move 0, 0, Screen.Width, Screen.Height
  Frm_Run.Show
  End Sub
  Public Sub CloseSCR()
  ShowCursor True ’显示鼠标
  Unload Frm_Setup ’卸载窗体关闭屏保
  Unload Frm_Run ’同上
  End Sub
  Public Function Scan_RUN() As Boolean’侦测当前屏保的运行方式
  If (Frm_Run.Caption = WM_RUN) Then ’如果屏保是以运行方式在运行则返回“真”,否则返回“假”
  Scan_RUN=True
  Else
  Scan_RUN=False
  End If
  End Function
  Frm_Run:
  Option Explicit
  Dim i As Integer ’定义循环变量
  Dim OldX As Integer ’定义存放旧的鼠标水平坐标
  Dim OldY As Integer ’定义存放旧的鼠标垂直坐标
  Dim Pic(1) As New StdPicture ’定义一个图片类的数组
  Private Sub Form_KeyDown(KeyCode As Integer, Shift As Integer)
  If Mod_Main.Scan_RUN Then ’如果此时是在运行屏保则关闭屏保
  Mod_Main.CloseSCR
  End If
  End Sub
  Private Sub Form_Load()
  i=1 ’为循环变量赋初值
  OldX=-1 ’为旧鼠标水平坐标赋初值
  OldY=-1 ’为旧鼠标垂直坐标赋初值
  Set Pic(0)=LoadPicture(请写入图片一的路径和名称) ’读取图片一
  Set Pic(1)=LoadPicture(请写入图片二的路径和名称) ’读取图片二
  End Sub
  Private Sub Form_MouseDown(Button As Integer, Shift As Integer, X As Single, Y A
  s Single)
  If Mod_Main.Scan_RUN Then ’如果此时是在运行屏保则关闭屏保
  Mod_Main.CloseSCR
  End If
  End Sub
  Private Sub Form_MouseMove(Button As Integer, Shift As Integer, X As Single, Y As Single)
  If Mod_Main.Scan_RUN Then
  If (OldX=-1) And (OldY=-1) Then
  OldX=X
  OldY=Y
  Else
  If (ScaleX(Abs(X-OldX),vbTwips,vbPixels)>= 3) Then
  Mod_Main.CloseSCR ’将鼠标当前的水平坐标和垂直坐标与旧鼠标的水平坐标和垂直坐标相减其绝对值如果大于3个像素则退出屏保
  End If
  End If
  End If
  End Sub
  Private Sub Form_Unload(Cancel As Integer)
  Mod_Main.CloseSCR ’关闭屏保
  End Sub
  Private Sub Timer_Mov_Timer()
  If (i>=2) Then
  i=1 ’如果循环变量大于图片的数量则变量赋为1
  Else
  i=i+1 ’否则循环变量加一
  End If
  Frm_Run.PaintPicture Pic(i-1),0,0,Width,Height,0,0,ScaleX(Pic(i-1).Width,vbHimetric,vbTwips),ScaleY(Pic(i-1).Height,vbHimetric,vbTwips)’在Frm_Run上画图
  End Sub
  Frm_Setup:
  Option Explicit
  Private Sub Com_OK_Click()
  Mod_Main.CloseSCR
  End Sub
  Private Sub Form_Unload(Cancel As Integer)
  Mod_Main.CloseSCR
  End Sub
  好了,一个标准的屏幕保护程序就编写好了。按下F5运行试试看。
  -----------------不用API------------------------
  利用VB6.0设计屏幕保护程序
  Windows操作平台设有一个屏幕的保护措施,即屏幕保护功能。经常在Windows操作平台上使用电脑的人们对系统提供给我们的几个屏幕保护程序是不是感到非常平常了,没有新鲜感了,是不是想自己设计屏幕保护程序。下面介绍如何利用VB设计用户自己的屏幕保护程序。屏幕保护程序可以保护显示屏不被损坏,同时节约能源。作为屏幕保护程序,应该具有如下特性:
  1)屏幕保护程序运行时,鼠标光标被自动隐藏,在程序结束时,光标显示。2)当单击、移动鼠标或按下键盘时,屏幕保护结束,回到正常操作状态。为了实现这些特性,在编写VB应用程序时,可以采用如下方法:
  1、VB应用程序的窗体都采用有边框的窗体外观,但作为屏幕保护程序,应设置窗体为无边框,且为最大化。
  2、隐藏及显示鼠标光标在Visual Basic应用程序中隐藏及显示鼠标光标需要运用Windows的API函数,该函数名为ShowCursor。当用参数值True调用时显示鼠标光标,当用参数值False调用时,鼠标光标自动隐藏。
  3、检测鼠标移动VB中有一个检测鼠标移动的对象事件MouseMove事件。MouseMove事件通常在应用程序启动时就会触发,有时在鼠标并未移动的情况下,MouseMove事件仍有可能被触发。因此如果在程序中直接用MouseMove事件检测鼠标是否发生了移动,并不能正确反映鼠标的移动状况。应该在MouseMove事件中编写代码加以控制。
  为了正确反映鼠标的移动,先用变量记录下程序运行时的鼠标当前位置,然后用另外一组变量记录鼠标移动后的位置,当鼠标移动前后的位置差大于一定范围时,触发MouseMove事件。编写代码如下:
  Private Sub Form-MouseMove(Button As Integer,shift As Inteqer,X As Single,Y As Single)
  Static currentX,currentY As Single
  Dim orignX,orignY As Single
  ’把当前的鼠标值赋给orignX和orignY
  orignX=X
  orignY=Y
  ’初始化currentX和currentY
  if currentX=0 and currentY=0 Then
  currentX=orignX
  currentY=orignY
  Exit Sub
  Endif
  ’当鼠标移动大于一个象素时,显示鼠标光标并退出程序
  If Abs(oriqnX-currentX)>1 or Abs(orignY-currentY)>1Then
  X=ShowCursor(True)
  End
  Endif
  EndSub
  4、检测鼠标单击在Visual Basic中,单击事件是由“Click”触发的。当屏幕保护程序运行时遇到单击事件,则程序运行终止。代码编辑如下:
  Private Sub Form-Click()
  X=ShowCursor(True)
  End
  EndSub
  注意在结束之前先设光标的显示为真,以免在程序结束后丢失光标。
  5、检测键盘上各按键的状态Visual Basic中的键盘活动由KeyDown触发。代码与单击事件的代码一样。
  Private Sub Form-KeyDown(KeyCode As Integer,Shift As Integer)
  X=ShowCursor(True)
  End
  EndSub
  下面我们将设计一个简单的屏幕保护程序,该程序运行时,从左至右显示一张图片,图片从屏幕左边出现,至屏幕右面消失,象拉幕一样,且重不停复该过程。假设图片文件名为PIC.BMP,并存放在Windows文件夹中。实际操作如下:
  创建一新工程,在窗体中添加一图片框和一个Timer控件。设置它们的属性如下:
  Form
  BackColor=&H80000007&
  BorderStyle=0 ’None
  MaxButton=False
  MinButton=False
  Windowstate=2 ’Maximized
  Timer
  Intelval=5
  PictureBox
  BackColor=&H80000007&
  BorderStyle=0 ’None
  AutoSize=Ture
  输入代码如下:
  ’在窗体的声明部分声明ShowCursor函数。
  Private Declare Function ShowCursor Lib“user32”(By Val bShow As Long) As Long
  ’在窗体上单击鼠标时退出程序
  Private Sub Form-Click()
  X=ShowCursor(True)
  End
  EndSub
  ’在窗体上按下按键时退出程序
  Private Sub Form-KeyDown(KeyCode As Integer,Shift As Integer) X=ShowCursor(True)
  End
  EndSub
  ’加载窗体时隐藏鼠标
  Private Sub Form-Load()
  Dim X As Long
  X=ShowCursor(False)
  Picture1.Visible=False
  Picture1.PICTure=LoadPICTure(“C:/windows/PIC.BMP”)
  Picture1.Left=-Picture1.Width
  EndSub
  ’在窗体上移动鼠标时退出程序
  Private Sub Form-MouseMove(Button As Integer,Shift As Integer,X As Single,Y As Single)
  Static currentX,currentY As Single
  Dim orignX,orignY As Single
  ’把当前的鼠标值赋给orignX和orignY
  orignX=X
  orignY=Y
  ’初始化currentX和currentY
  If currentX=0 And currentY=0 Then
  currentX=orignX
  currentY=orignY
  ExitSub
  EndIf
  If Abs(orignX-currentX)>1 Or Abs(orignY-currentY)>1
  Then X=ShowCursor(True)
  End
  EndIf
  EndSub
  Private Sub Picture1-Click()
  X=ShowCursor(True)
  End
  EndSub
  Private Sub Picture1-KeyDown(KeyCode As Integer,Shift As Integer)
  X=ShowCursor(True)
  End
  EndSub
  Private Sub Picture1-MouseMove(Button As Integer,Shift As Integer,X As Single,Y As Single)
  Static Xlast,Ylast As Single
  Dim Xnow,Ynow As Single
  Xnow=X
  Ynow=Y
  If Xlast=0 And Ylast=0 Then
  Xlast=Xnow
  Ylast=Ynow
  ExitSub
  EndIf
  If Abs(Xnow-Xlast)>1 Or Abs(Ynow-Ylast)>1 Then
  X=ShowCursor(True)
  End
  EndIf
  EndSub
  Private Sub Timer1-Timer()
  Picture1.Visible=True
  Picture1.Top=(Form1.Height-Picture1.Height)/2
  Picture1.Left=Picture1.Left+50
  If Picture1.Left>Form1.Width Then
  Picture1.Left=-Picture1.Width
  EndIf
  EndSub
  将以上代码编译生成可执行文件,在保存文件对话窗中输入文件名称时把扩展名改为”SCR”,最后将生成的屏幕保护程序添加到Windows的系统下即可。
  参考资料:http://www.softhouse.com.cn/html/200411/2004110514230500001654.html

本文转自
http://zhidao.baidu.com/question/54653050.html?si=1
  • 0
    点赞
  • 0
    评论
  • 0
    收藏
  • 一键三连
    一键三连
  • 扫一扫,分享海报

©️2021 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值