March 16th, 2007

Coding a Euchre Game, Part 4: Tooltips and Menus (Matt Gertz)

Coding a Euchre Game, Part 4: Tooltips and Menus

Menus and tooltips are so incredibly important to applications, and yet implementing them sometimes seems to be arcane magic.  In this post, I’m going to do my best to demystify them.

Menus

To add a menu, you simply drag a MenuStrip control from the toolbox to your form.  As with the Timer control from the last post, the control will end up in the grey area at the bottom of the form; however, there will also be a visible component at the top of the form.  Use the former to change the name of the control, but mostly you’ll be working with the visual instance at the top.

Now, if you wanted to quickly add a whole set of standard menu items such as File, Edit, Tools, etc., you could click on the smart tag in the upper-right corner of the menustrip and choose “Insert Standard Menu Items,” thus populating the menu with those menus automatically.  However, we don’t want standard menus for this game, as most of the menu items would not be relevant.  Instead, place your cursor in the editbox on the menustrip and type “&VBEuchre.”  This creates a Drop-down menu called VBEuchre. Note how the ampersand (&) forces the following character to be underlined!  That’s the character that will automatically be used when navigating with the ALT key.  You can also add commands under the “VBEuchre” menu item by typing into the edit boxes.  You can specify ALT mnemonics using the ampersand character, or can assign key shortcuts by selecting the menu item and setting its various ShortcutKey properties.  You can even define secondary tiers of menus, though we won’t be doing that for this game.  The attached picture shows what menu-editing looks like.

Adding command handlers to menu items is identical to adding them to any other control – just double-click them to create the handler and then add your code to it.  For “New Game,” this Is very simple:

    Private Sub NewGameToolStripMenuItem_Click(ByVal sender _

As System.Object, ByVal e As System.EventArgs) _

Handles NewGameToolStripMenuItem.Click

        NewGame()

    End Sub

 

NewGame() then has to check to see if a game is already under way, and pops up a “Do you want to quit your current game?” messagebox if so:

    Public Sub NewGame()

        If GameStarted = True Then

            If MessageBox.Show(My.Resources.Command_New,_

 My.Resources.Command_NewTitle, MessageBoxButtons.OKCancel, _

 MessageBoxIcon.Question, MessageBoxDefaultButton.Button2) <> _

 Windows.Forms.DialogResult.OK Then

                Return

            End If

        End If

 

        If InvokedOnce = False Then

            NewGameInvoked()

        Else

            ‘ NewGameInvoked is already on the stack.  We don’t want to recurse into it.

            ‘ Throw back up to that method so it can restart itself.

            newEuchre = True

        End If

    End Sub

Note that my variables “GameStarted” and “InvokedOnce” are checking two different things.  “GameStarted” means that play is actually happening.  “InvokedOnce” is simply a re-entrancy check – the gameplay might not have started yet, but the user might nevertheless be involved in other code in NewGameInvoked such as setting the options or whatever.  (In other words, if GameStarted is True, then InvokedOnce must be True, but the converse is not necessarily true.)  The “newEuchre” variable will be used exactly like the “exitEuchre” variable from the previous post – it causes the timer to be turned off and an exception to be thrown to start a new game, so those methods now become:

    Private Sub TableSleep()

        StopPumpingMessagesDuringPause = False

        EuchreTimer.Start()

        While StopPumpingMessagesDuringPause = False

            Application.DoEvents()

        End While

        If exitEuchre = True Then

            Throw New EuchreException(“ExitGame”)

        End If

        If newEuchre = True Then

            Throw New EuchreException(“NewGame”)

        End If

    End Sub

    Private Sub NewGameInvoked()

Begin: Try

            Do While exitEuchre = False

 

‘ (this is where the game actually happens –

   I’ve omitted the actual code for brevity’s sake)

 

            Loop

        Catch ex As EuchreException

            ‘ Reset the flag and let it go back through to

            ‘ the loop to restart the game

            If ex.Message = “NewGame” Then

                newEuchre = False

                GoTo Begin

            End If

            If ex.Message = “ExitGame” Then

                exitEuchre = False

                ‘ Do nothing; fall through and exit the application

            End If

        End Try

    End Sub

I usually detest using GoTo statements, actually, but in this case it made sense to do so.

Tooltips

Tooltips are another control which get dragged to the form but are acted upon in the grey area.  The only changes that need to be made to its properties are changing the name and setting AlwaysShow to True.  It seems counterintuitive, but you only need one tooltip per form.  This is because the tooltip keeps a list of what each control’s tooltip should be.  (This is where many people get confused – they expect the tooltip to be a property on each control, but that’s not the way it works.)

To set a tooltip for a control is a very simple thing.  Recall that the labels which represent the places where cards can go on the table are referenced in an array called TableTopCards.   Let’s pretend that a card is face down and we want to set a tooltip for it.  The tooltip string is in the resource manager and is named CARDNAME_BACK.  We would then set the tooltip like this:

    EuchreTooltip.SetToolTip(Me.TableTopCards(player, slot), _

          My.Resources.CARDNAME_BACK)

 

And we could clear it like this

    EuchreTooltip.SetToolTip(Me.TableTopCards(player, slot), _

          Nothing)

 

Most of the tooltips resource string names in Euchre need to be constructed based on the name and suit of the card, are are thus referenced similar to the image resource names I presented several posts ago.  However, there’s a catch:  If the Right Bower or Left Bower (the Jack of the trump suit or the Jack in the suit of the same color as the trump suit) are involved, then I want the tooltip to call them Bowers instead of their actual name, so that the user remembers that they are trump cards.  So, we add a method in EuchreCard thusly:

    Public Function GetDisplayStringResourceName(Optional ByVal TrumpSuit _

As Suits = Suits.NoSuit) As String

        If State = States.FaceUp OrElse Table.PeekAtOtherCards = True _

OrElse Table.SelectingDealer Then

            Dim FileName As New StringBuilder()

            If TrumpSuit <> Suits.NoSuit AndAlso _

(Me.GetValue(TrumpSuit) = Values.LeftBower _

OrElse Me.GetValue(TrumpSuit) = Values.RightBower) Then

                If Me.GetValue(TrumpSuit) = Values.RightBower Then

                    FileName.Append(“CARDNAMERightBower”)

                    Return FileName.ToString

                Else

                    FileName.Append(“CARDNAMELeftBower”)

                    Return FileName.ToString

                End If

            Else

                FileName.AppendFormat(“CARDNAME{0}Of{1}”, Rank, Suit)

                Return FileName.ToString

            End If

        Else

            Return “CARDNAME_BACK”

        End If

    End Function

 

And now we use the resulting string as follows to set the tooltip for an arbitrary card at position “i” in the player’s hand:

      EuchreTooltip.SetToolTip(

Me.TableTopCards(EuchrePlayer.Seats.Player, i), _

 My.Resources.ResourceManager.GetString( _

Players(EuchrePlayer.Seats.Player).CardsHeldThisHand(i _

).GetDisplayStringResourceName))

 

That’s it for today.  Next time, we’ll discuss modality and user controls.  Until then…

–Matt–*

EuchreMenuEdit.jpg

0 comments