[AccessD] Same form, different actions

John Clark John.Clark at niagaracounty.com
Mon Jan 30 10:10:53 CST 2006


OK John, as usual, you are making my brain hurt ;) Seriously, I have
downloaded your demo, and I am looking through your code, in the
clsOpenArgs module. I think I actually understand some of it too. 

I guess the first thing that comes to mind is, does it work with
non-properties of forms? One of the things I want to do is tell the form
where to go on close...actually, when the exit button is clicked. If the
user goes there from the menu, it should return to the menu. However, if
the user goes there, from anther form, it should return to that form,
and pass the name with it.

I am going to try something, with your demo, to test this. Nope...just
did this real quick, and it didn't work. Actually the whole thing pretty
much quit working...what the heck did I do ;)

>>> jwcolby at colbyconsulting.com 1/29/2006 11:20:21 PM >>>
>>The open Args are quite useful, but tacked on the end of a Do.cmd
statement they don't do much Just text or numbers (usually) tacked on
the
end of an OpenForm Do.Cmd statement.  The real 'power' is in the
interpretation at the other end

Darren,

First let me say that I have put up a demo on my site which shows how
to use
the two classes in the following discussion.  Go to my website -
www.colbyconsulting.com.  If you haven't already, Register.  Log in. 
Click
Example Code / Utilities.  Select C2DbOpenArgs to download the demo.


I pass openargs in the form Varname1=VarVal1;VarName2=VarVal2;etc=etc1;
This
is a syntax that has been around since early Access days and was used
extensively by, and perhaps even "invented by" Ken Getz.  The = is used
as
the separator, the semicolon as the delimiter.

I wrote a pair of classes which handle my openargs.  A "control class"
which
I name OpenArgs (plural) grabs the openargs string and parses the
varname
and value for each openarg.  It then instantiates a OpenArg (singular)
class
to store this VarName and VarVal in.  VarName is a string type, VarVal
is a
variant type.  Making it Variant causes an automatic coercion in a lot
of
cases - currency values get turned into Currency variants, dates get
turned
into dates etc.  At any rate, each OpenArg instance is stored in a
collection in the OpenArgs (plural) class.

The OpenArgs class is dimensioned in the form header, and instantiated
in
the OnOpen of the form.  By the time it finishes loading, all of the
OpenArgs are parsed and sitting in OpenArg class instances in the
collection
in OpenArgs.  OpenArgs has a method for reading OpenArg instances from
the
collection.  OpenArg instances are stored in the collection keyed on
the
VarName.  

Thus you have a pair of classes which automatically handles getting
openargs
for you, one or a hundred, it doesn't matter, and it doesn't have to
be
"re-invented" every time you want to get your openargs.  Syntax is now
standardized everywhere in your app.

The form of course must know what openargs it is expecting, so it can
now
simply call OpenArgs.Arg("VarName") and get back VarVal (a variant). 
What
the form does with the OpenArg is up to you, but getting the OpenArgs
available to the form is trivial.

All of the code follows.  My clsOpenArgs (plural, the controller) also
has
the ability to automatically interpret openargs as properties of the
form.
In other words, if a param is passed in to clsOpenArgs that says to do
so,
then if an OpenArg VarName is the same as a property of the form, the
form
property is set to VarVal.  I have actually demoed this in the state
form,
where the form that opens frmState passes in the values for properties


strOpenArgs = "DataEntry=True;AllowEdits=True;AllowDeletions=False;"

The state form opens, the OpenArgs class parses the OpenArgs string
passed
in, discovers that I want it to use the openargs to set form
properties, and
does so.

Any OpenArgs which do not match the name of a form property are left
uninterpreted and the form can use them for whatever purpose you
choose.
You can also not tell clsOpenArgs to interpret the openargs as form
properties, in which case all OpenArgs are left uninterpreted and you
can do
with them all as you will.

The code:

'===========================================================
'Class OpenArg
'===========================================================
Option Compare Database
Option Explicit

'
'This class stores one OpenArg
'
Private mstrArgName As String
Private mvarArgVal As Variant
Private mblnIsProperty As Boolean

Function mInit(lstrArgName As String, lvarArgVal As Variant)
    mstrArgName = lstrArgName
    mvarArgVal = lvarArgVal
End Function
Function pName() As String
    pName = mstrArgName
End Function
Function pVal() As Variant
    pVal = mvarArgVal
End Function
Property Let pIsPrp(lmblnIsProperty As Boolean)
    mblnIsProperty = lmblnIsProperty
End Property
Property Get pIsPrp() As Boolean
    pIsPrp = mblnIsProperty
End Property


