Finally a Timer class in VBA

tofo

New member
Local time
Today, 05:07
Joined
Aug 20, 2012
Messages
1
Wanted multiple timer events without having to use a form timer.

Have been accepting that callback functions can not be used in a VBA class like msaccess until got a hint from vb.mvps.org

Using the following declarations in my VBA class:

Code:
Private Declare Function SetTimer Lib "user32" ( _
    ByVal hWnd As Long, ByVal nIDEvent As Long, _
    ByVal uElapse As Long, ByVal lpTimerFunc As Long) As Long

Private Declare Function KillTimer Lib "user32" ( _
    ByVal hWnd As Long, ByVal nIDEvent As Long) As Long


Public Event OnTimer()

Private TimerID As Long

'Start timer
Public Sub Startit(IntervalMs As Long)
    TimerID = SetTimer(Application.hWndAccessApp, ObjPtr(Me), IntervalMs, AddressOf Timers.TimerProc)
End Sub

'Stop timer
Public Sub Stopit()
    If TimerID <> -1 Then
        KillTimer 0&, TimerID
        TimerID = 0
    End If
End Sub


'Trigger Public event
Public Sub RaiseTimerEvent()
    RaiseEvent OnTimer
End Sub


Please notice the use of the call to undocumented VBA function ObjPtr(Me) with the instance of the class when SetTimer is called.

Then created a module named "Timers" to place the callback function:

Code:
Public Sub TimerProc(ByVal hWnd As Long, _
                     ByVal uMsg As Long, _
                     ByVal oTimer As clsTimer, _
                     ByVal dwTime As Long)
   ' Alert appropriate timer object instance.
   If Not oTimer Is Nothing Then
     oTimer.RaiseTimerEvent
   End If
End Sub


This is all that you need. The callback function TimerProc() will receive the class instance from the timer callback and calls RaiseTimerEvent() that will raise OnTimer() event of the particular instance.

Any number of timers can now be used in your modules or classes declared like this:

Code:
Dim WithEvents HappyTimer As clsTimer

'Start the timer
Private Sub StartMyTimer()
  Set HappyTimer = New clsTimer
  HappyTimer.Startit 2000
End Sub

'Using the timer event from the timer
Private Sub HappyTimer_OnTimer()
    Debug.Print "Timer Event " & Now
End Sub
 
For anyone who stumbles across this code (like I did) please note that the code, as written will not kill the timer!

To stop a timer via KillTimer, you need to pass it the hWnd that it was created with.

i.e. in this particular case:
Code:
 'Stop timer 
Public Sub Stopit()     
If TimerID <> -1 Then         
    KillTimer Application.hWndAccessApp, TimerID         
    TimerID = 0     
End If 
End Sub
Personally, I'd put the hWnd into a variable to store it.
 
Thanks for your code!

I have used this code on my navigation pane in my access db.
it works well but when i leave the form the timer stops also.

How can i let the timer continue when the user opens other forms ?

Regards Martin
 
Hi,

This looks quite promising. I have a long running query (over 5 minutes) initiated from a form. I want to display some type of progress bar while the query is running. The standard Access progress bar is not prominent enough. I tried using the form Timer Event, but when the query starts, the timer event is negated (because VBA is single threaded and can only do one thing at a time). Can you please provide a bit more detail. Where does the first part of your code go? A Class module? Somewhere else? How about the second snippet of code? I assume the last snippet of code goes somewhere in for form's code.

Cheers
 
Caution! I found that you not only need to do what ainwood says but also:
Code:
Dim WithEvents HappyTimer As clsTimer
needs to be changed to
Code:
Public WithEvents HappyTimer As clsTimer
Although it may look like it stops, it doesn't! The declaration has to be made public, otherwise access loses it and it never stops running in the background. You can prove whether its running or not by trying to enter some VBA like "me. ..." if your dropdown options disappear, it is still running an OnTimer in the background!
Hope this helps someone, I got really worried for a moment.
-Jonathan M.
 

Users who are viewing this thread

Back
Top Bottom