'===========================================================
'Class OpenArgs (Plural) - the controller class
'===========================================================
Option Compare Database
Option Explicit
'.
'.Written By   : John W. Colby
'.Date Created : 04/10/2004
' Rev. History :
'
' BEHAVIORS:
'
'
'When a form opens, it can accept open args, a text string argument.
'The framework will check for openargs, and if they exist will check
'inside the Openagrs for arguments named the same thing as form
properties.
'If such arguments are found, the framework will set the form's
property
'equal to the argument value.
'
'I use this specifically to set up normal forms to be DataEntry forms
'(only allow new records to be added, not existing records edited)
'
'I decided to encapsulate this functionality in a class to make the
OpenArgs
'processing cleanly defined.
'
'--------------------------------------------------------------------------
'THESE CONSTANTS AND VARIABLES ARE USED INTERNALLY TO THE CLASS
'*+ Class constant declaration
Private Const mcstrModuleName As String = "clsOpenArgs"
'*- Class constants declaration
'*+ Class variables declarations
'*- Class variables declarations
'.-------------------------------------------------------------------------
'THESE CONSTANTS AND VARIABLES ARE USED BY THE CLASS TO IMPLEMENT
CLASS
FUNCTIONALITY
'*+ custom constants declaration
'
'*- Custom constants declaration
'*+ custom variables declarations
'
Private mfrm As Form     'A form reference passed in
Private mstrOpenArgs As String
Private mcolOpenArg As Collection
'*- custom variables declarations
'
'Define any events this class will raise here
'*+ custom events Declarations
'Public Event MyEvent(Status As Integer)
'*- custom events declarations
'.-------------------------------------------------------------------------
'THESE FUNCTIONS / SUBS ARE USED INTERNALLY TO THE CLASS
'*+ Private Init/Terminate Interface
Private Sub Class_Initialize()
On Error GoTo Err_Class_Initialize
    Set mcolOpenArg = New Collection
Exit_Class_Initialize:
Exit Sub
Err_Class_Initialize:
        MsgBox Err.Description, , "Error in Sub
clsTemplate.Class_Initialize"
        Resume Exit_Class_Initialize
    Resume 0    '.FOR TROUBLESHOOTING
End Sub
Private Sub Class_Terminate()
On Error Resume Next
    mTerm
End Sub
'INITIALIZE THE CLASS
Public Sub mInit(lfrm As Form, _
                Optional blnApplyProperties As Boolean = False)
    Set mfrm = lfrm
    'The openargs string might be null
    On Error Resume Next
    mstrOpenArgs = mfrm.OpenArgs
    
    ParseOpenArgs
    '
    'The default is false, do not try and apply OpenArgs as form
properties
    'If the deveopler wants to
    '
    If blnApplyProperties Then
        ApplyFrmProperties
    End If
End Sub
'CLEAN UP ALL OF THE CLASS POINTERS
Public Sub mTerm()
Static blnRan As Boolean    'The term may run more than once so
    If blnRan Then Exit Sub 'just exit if it already ran
    blnRan = True
    On Error Resume Next
    mColEmpty mcolOpenArg
    Set mcolOpenArg = Nothing
End Sub
'*- Public Init/Terminate interface
'.-------------------------------------------------------------------------
'THESE FUNCTIONS SINK EVENTS DECLARED WITHEVENTS IN THIS CLASS
'*+ Form WithEvent interface
'*- Form WithEvent interface
'THESE FUNCTIONS / SUBS ARE USED TO IMPLEMENT CLASS FUNCTIONALITY
'*+PRIVATE Class function / sub declaration
Private Function cOpenArg(strOpenArg As String) As clsOpenArg
Dim intPosEqual As Integer
Dim strArgName As String
Dim varArgVal As Variant
Dim lclsOpenArg As clsOpenArg
    intPosEqual = InStr(strOpenArg, "=")
    If intPosEqual > 0 Then
        strArgName = Left$(strOpenArg, intPosEqual - 1)
        varArgVal = Right$(strOpenArg, Len(strOpenArg) - intPosEqual)
        Set lclsOpenArg = New clsOpenArg
        lclsOpenArg.mInit strArgName, varArgVal
        Set cOpenArg = lclsOpenArg
    End If
End Function
'
Private Sub ParseOpenArgs()
Dim lstrOpenArgs As String
Dim strOpenArg As String
Dim intPos As Integer
Dim lclsOpenArg As clsOpenArg
    lstrOpenArgs = mstrOpenArgs
    intPos = InStr(lstrOpenArgs, ";")
    
    While intPos > 0
        strOpenArg = Left$(lstrOpenArgs, intPos - 1)
        Set lclsOpenArg = OpenArg(strOpenArg)
        mcolOpenArg.Add lclsOpenArg, lclsOpenArg.pName
        lstrOpenArgs = Right$(lstrOpenArgs, Len(lstrOpenArgs) -
intPos)
        intPos = InStr(lstrOpenArgs, ";")
    Wend

End Sub
'
'This function cycles through all the openargs applying them to form
properties
'if an argument is named the same as a form property, and the property
is
writeable
'(doesn't require being in design view to set it) then the application
of
the value
'to the property will be performed and Err will not be set.  For these
OpenArgs we
'set the IsPrp to true
'
'All of this provides a way for the developer to ppass in openargs to
an
opening form
'which are then used to set form properties.  It is up to the developer
to
ensure that
'the property is settable, that the value they pass in is valid
(correct
data type,
'correct value range etc.)
'
'In the end, the only way to know whether a passed in OpenArg is a
property
is to try
'it and see.  If there is no error then the name is a property name, it
is
settable in
'form view mode, and the value is acceptable.
'
Public Sub ApplyFrmProperties()
Dim lclsOpenArg As clsOpenArg
    On Error Resume Next
    For Each lclsOpenArg In mcolOpenArg
        mfrm.Properties(lclsOpenArg.pName) = lclsOpenArg.pVal
        lclsOpenArg.pIsPrp = (Err.Number = 0)
        Err.Clear
    Next lclsOpenArg
End Sub
'
'Empties out a collection containing class instances
'
Public Function mColEmpty(col As Collection)
On Error GoTo Err_mColEmpty

Dim obj As Object
On Error Resume Next
    
    For Each obj In col
        obj.mTerm
    Next obj

On Error GoTo Err_mColEmpty
    While col.Count > 0
        col.Remove 1
    Wend
exit_mColEmpty:
Exit Function
Err_mColEmpty:
    Select Case Err
    Case 91 'Collection empty
        Resume exit_mColEmpty
    Case Else
        MsgBox Err.Description, , "Error in Function
clsSysVars.mColEmpty"
        Resume exit_mColEmpty
    End Select
    Resume 0    '.FOR TROUBLESHOOTING
End Function

'*-PRIVATE Class function / sub declaration
'*+PUBLIC Class function / sub declaration
Function OpenArg(strArgName As String) As Variant
On Error GoTo Err_OpenArg
    OpenArg = mcolOpenArg(strArgName).pVal
Exit_OpenArg:
Exit Function
Err_OpenArg:
    Select Case Err
    Case 91
        MsgBox "OpenArg " & strArgName & " does not exist"
        Resume Exit_OpenArg
    Case Else
        MsgBox Err.Description, , "Error in Function
clsOpenArgs.OpenArg"
        Resume Exit_OpenArg
    End Select
    Resume 0    '.FOR TROUBLESHOOTING
End Function'*-PUBLIC Class function / sub declaration

'===========================================================
'Form frmStates - a demo form that displays a state lookup table
'===========================================================
Option Compare Database
Option Explicit

Private lclsOpenArgs As clsOpenArgs

Private Sub Form_Open(Cancel As Integer)
    Set lclsOpenArgs = New clsOpenArgs
    lclsOpenArgs.mInit Me, True
End Sub
Private Sub cmdClose_Click()
On Error GoTo Err_cmdClose_Click

    DoCmd.Close

Exit_cmdClose_Click:
    Exit Sub
Err_cmdClose_Click:
    MsgBox Err.Description
    Resume Exit_cmdClose_Click
End Sub

'===========================================================
'Form frmOpenArgsDemo - a demo form that opens the state form, 
'passing in openargs
'===========================================================
Option Compare Database
Option Explicit

Private Sub cmdStatesDataEntry_Click()
On Error GoTo Err_cmdStatesDataEntry_Click

    Dim stDocName As String
    Dim stLinkCriteria As String
    Dim strOpenArgs As String
    
    strOpenArgs =
"DataEntry=True;AllowEdits=True;AllowDeletions=False;"
    stDocName = "frmStates"
    DoCmd.OpenForm stDocName, , , stLinkCriteria, , , strOpenArgs

Exit_cmdStatesDataEntry_Click:
    Exit Sub

Err_cmdStatesDataEntry_Click:
    MsgBox Err.Description
    Resume Exit_cmdStatesDataEntry_Click
    Resume 0
End Sub

John W. Colby
www.ColbyConsulting.com 


-----Original Message-----
From: accessd-bounces at databaseadvisors.com 
[mailto:accessd-bounces at databaseadvisors.com] On Behalf Of Darren DICK
Sent: Sunday, January 29, 2006 5:29 PM
To: 'Access Developers discussion and problem solving'
Subject: Re: [AccessD] Same form, different actions

Hi John
Anyone feel free to jump in if I have this wrong 

The open Args are quite useful, but tacked on the end of a Do.cmd
statement
they don't do much Just text or numbers (usually) tacked on the end of
an
OpenForm Do.Cmd statment

The real 'power' is in the interpretation at the other end

.....

-- 
AccessD mailing list
AccessD at databaseadvisors.com 
http://databaseadvisors.com/mailman/listinfo/accessd 
Website: http://www.databaseadvisors.com



More information about the AccessD mailing list