01-05-2025, 07:33 PM (This post was last modified: 01-05-2025, 07:59 PM by NakedApe.)
(01-05-2025, 05:27 PM)grymmjack Wrote: Hey @nakedape I haven't had a chance to check after you told me the right way to use the mouse. I'll check it again.
I have plans to review it in a YouTube video too.
Just need to find time and ambition at once
Thanks vm @grymmjack. Sounds great! I found your first impressions of the game from June here (#4) and you liked it a lot back then. Make sure to set the game difficulty to Easy since you're a Rock Jockey rookie. And good luck finding that time and ambition.
I'll post a slightly updated version of the game that gives a message if you try to steer wrong by moving the mouse up and down - not side to side - and sets the default difficulty to easy (since some find it hard to play).
Excellent, a740g, I'm glad it worked for you - and nice flying!
MacOS has a built-in speech synthesizer, and I used a free, little app called "Speech" from the Mac App Store. The voice is US female "Samantha." Type in what you want her to say and save to file. Then I used WavePad Masters Edition to edit the VOs and all the other sound files... It was a fun project and I learned a lot!
01-08-2025, 02:12 PM (This post was last modified: 01-08-2025, 02:14 PM by madscijr.)
(12-17-2024, 12:08 AM)NakedApe Wrote: And now, just in time for X-Mas fun, here's my space game. I posted version one of this arcade-style game as a WIP six months ago. Now I think it's done - I think I've actually finished writing a computer game - omg! Many thanks to Terry Ritchie, MasterGy, bplus and Amazing Steve for the routines I stole - and attributed to them.
Thanks for sharing this!
I finally got around to running it, didn't have a ton of time to go too deep, but here is some initial feedback, if it helps?
This is a great looking game!
The different screens / chapters are very interesting looking and have me very curious to try. I don't know if the ones that show the planet's surface use different game mechanics (ie the game changes to a Lunar Lander or Gravitar type rules) but if it did that would be cool.
The idea for the controls is original, but for a game like this, I really think just stick to the tried and true Asteroids / Spacewar! model, where left/right = rotate, up = thrust, down = other action, and the ship builds momentum. Having to deal with new/awkward controls, where the ship moves this slowly, just adds frustration for me.
I haven't looked at the code yet, but if this can be tweaked to be configurable to be a sort of Asteroids / Spacewar / Lunar Lander / Gravitar construction set, that would be absolutely killer (for me at least)!
I'll definitely play with it as I get time. I have a Spacewar game in the works that stalled a while back, and your game really inspires me to go back and finally finish it!
Hey @madscijr. Thanks for downloading, playing and commenting! I'm glad you mostly like it. Without getting defensive or anything, lemme respond.
> I tried using the arrow keys for this at first, but found them too limiting and clunky. It's hard for me to switch from the up key to the down key in real time, whereas the mouse is adjustable in sensitivity and (I think) more natural for this kind of play - your mileage may vary. You might benefit from trying a practice round to get a feel for the controls.
> Not sure why you think the ship moves slowly, but I realize these are your initial reactions. The ship in deep space mode (no gravity) can go from -1 to 6 pixels per cycle, which is way faster than needed. The left and right mouse buttons control the thrusters. This game is about accurate maneuvering much of the time so high speed isn't really that useful - unless you're trying to break out of the force field cage! Hint, hint.
> The game does go into lunar lander mode once you grab a rock and make it through the comets and rock field. It was fun trying to recreate one of my favorite games from 1982.
> I'm not a very advanced programmer, though I'm tenacious and learning more all the time thanks to this forum. The code may not lend itself to modding very easily, but it's well documented so maybe. I tried to prevent spaghetti coding, still there sure are a lot of events and variables to track in this wacky, multi-chapter game.
Give it another try if you find the time please. And good luck finishing your Spacewar game! I wanna play.
(01-08-2025, 05:25 PM)NakedApe Wrote: Hey @madscijr. Thanks for downloading, playing and commenting! I'm glad you mostly like it. Without getting defensive or anything, lemme respond.
> I tried using the arrow keys for this at first, but found them too limiting and clunky. It's hard for me to switch from the up key to the down key in real time, whereas the mouse is adjustable in sensitivity and (I think) more natural for this kind of play - your mileage may vary. You might benefit from trying a practice round to get a feel for the controls.
> Not sure why you think the ship moves slowly, but I realize these are your initial reactions. The ship in deep space mode (no gravity) can go from -1 to 6 pixels per cycle, which is way faster than needed. The left and right mouse buttons control the thrusters. This game is about accurate maneuvering much of the time so high speed isn't really that useful - unless you're trying to break out of the force field cage! Hint, hint.
> The game does go into lunar lander mode once you grab a rock and make it through the comets and rock field. It was fun trying to recreate one of my favorite games from 1982.
> I'm not a very advanced programmer, though I'm tenacious and learning more all the time thanks to this forum. The code may not lend itself to modding very easily, but it's well documented so maybe. I tried to prevent spaghetti coding, still there sure are a lot of events and variables to track in this wacky, multi-chapter game.
Give it another try if you find the time please. And good luck finishing your Spacewar game! I wanna play.
Thanks NakedApe for explaining all that, I'll definitely give it a more thorough play-through when I have some time to dedicate to it. Maybe I'm "doing it wrong" with the controls and didn't know how to speed up? I'll definitely try again. Perhaps building in some control options so people who prefer to use just the keyboard or just the mouse or gamepad will open it up to more players? Anyway, you've done some very inpressive work for someone not advanced! Being tenacious is a very good trait or approach for this kind of thing - keep it up and you'll go far. Cheers!
madscijr: "Perhaps building in some control options so people who prefer to use just the keyboard or just the mouse or gamepad will open it up to more players?"
That's a good idea. I could easily have the arrow keys *and* the mouse do what you're saying and then leave the side thruster controls for the letters 'a' and 'd'. But then I'd have to update the directions. Yawn... No, I like the idea and will make the changes.
(01-08-2025, 11:17 PM)NakedApe Wrote: madscijr: "Perhaps building in some control options so people who prefer to use just the keyboard or just the mouse or gamepad will open it up to more players?"
That's a good idea. I could easily have the arrow keys *and* the mouse do what you're saying and then leave the side thruster controls for the letters 'a' and 'd'. But then I'd have to update the directions. Yawn... No, I like the idea and will make the changes.
Thanks for the kind words too.
Better yet, give the user the option to map their own keys for each function. Don't like 'a' and 'd'? No problem! Choose whichever keys you want!
If you need help with that, I could dig up some code, or you can search the forums or at qb64.com, it's a common practice.
8 hours ago(This post was last modified: 8 hours ago by madscijr.)
(Yesterday, 05:57 AM)NakedApe Wrote: Since you're kind enough to offer, sure, @madscijr, I'd take a look at your key mapping code. Thanks vm!
Sure! I normally use _BUTTON to detect keypresses because it's less laggy and works well detecting multiple simultaneous keypresses (disclaimer: others may have different experiences/opinions on that).
In the first program see Sub ReadKeyboardWithButtonOnDevice1, which checks every key code, and a "keydown" event is detected for any, displays it. It's currently set up so pressing ESC and then ENTER exits.
For key mapping, you could put that logic in a reusable function like
Function MapKey%(description$)
that display a message on the screen like
PRINT "Press the key you would like to use for " + $description + " or ESC to cancel",
then loop until a key is pressed (or user cancels). When a key is detected, exit for and exit do,
then have the function return the detected key code and save the key code to the variable like UpKeyCode%.
Call the function for all the controls, e.g.,
UpKeyCode% = MapKey%("Up")
DownKeyCode% = MapKey%("Down")
LeftKeyCode% = MapKey%("Left")
RightKeyCode% = MapKey%("Right")
etc.
Code: (Select All)
_Title "keyboard_button_test_2-00-03.bas"
' ################################################################################################################################################################
' Keycode detection test (_BUTTON + DEVICES)
Const FALSE = 0
Const TRUE = Not FALSE
Dim Shared m_ProgramPath$: m_ProgramPath$ = Left$(Command$(0), _InStrRev(Command$(0), "\"))
Dim Shared m_ProgramName$: m_ProgramName$ = Mid$(Command$(0), _InStrRev(Command$(0), "\") + 1)
main
System
' /////////////////////////////////////////////////////////////////////////////
Sub main
Dim in$: in$ = ""
Do
Cls
Print m_ProgramName$
Print
Print "Keycode Detection Test " + m_VersionInfo$
Print "by Softintheheadware (Jan, 2022)"
Print
Print "Test of different methods of detecting keypresses"
Print
Print "1. Detect keys using _BUTTON with _DeviceInput(1)"
Print
Print "2. Enumerate _DEVICES"
Print
Print "What to do ('q' to exit)"
Input in$: in$ = LCase$(Left$(in$, 1))
If in$ = "1" Then
ReadKeyboardWithButtonOnDevice1
ElseIf in$ = "2" Then
EnumerateDevices
End If
Loop Until in$ = "q"
End Sub ' main
' ################################################################################################################################################################
' BEGIN READ KEYBOARD Device #1 _Button
' ################################################################################################################################################################
' /////////////////////////////////////////////////////////////////////////////
'DEVICES Button
'_LASTBUTTON(1) keyboards will normally return 512 buttons. One button is read per loop through all numbers.
'_BUTTONCHANGE(number) returns -1 when pressed, 1 when released and 0 when there is no event since the last read.
'_BUTTON(number) returns -1 when a button is pressed and 0 when released
' Detects most keys (where the codes are documented?)
' However, does not seem to detect:
' F10
' Alt
' Left Alt
' Right Alt
' Print Screen
' Pause/Break
Sub ReadKeyboardWithButtonOnDevice1
Dim iLoop As Integer
Dim iCode As Integer
Dim iLastPressed As Integer ' useful for tracking key up/key down state
Dim iPreviousKey As Integer ' use to track multi-key presses, track separately than last pressed
Dim bFinished As Integer
Cls
Print "Press a key to see what _BUTTON code is detetected."
Print
Print
Print
Print "(Press <ESC> then <ENTER> to exit)."
_KeyClear: _Delay 1
iLastPressed = -1
iPreviousKey = -1
Do
While _DeviceInput(1): Wend ' clear and update the keyboard buffer
' Detect changed key state
For iLoop = 1 To 512
iCode = _Button(iLoop)
' If the last key pressed is still held down, don't keep printing the code
If (iLoop <> iLastPressed) Then
If iCode <> 0 Then
' Quit if user pressed Enter right after Esc
If iPreviousKey = 2 Then
If iLoop = 29 Then
bFinished = TRUE
Exit For
End If
End If
Cls
Print "Press a key to see what _BUTTON code is detetected."
Print
Print "Detected key press [" + KeyDescription$(iLoop) + "], _BUTTON(" + _Trim$(Str$(iLoop)) + ") = " + _Trim$(Str$(iCode))
Print
Print "(Press <ESC> then <ENTER> to exit)."
iLastPressed = iLoop
iPreviousKey = iLoop
End If
Else
' If last key is released, clear the code so it can be pressed again:
If iCode = 0 Then
iLastPressed = -1
End If
End If
Next iLoop
'_LIMIT 100
'LOOP UNTIL _KEYDOWN(27)
Loop Until bFinished = TRUE
_KeyClear: '_Delay 1 ' CLEAR KEYBOARD BUFFER
End Sub ' ReadKeyboardWithButtonOnDevice1
' /////////////////////////////////////////////////////////////////////////////
' Example: Checking for the system's input devices.
' _DEVICES FUNCTION (QB64 REFERENCE)
' http://www.qb64.net/wiki/index_title_DEVICES/
' ^^^ abandoned, outdated and now likely malicious qb64 dot net website - don’t go there
'
' The _DEVICES function returns the number of INPUT devices on your computer
' including keyboard, mouse and game devices.
'
' Syntax:
'
' device_count% = _DEVICES
'
' Returns the number of devices that can be listed separately with the _DEVICE$
' function by the device number.
' Devices include keyboard, mouse, joysticks, game pads and multiple stick game
' controllers.
' Note: This function MUST be read before trying to use the _DEVICE$,
' _DEVICEINPUT or _LAST control functions!
' Note: The STRIG/STICK commands won't read from the keyboard
' or mouse device the above example lists.
Sub EnumerateDevices
Dim devices%
Dim iLoop%
Dim sCount$
Dim iLen As Integer
devices% = _Devices ' MUST be read in order for other 2 device functions to work!
Cls
Print "Total devices found: "; Str$(devices%)
For iLoop% = 1 To devices%
iLen = 4
sCount$ = Left$(LTrim$(RTrim$(Str$(iLoop%))) + String$(iLen, " "), iLen)
Print sCount$ + _Device$(iLoop%) + " (" + LTrim$(RTrim$(Str$(_LastButton(iLoop%)))) + " buttons)"
Next iLoop%
Print
Print "PRESS ANY KEY TO CONTINUE"
Sleep
_KeyClear: '_Delay 1 ' CLEAR KEYBOARD BUFFER
End Sub ' EnumerateDevices
' ################################################################################################################################################################
' END READ KEYBOARD Device #1 _Button
' ################################################################################################################################################################
' ################################################################################################################################################################
' BEGIN KEYBOARD CODE FUNCTIONS
' NOTE: ALL CODES ARE FOR _BUTTON, EXCEPT:
' cF10 (_KEYDOWN)
' cAltLeft (_KEYHIT)
' cAltRight (_KEYHIT)
' cPrintScreen (_KEYHIT) <- may slow down pc?
' cPauseBreak (_KEYHIT) <- may not work?
' ################################################################################################################################################################
Function KeyCode_Escape% ()
KeyCode_Escape% = 2
End Function
Function KeyCode_F1% ()
KeyCode_F1% = 60
End Function
Function KeyCode_F2% ()
KeyCode_F2% = 61
End Function
Function KeyCode_F3% ()
KeyCode_F3% = 62
End Function
Function KeyCode_F4% ()
KeyCode_F4% = 63
End Function
Function KeyCode_F5% ()
KeyCode_F5% = 64
End Function
Function KeyCode_F6% ()
KeyCode_F6% = 65
End Function
Function KeyCode_F7% ()
KeyCode_F7% = 66
End Function
Function KeyCode_F8% ()
KeyCode_F8% = 67
End Function
Function KeyCode_F9% ()
KeyCode_F9% = 68
End Function
'_KEYDOWN CODE, NOT _BUTTON CODE
Function KeyCode_F10% ()
KeyCode_F10% = 17408
End Function
Function KeyCode_F11% ()
KeyCode_F11% = 88
End Function
Function KeyCode_F12% ()
KeyCode_F12% = 89
End Function
'_KEYHIT CODE, NOT _BUTTON CODE
Function KeyCode_PrintScreen% ()
KeyCode_PrintScreen% = -44
End Function
Function KeyCode_ScrollLock% ()
KeyCode_ScrollLock% = 71
End Function
'_KEYHIT CODE, NOT _BUTTON CODE
Function KeyCode_PauseBreak% ()
KeyCode_PauseBreak% = 31053
End Function
Function KeyCode_Tilde% ()
KeyCode_Tilde% = 42
End Function
Function KeyCode_1% ()
KeyCode_1% = 3
End Function
Function KeyCode_2% ()
KeyCode_2% = 4
End Function
Function KeyCode_3% ()
KeyCode_3% = 5
End Function
Function KeyCode_4% ()
KeyCode_4% = 6
End Function
Function KeyCode_5% ()
KeyCode_5% = 7
End Function
Function KeyCode_6% ()
KeyCode_6% = 8
End Function
Function KeyCode_7% ()
KeyCode_7% = 9
End Function
Function KeyCode_8% ()
KeyCode_8% = 10
End Function
Function KeyCode_9% ()
KeyCode_9% = 11
End Function
Function KeyCode_0% ()
KeyCode_0% = 12
End Function
Function KeyCode_Minus% ()
KeyCode_Minus% = 13
End Function
Function KeyCode_Equal% ()
KeyCode_Equal% = 14
End Function
Function KeyCode_BkSp% ()
KeyCode_BkSp% = 15
End Function
Function KeyCode_Ins% ()
KeyCode_Ins% = 339
End Function
Function KeyCode_Home% ()
KeyCode_Home% = 328
End Function
Function KeyCode_PgUp% ()
KeyCode_PgUp% = 330
End Function
Function KeyCode_Del% ()
KeyCode_Del% = 340
End Function
Function KeyCode_End% ()
KeyCode_End% = 336
End Function
Function KeyCode_PgDn% ()
KeyCode_PgDn% = 338
End Function
Function KeyCode_NumLock% ()
KeyCode_NumLock% = 326
End Function
Function KeyCode_KeypadSlash% ()
KeyCode_KeypadSlash% = 310
End Function
Function KeyCode_KeypadMultiply% ()
KeyCode_KeypadMultiply% = 56
End Function
Function KeyCode_KeypadMinus% ()
KeyCode_KeypadMinus% = 75
End Function
Function KeyCode_Keypad7Home% ()
KeyCode_Keypad7Home% = 72
End Function
Function KeyCode_Keypad8Up% ()
KeyCode_Keypad8Up% = 73
End Function
Function KeyCode_Keypad9PgUp% ()
KeyCode_Keypad9PgUp% = 74
End Function
Function KeyCode_KeypadPlus% ()
KeyCode_KeypadPlus% = 79
End Function
Function KeyCode_Keypad4Left% ()
KeyCode_Keypad4Left% = 76
End Function
Function KeyCode_Keypad5% ()
KeyCode_Keypad5% = 77
End Function
Function KeyCode_Keypad6Right% ()
KeyCode_Keypad6Right% = 78
End Function
Function KeyCode_Keypad1End% ()
KeyCode_Keypad1End% = 80
End Function
Function KeyCode_Keypad2Down% ()
KeyCode_Keypad2Down% = 81
End Function
Function KeyCode_Keypad3PgDn% ()
KeyCode_Keypad3PgDn% = 82
End Function
Function KeyCode_KeypadEnter% ()
KeyCode_KeypadEnter% = 285
End Function
Function KeyCode_Keypad0Ins% ()
KeyCode_Keypad0Ins% = 83
End Function
Function KeyCode_KeypadPeriodDel% ()
KeyCode_KeypadPeriodDel% = 84
End Function
Function KeyCode_Tab% ()
KeyCode_Tab% = 16
End Function
Function KeyCode_Q% ()
KeyCode_Q% = 17
End Function
Function KeyCode_W% ()
KeyCode_W% = 18
End Function
Function KeyCode_E% ()
KeyCode_E% = 19
End Function
Function KeyCode_R% ()
KeyCode_R% = 20
End Function
Function KeyCode_T% ()
KeyCode_T% = 21
End Function
Function KeyCode_Y% ()
KeyCode_Y% = 22
End Function
Function KeyCode_U% ()
KeyCode_U% = 23
End Function
Function KeyCode_I% ()
KeyCode_I% = 24
End Function
Function KeyCode_O% ()
KeyCode_O% = 25
End Function
Function KeyCode_P% ()
KeyCode_P% = 26
End Function
Function KeyCode_BracketLeft% ()
KeyCode_BracketLeft% = 27
End Function
Function KeyCode_BracketRight% ()
KeyCode_BracketRight% = 28
End Function
Function KeyCode_Backslash% ()
KeyCode_Backslash% = 44
End Function
Function KeyCode_CapsLock% ()
KeyCode_CapsLock% = 59
End Function
Function KeyCode_A% ()
KeyCode_A% = 31
End Function
Function KeyCode_S% ()
KeyCode_S% = 32
End Function
Function KeyCode_D% ()
KeyCode_D% = 33
End Function
Function KeyCode_F% ()
KeyCode_F% = 34
End Function
Function KeyCode_G% ()
KeyCode_G% = 35
End Function
Function KeyCode_H% ()
KeyCode_H% = 36
End Function
Function KeyCode_J% ()
KeyCode_J% = 37
End Function
Function KeyCode_K% ()
KeyCode_K% = 38
End Function
Function KeyCode_L% ()
KeyCode_L% = 39
End Function
Function KeyCode_Semicolon% ()
KeyCode_Semicolon% = 40
End Function
Function KeyCode_Apostrophe% ()
KeyCode_Apostrophe% = 41
End Function
Function KeyCode_Enter% ()
KeyCode_Enter% = 29
End Function
Function KeyCode_ShiftLeft% ()
KeyCode_ShiftLeft% = 43
End Function
Function KeyCode_Z% ()
KeyCode_Z% = 45
End Function
Function KeyCode_X% ()
KeyCode_X% = 46
End Function
Function KeyCode_C% ()
KeyCode_C% = 47
End Function
Function KeyCode_V% ()
KeyCode_V% = 48
End Function
Function KeyCode_B% ()
KeyCode_B% = 49
End Function
Function KeyCode_N% ()
KeyCode_N% = 50
End Function
Function KeyCode_M% ()
KeyCode_M% = 51
End Function
Function KeyCode_Comma% ()
KeyCode_Comma% = 52
End Function
Function KeyCode_Period% ()
KeyCode_Period% = 53
End Function
Function KeyCode_Slash% ()
KeyCode_Slash% = 54
End Function
Function KeyCode_ShiftRight% ()
KeyCode_ShiftRight% = 55
End Function
Function KeyCode_Up% ()
KeyCode_Up% = 329
End Function
Function KeyCode_Left% ()
KeyCode_Left% = 332
End Function
Function KeyCode_Down% ()
KeyCode_Down% = 337
End Function
Function KeyCode_Right% ()
KeyCode_Right% = 334
End Function
Function KeyCode_CtrlLeft% ()
KeyCode_CtrlLeft% = 30
End Function
Function KeyCode_WinLeft% ()
KeyCode_WinLeft% = 348
End Function
' _KEYHIT CODE NOT _BUTTON CODE
Function KeyCode_AltLeft% ()
KeyCode_AltLeft% = -30764
End Function
Function KeyCode_Spacebar% ()
KeyCode_Spacebar% = 58
End Function
' _KEYHIT CODE NOT _BUTTON CODE
Function KeyCode_AltRight% ()
KeyCode_AltRight% = -30765
End Function
Function KeyCode_WinRight% ()
KeyCode_WinRight% = 349
End Function
Function KeyCode_Menu% ()
KeyCode_Menu% = 350
End Function
Function KeyCode_CtrlRight% ()
KeyCode_CtrlRight% = 286
End Function
' /////////////////////////////////////////////////////////////////////////////
' Receives
' KeyCode% = the key code to get the description for
Function KeyDescription$ (KeyCode%)
Dim MyString As String
Select Case (KeyCode%)
Case KeyCode_Escape%:
MyString = "Escape"
Case KeyCode_F1%:
MyString = "F1"
Case KeyCode_F2%:
MyString = "F2"
Case KeyCode_F3%:
MyString = "F3"
Case KeyCode_F4%:
MyString = "F4"
Case KeyCode_F5%:
MyString = "F5"
Case KeyCode_F6%:
MyString = "F6"
Case KeyCode_F7%:
MyString = "F7"
Case KeyCode_F8%:
MyString = "F8"
Case KeyCode_F9%:
MyString = "F9"
Case KeyCode_F10%: ' 17408 = _KEYDOWN CODE, NOT _BUTTON CODE
MyString = "F10"
Case KeyCode_F11%:
MyString = "F11"
Case KeyCode_F12%:
MyString = "F12"
Case KeyCode_PrintScreen%: ' -44 = _KEYHIT CODE, NOT _BUTTON CODE
MyString = "Print Screen"
Case KeyCode_ScrollLock%:
MyString = "Scroll Lock"
Case KeyCode_PauseBreak%: ' 31053 = _KEYHIT CODE, NOT _BUTTON CODE
MyString = "Pause/Break"
Case KeyCode_Tilde%:
MyString = "`"
Case KeyCode_1%:
MyString = "1"
Case KeyCode_2%:
MyString = "2"
Case KeyCode_3%:
MyString = "3"
Case KeyCode_4%:
MyString = "4"
Case KeyCode_5%:
MyString = "5"
Case KeyCode_6%:
MyString = "6"
Case KeyCode_7%:
MyString = "7"
Case KeyCode_8%:
MyString = "8"
Case KeyCode_9%:
MyString = "9"
Case KeyCode_0%:
MyString = "0"
Case KeyCode_Minus%:
MyString = "-"
Case KeyCode_Equal%:
MyString = "="
Case KeyCode_BkSp%:
MyString = "Backspace"
Case KeyCode_Ins%:
MyString = "Insert"
Case KeyCode_Home%:
MyString = "Home"
Case KeyCode_PgUp%:
MyString = "Page Up"
Case KeyCode_Del%:
MyString = "Delete"
Case KeyCode_End%:
MyString = "End"
Case KeyCode_PgDn%:
MyString = "Page Down"
Case KeyCode_NumLock%:
MyString = "Num Lock"
Case KeyCode_KeypadSlash%:
MyString = "[keypad] /"
Case KeyCode_KeypadMultiply%:
MyString = "[keypad] *"
Case KeyCode_KeypadMinus%:
MyString = "[keypad] -"
Case KeyCode_Keypad7Home%:
MyString = "[keypad] 7/Home"
Case KeyCode_Keypad8Up%:
MyString = "[keypad] 8"
Case KeyCode_Keypad9PgUp%:
MyString = "[keypad] 9/PgUp"
Case KeyCode_KeypadPlus%:
MyString = "[keypad] +"
Case KeyCode_Keypad4Left%:
MyString = "[keypad] 4/Left"
Case KeyCode_Keypad5%:
MyString = "[keypad] 5"
Case KeyCode_Keypad6Right%:
MyString = "[keypad] 6/Right"
Case KeyCode_Keypad1End%:
MyString = "[keypad] 1/End"
Case KeyCode_Keypad2Down%:
MyString = "[keypad] 2/Down"
Case KeyCode_Keypad3PgDn%:
MyString = "[keypad] 3/PgDn"
Case KeyCode_KeypadEnter%:
MyString = "[keypad] Enter"
Case KeyCode_Keypad0Ins%:
MyString = "[keypad] 0/Ins"
Case KeyCode_KeypadPeriodDel%:
MyString = "[keypad] ./Del"
Case KeyCode_Tab%:
MyString = "Tab"
Case KeyCode_Q%:
MyString = "Q"
Case KeyCode_W%:
MyString = "W"
Case KeyCode_E%:
MyString = "E"
Case KeyCode_R%:
MyString = "R"
Case KeyCode_T%:
MyString = "T"
Case KeyCode_Y%:
MyString = "Y"
Case KeyCode_U%:
MyString = "U"
Case KeyCode_I%:
MyString = "I"
Case KeyCode_O%:
MyString = "O"
Case KeyCode_P%:
MyString = "P"
Case KeyCode_BracketLeft%:
MyString = "["
Case KeyCode_BracketRight%:
MyString = "]"
Case KeyCode_Backslash%:
MyString = "\"
Case KeyCode_CapsLock%:
MyString = "Caps Lock"
Case KeyCode_A%:
MyString = "A"
Case KeyCode_S%:
MyString = "S"
Case KeyCode_D%:
MyString = "D"
Case KeyCode_F%:
MyString = "F"
Case KeyCode_G%:
MyString = "G"
Case KeyCode_H%:
MyString = "H"
Case KeyCode_J%:
MyString = "J"
Case KeyCode_K%:
MyString = "K"
Case KeyCode_L%:
MyString = "L"
Case KeyCode_Semicolon%:
MyString = ";"
Case KeyCode_Apostrophe%:
MyString = "'"
Case KeyCode_Enter%:
MyString = "Enter"
Case KeyCode_ShiftLeft%:
MyString = "Shift Left"
Case KeyCode_Z%:
MyString = "Z"
Case KeyCode_X%:
MyString = "X"
Case KeyCode_C%:
MyString = "C"
Case KeyCode_V%:
MyString = "V"
Case KeyCode_B%:
MyString = "B"
Case KeyCode_N%:
MyString = "N"
Case KeyCode_M%:
MyString = "M"
Case KeyCode_Comma%:
MyString = ","
Case KeyCode_Period%:
MyString = "."
Case KeyCode_Slash%:
MyString = "/"
Case KeyCode_ShiftRight%:
MyString = "Shift Right"
Case KeyCode_Up%:
MyString = "Up"
Case KeyCode_Left%:
MyString = "Left"
Case KeyCode_Down%:
MyString = "Down"
Case KeyCode_Right%:
MyString = "Right"
Case KeyCode_CtrlLeft%:
MyString = "Ctrl Left"
Case KeyCode_WinLeft%:
MyString = "Win Left"
Case KeyCode_AltLeft%: ' -30764 = _KEYHIT CODE NOT _BUTTON CODE
MyString = "Alt Left"
Case KeyCode_Spacebar%:
MyString = "Spacebar"
Case KeyCode_AltRight%: ' -30765 = _KEYHIT CODE NOT _BUTTON CODE
MyString = "Alt Right"
Case KeyCode_WinRight%:
MyString = "Win Right"
Case KeyCode_Menu%:
MyString = "Menu"
Case KeyCode_CtrlRight%:
MyString = "Ctrl Right"
Case Else:
' UNKNOWN VALUE
MyString = "_BUTTON(" + _Trim$(Str$(KeyCode%)) + ")"
End Select ' KeyCode%
KeyDescription$ = MyString
End Function ' KeyDescription$
' ################################################################################################################################################################
' END KEYBOARD CODE FUNCTIONS
' ################################################################################################################################################################
' ################################################################################################################################################################
' BEGIN GENERAL PURPOSE ROUTINES
' ################################################################################################################################################################
' /////////////////////////////////////////////////////////////////////////////
' IIF function for QB for integers
Function IIF (Condition, IfTrue, IfFALSE)
If Condition Then IIF = IfTrue Else IIF = IfFALSE
End Function
' /////////////////////////////////////////////////////////////////////////////
' IIF function for QB for strings
Function IIFSTR$ (Condition, IfTrue$, IfFALSE$)
If Condition Then IIFSTR$ = IfTrue$ Else IIFSTR$ = IfFALSE$
End Function
' /////////////////////////////////////////////////////////////////////////////
' returns random number
Function RandomNum (num)
RandomNum = Int(Rnd * num)
End Function 'RandomNum
' /////////////////////////////////////////////////////////////////////////////
' Generate random value between Min and Max.
Function RandomNumber% (Min%, Max%)
Dim NumSpread%
' SET RANDOM SEED
'Randomize ' Initialize random-number generator.
Randomize Timer
' GET RANDOM # Min%-Max%
'RandomNumber = Int((Max * Rnd) + Min) ' generate number
NumSpread% = (Max% - Min%) + 1
RandomNumber% = Int(Rnd * NumSpread%) + Min% ' GET RANDOM # BETWEEN Max% AND Min%
End Function ' RandomNumber%
' /////////////////////////////////////////////////////////////////////////////
Function StrPadLeft$ (sValue As String, iWidth As Integer)
StrPadLeft$ = Right$(String$(iWidth, " ") + sValue, iWidth)
End Function ' StrPadLeft$
' /////////////////////////////////////////////////////////////////////////////
Function StrPadRight$ (sValue As String, iWidth As Integer)
StrPadRight$ = Left$(sValue + String$(iWidth, " "), iWidth)
End Function ' StrPadRight$
' ################################################################################################################################################################
' END GENERAL PURPOSE ROUTINES
' ################################################################################################################################################################
If you prefer _KEYDOWN or _KEYHIT, here is some test code for those methods:
Code: (Select All)
_Title "keyboard_test_1-02.keydown+keyhit+button.bas"
' ################################################################################################################################################################
' Keycode detection test
'
' CHANGE LOG:
' Date Who What
' 12/16/2020 madscijr completed mostly working stable version 0.70
' 01/08/2022 madscijr updated so ESC+ENTER exits each test, added change log + cleaned up for release
' DESCRIPTION:
' A little script to show which keycodes are detected for the different keys
' on the keyboard using 3 different methods (_KEYDOWN, _KEYHIT, _BUTTON).
' While none of them seems to be able to detect every key, _BUTTON catches
' most of them (except for F10 and right/left ALT). So I will use _BUTTON
' to catch most of the codes, and _KEYDOWN to detect F10, and _KEYHIT for ALT.
' Here are the results for each method, compiled here in a handy table for you:
' Key _BUTTON _KEYDOWN _KEYHIT
' --------------------- ----------- -------- -----------------------------------------------------
' Esc 2 27 27
' F1 60 15104 15104
' F2 61 15360 15360
' F3 62 15616 15616
' F4 63 15872 15872
' F5 64 16128 16128
' F6 65 16384 16384
' F7 66 16640 16640
' F8 67 16896 16896
' F9 68 17152 17152
' F10 ? 17408 17408
' F11 88 ? -31488
' F12 89 ? -31232
' SysReq (Print Screen) ? ? (-44 on press/release, seems to slow down PC)
' ScrL (Scroll Lock) 71 ? (-145 on press/release, seems to slow down PC)
' Pause / Break ? ? (31053 momentarily on pressing another key)
' ~ 42 96 96
' 1! 3 49 49
' 2@ 4 50 50
' 3# 5 51 51
' 4$ 6 52 52
' 0.05 7 53 53
' 6^ 8 54 54
' 7& 9 55 55
' 8* 10 56 56
' 9( 11 57 57
' 0) 12 48 48
' -_ 13 45 45
' =+ 14 61 61
' BkSp 15 8 8
' Ins 339 20992 20992
' Home 328 18176 18176
' PgUp 330 18688 18688
' Del 340 21248 21248
' End 336 20224 20224
' PgDn 338 20736 20736
' NumLock 326 ? 30772 (flip flops with -30772 on release, -144 after)
' KEYPAD / 310 (47) 47
' KEYPAD * 56 (42) 42
' KEYPAD - 75 (45) 45
' KEYPAD 7/Home 72 18176 18176
' KEYPAD 8 Up 73 18432 18432
' KEYPAD 9 PgUp 74 18688 18688
' KEYPAD + 79 (43) 43
' KEYPAD 4 Left 76 19200 19200
' KEYPAD 5 77 ? ? (-12 after release)
' KEYPAD 6 Right 78 19712 19712
' KEYPAD 1 End 80 20224 20224
' KEYPAD 2 Down 81 20480 20480
' KEYPAD 3 PgDn 82 20736 20736
' KEYPAD ENTER 285 (13) 13
' KEYPAD 0 Ins 83 20992 20992
' KEYPAD . Del 84 21248 21248
' Tab 16 9 9
' Q 17 113 113
' W 18 119 119
' E 19 101 101
' R 20 114 114
' T 21 116 116
' Y 22 121 121
' U 23 117 117
' I 24 105 105
' O 25 111 111
' P 26 112 112
' [{ 27 91 91
' ]} 28 93 93
' \| 44 92 92
' Caps Lock 59 ? 30771 (flip flops with -30771 on release, -20 after)
' A 31 97 97
' S 32 115 115
' D 33 100 100
' F 34 102 102
' G 35 103 103
' H 36 104 104
' J 37 106 106
' K 38 107 107
' L 39 108 108
' ;: 40 59 59
' '" 41 39 39
' Enter 29 13 13
' Left Shift 43 ? -30768 (-16 on release)
' Shift (44 or 55) ? (-30768 or -30769)
' Z 45 22 22
' X 46 120 120
' C 47 99 99
' V 48 118 118
' B 49 98 98
' N 50 110 110
' M 51 109 109
' ,< 52 44 44
' .> 53 46 46
' /? 54 47 47
' Right Shift 55 ? -30769 (-16 on release)
' Up 329 18432 18432
' Left 332 19200 19200
' Down 337 20480 20480
' Right 334 19712 19712
' Left Ctrl 30 ? -30766 (-17 on release)
' Ctrl (30 or 286) ? (-30766 or -30767)
' Left Win 348 ? ? (-91 after release)
' Left Alt ? ? -30764 (-18 on release)
' Alt ? ? (-30764 or -30765)
' Spacebar 58 32 32
' Right Alt ? ? -30765 (-18 on release)
' Right Win 349 ? ? (-92 after release)
' Menu 350 ? ? (-93 after release)
' Right Ctrl 286 ? -30767 (-17 on release)
' =============================================================================
' GLOBAL DECLARATIONS a$=string, i%=integer, L&=long, s!=single, d#=double
' =============================================================================
' GLOBAL VARIABLES
Dim Shared m_ProgramPath$: m_ProgramPath$ = Left$(Command$(0), _InStrRev(Command$(0), "\"))
Dim Shared m_ProgramName$: m_ProgramName$ = Mid$(Command$(0), _InStrRev(Command$(0), "\") + 1)
Dim Shared m_VersionInfo$: m_VersionInfo$ = "1.00"
' =============================================================================
' RUN TEST
main
' =============================================================================
' FINISH
System ' return control to the operating system
Print m_ProgramName$ + " finished."
End
Sub main
Dim in$: in$ = ""
Do
Cls
Print m_ProgramName$
Print
Print "Keycode Detection Test " + m_VersionInfo$
Print "by Softintheheadware (Jan, 2022)"
Print
Print "Test of different methods of detecting keypresses"
Print
Print "1. Test using _KEYDOWN"
Print
Print "2. Test using _KEYHIT"
Print
Print "3. Test using _DEVICE commands"
Print
Print "4. Enumerate _DEVICES"
Print
Print "What to do ('q' to exit)"
Input in$: in$ = LCase$(Left$(in$, 1))
If in$ = "1" Then
KeyboardKeydownTest
ElseIf in$ = "2" Then
KeyboardKeyhitTest
ElseIf in$ = "3" Then
KeyboardDeviceTest
ElseIf in$ = "4" Then
EnumerateDevices
End If
Loop Until in$ = "q"
End Sub ' main
' /////////////////////////////////////////////////////////////////////////////
' MOSTLY WORKS TO RETURN KEYHIT and KEYDOWN Codes listed at
' https://www.qb64.org/wiki/Keyboard_scancodes#KEYHIT_and_KEYDOWN_Codes
' However does not seem to detect:
' F11
' F12
' SysReq (Print Screen)
' ScrL (Scroll Lock)
' Pause / Break
' NumLock
' KEYPAD 5
' Caps Lock
' Left Shift
' Shift
' Right Shift
' Left Ctrl
' Ctrl
' Left Win
' Left Alt
' Alt
' Right Alt
' Right Win
' Menu
' Right Ctrl
' Mouse Left
' Mouse Middle
' Mouse Right
' KEYPAD ENTER returns same code as regular ENTER key (13)
' KEYPAD "-" returns same code as regular "-" key (45)
' KEYPAD "/" returns same code as regular enter key (47)
Sub KeyboardKeydownTest
Dim iLoop As Integer
Dim iCode As Integer
Dim iLastPressed As Integer ' useful for tracking key up/key down state
Dim iPreviousKey As Integer ' use to track multi-key presses, track separately than last pressed
Dim bFinished As Integer
Cls
Print "Press a key to see what _KEYDOWN code is detetected."
Print
Print
Print
Print "(Press <ESC> then <ENTER> to exit)."
_KeyClear: _Delay 1
iLastPressed = -1
iPreviousKey = -1
bFinished = FALSE
Do
For iLoop = 1 To 32767
iCode = _KeyDown(iLoop)
' If the last key pressed is still held down, don't keep printing the code
If (iLoop <> iLastPressed) Then
'IF _KEYDOWN(iLoop) THEN
If iCode = TRUE Then
' Quit if user pressed Enter right after Esc
If iLoop = 13 Then
If iPreviousKey = 27 Then
bFinished = TRUE
Exit For
End If
End If
Cls
Print "Press a key to see what _KEYDOWN code is detetected."
Print
Print "Detected key press with _KEYDOWN(" + Str$(iLoop) + ") = " + Str$(iCode)
Print
Print "(Press <ESC> then <ENTER> to exit)."
iLastPressed = iLoop
iPreviousKey = iLoop
End If
Else
' If last key is released, clear the code so it can be pressed again:
If iCode = FALSE Then
iLastPressed = -1
End If
End If
Next iLoop
'_LIMIT 100
'LOOP UNTIL _KEYDOWN(27)
Loop Until bFinished = TRUE
Sub KeyboardKeyhitTest
Dim iLoop As Integer
Dim iCode As Integer
Dim iLastPressed As Integer
Dim iKey As Integer
Dim sMessage As String
Dim z$
Dim bFinished As Integer
Cls
Print "Press a key to see what _KEYHIT code is detetected."
Print
Print
Print
Print "(Press <ESC> then <ENTER> to exit)."
_KeyClear: _Delay 1
iLastPressed = 0
bFinished = FALSE
Do
iCode = _KeyHit
If iCode <> 0 Then
If iLastPressed <> iCode Then
If iCode > 0 Then
' Quit if user pressed Enter right after Esc
If iCode = 13 Then
If iLastPressed = 27 Then
bFinished = TRUE
Exit Do
End If
End If
' positive value means key pressed
sMessage = "Detected key pressed with _KEYHIT = " + cstr$(iCode)
Else
' negative value means key released
sMessage = "Detected key released with _KEYHIT = " + cstr$(iCode)
iCode = -iCode ' get code of key released
End If
If iCode < 256 Then ' ASCII code values
sMessage = sMessage + ", ASCII"
If iCode > 31 Then
sMessage = sMessage + " " + Chr$(34) + Chr$(iCode) + Chr$(34)
Else
sMessage = sMessage + " (UNPRINTABLE)"
End If
ElseIf iCode > 255 And iCode < 65536 Then ' 2 byte key codes
sMessage = sMessage + ", 2-BYTE-COMBO (" + cstr$(iCode And 255) + "," + cstr$(iCode \ 256) + ")"
iKey = iCode \ 256
If iKey > 31 And iKey < 256 Then
sMessage = sMessage + " " + Chr$(34) + Chr$(iKey) + Chr$(34)
End If
ElseIf iCode > 99999 And iCode < 200000 Then ' QB64 Virtual Key codes
sMessage = sMessage + ", QB64 SDL Virtual Key code (" + cstr$(iCode - 100000) + ")"
ElseIf iCode >= &H40000000 Then ' Unicode values (IME Input mode)
Print "IME input mode, UNICODE (" + cstr$(iCode - &H40000000) + "0x" + Hex$(iCode - &H40000000) + ")"
' The MKL$ function encodes a LONG numerical value into a 4-byte ASCII STRING value.
z$ = MKL$(iCode - &H40000000) + MKL$(0)
sMessage = sMessage + " " + z$ + z$ + z$
End If
Cls
Print "Press a key to see what _KEYHIT code is detetected."
Print
Print sMessage
Print
Print "(Press <ESC> then <ENTER> to exit)."
iLastPressed = iCode
End If
End If ' iCode <> 0
'_LIMIT 100
'LOOP UNTIL _KEYDOWN(27)
Loop Until bFinished = TRUE
_KeyClear
End Sub ' KeyboardKeyhitTest
' /////////////////////////////////////////////////////////////////////////////
'DEVICES Button
'_LASTBUTTON(1) keyboards will normally return 512 buttons. One button is read per loop through all numbers.
'_BUTTONCHANGE(number) returns -1 when pressed, 1 when released and 0 when there is no event since the last read.
'_BUTTON(number) returns -1 when a button is pressed and 0 when released
' Detects most keys (where the codes are documented?)
' However, does not seem to detect:
' F10
' Alt
' Left Alt
' Right Alt
' Print Screen
' Pause/Break
Sub KeyboardDeviceTest
Dim iLoop As Integer
Dim iCode As Integer
Dim iLastPressed As Integer ' useful for tracking key up/key down state
Dim iPreviousKey As Integer ' use to track multi-key presses, track separately than last pressed
Dim bFinished As Integer
Cls
Print "Press a key to see what _BUTTON code is detetected."
Print
Print
Print
Print "(Press <ESC> then <ENTER> to exit)."
_KeyClear: _Delay 1
iLastPressed = -1
iPreviousKey = -1
Do
While _DeviceInput(1): Wend ' clear and update the keyboard buffer
' Detect changed key state
For iLoop = 1 To 512
iCode = _Button(iLoop)
' If the last key pressed is still held down, don't keep printing the code
If (iLoop <> iLastPressed) Then
If iCode <> 0 Then
' Quit if user pressed Enter right after Esc
If iPreviousKey = 2 Then
If iLoop = 29 Then
bFinished = TRUE
Exit For
End If
End If
Cls
Print "Press a key to see what _BUTTON code is detetected."
Print
Print "Detected key press with _BUTTON(" + Str$(iLoop) + ") = " + Str$(iCode)
Print
Print "(Press <ESC> then <ENTER> to exit)."
iLastPressed = iLoop
iPreviousKey = iLoop
End If
Else
' If last key is released, clear the code so it can be pressed again:
If iCode = 0 Then
iLastPressed = -1
End If
End If
Next iLoop
'_LIMIT 100
'LOOP UNTIL _KEYDOWN(27)
Loop Until bFinished = TRUE
_KeyClear
End Sub ' KeyboardDeviceTest
' /////////////////////////////////////////////////////////////////////////////
' Same as QB64's str$ function, except removes the annoying space that
' str$ prepends to the result.
' For example:
' PRINT CHR$(34) + STR$(5) + CHR$(34)
' outputs
' " 5"
' If you do a lot of str$, you find yourself having to use ltrim$ a lot,
' which can make your code harder to read.
' With this function, something that would look like
' sRightCount = LEFT$(LTRIM$(RTRIM$(STR$(arrInfo(iIndex).RightCount))) + STRING$(iLen, " "), iLen)
' becomes easier to read:
' sRightCount = LEFT$(cstr$(arrInfo(iIndex).RightCount) + STRING$(iLen, " "), iLen)
' And cstr is easy to remember for those familiar with vbscript/VBA/VB6/ASP.
Function cstr$ (myValue)
cstr$ = LTrim$(RTrim$(Str$(myValue)))
End Function ' cstr$
' /////////////////////////////////////////////////////////////////////////////
' Example: Checking for the system's input devices.
' _DEVICES FUNCTION (QB64 REFERENCE)
' http://[www.qb64.net]/wiki/index_title_DEVICES/
' ^^^ abandoned, outdated and now likely malicious qb64 dot net website - don’t go there
'
' The _DEVICES function returns the number of INPUT devices on your computer
' including keyboard, mouse and game devices.
'
' Syntax:
'
' device_count% = _DEVICES
'
' Returns the number of devices that can be listed separately with the _DEVICE$
' function by the device number.
' Devices include keyboard, mouse, joysticks, game pads and multiple stick game
' controllers.
' Note: This function MUST be read before trying to use the _DEVICE$,
' _DEVICEINPUT or _LAST control functions!
' Note: The STRIG/STICK commands won't read from the keyboard
' or mouse device the above example lists.
Sub EnumerateDevices
Dim devices%
Dim iLoop%
Dim sCount$
Dim iLen As Integer
devices% = _Devices ' MUST be read in order for other 2 device functions to work!
Cls
Print "Total devices found: "; Str$(devices%)
For iLoop% = 1 To devices%
iLen = 4
sCount$ = Left$(LTrim$(RTrim$(Str$(iLoop%))) + String$(iLen, " "), iLen)
Print sCount$ + _Device$(iLoop%) + " (" + LTrim$(RTrim$(Str$(_LastButton(iLoop%)))) + " buttons)"
Next iLoop%
Print
Print "PRESS <ESC> TO CONTINUE"
Do: Loop Until _KeyDown(27) ' leave loop when ESC key pressed
_KeyClear: '_DELAY 1
End Sub ' EnumerateDevices
This earlier version also included INKEY:
Code: (Select All)
_Title "keyboard_test_1-00.keyhit+inkey.bas"
' ################################################################################################################################################################
' #CONSTANTS = GLOBAL CONSTANTS
' ################################################################################################################################################################
' UDT TO HOLD COLOR CODE INFO
Type ColorToTextType
color As _Unsigned Long
txt As String
End Type ' ColorToTextType
' ################################################################################################################################################################
' #VARS = GLOBAL VARIABLES
' ENABLE / DISABLE DEBUG CONSOLE
Dim Shared m_bTesting As Integer: m_bTesting = FALSE
' BASIC PROGRAM METADATA
Dim Shared m_ProgramPath$: m_ProgramPath$ = Left$(Command$(0), _InStrRev(Command$(0), "\"))
Dim Shared m_ProgramName$: m_ProgramName$ = Mid$(Command$(0), _InStrRev(Command$(0), "\") + 1)
Dim Shared m_VersionInfo$: m_VersionInfo$ = "1.00"
' =============================================================================
' LOCAL VARIABLES
Dim in$
' ****************************************************************************************************************************************************************
' ACTIVATE DEBUGGING WINDOW
If m_bTesting = TRUE Then
$Console
_Delay 4
_Console On
_Echo "Started " + m_ProgramName$
_Echo "Debugging on..."
End If
' ****************************************************************************************************************************************************************
' =============================================================================
' START THE MAIN ROUTINE
main
' ****************************************************************************************************************************************************************
' DEACTIVATE DEBUGGING WINDOW
If m_bTesting = TRUE Then
_Console Off
End If
' ****************************************************************************************************************************************************************
System ' return control to the operating system
End
If in$ = "1" Then
result$ = KeyHitTest1$
ElseIf in$ = "2" Then
result$ = InKeyTest1$
ElseIf in$ = "3" Then
result$ = "UNDER CONSTRUCTION"
ElseIf in$ = "4" Then
result$ = ""
End If
If Len(result$) > 0 Then
Print result$
End If
Loop Until in$ = "q"
' RETURN TO TEXT SCREEN
Screen 0
End Sub ' main
Function InKeyTest1$
Dim k$
Cls
Print "Press a key or <ESC> to exit."
Do
k$ = InKey$
If k$ <> "" Then
Print "You pressed " + Chr$(34) + k$ + Chr$(34) + "."
End If
Loop Until k$ = Chr$(27) ' Esc key exit
_KeyClear
InKeyTest1$ = ""
End Function ' InKeyTest1$
Function KeyHitTest1$
Dim RoutineName As String: RoutineName = "KeyHitTest1$"
Dim in$
Dim iKey As Long
Dim iKey2 As Long
Cls
Print "Press a key or <ESC> to exit."
iKey = 0
Do
iKey = _KeyHit
If iKey Then
If iKey < 0 Then ' negative value means key released
Print "Released ";
iKey = -iKey
Else
Print "Pressed "; ' positive value means key pressed
End If
If iKey < 256 Then ' ASCII code values
Print "ASCII "; iKey;
If iKey >= 32 And iKey <= 255 Then
Print "[" + Chr$(iKey) + "]"
Else
Print
End If
End If
If iKey >= 256 And iKey < 65536 Then ' 2 byte key codes
Print "2-BYTE-COMBO "; iKey And 255; iKey \ 256;
iKey2 = iKey \ 256
If iKey2 >= 32 And iKey2 <= 255 Then
Print "[" + Chr$(x2) + "]"
Else
Print
End If
End If
If iKey >= 100000 And iKey < 200000 Then ' QB84 Virtual Key codes
Print "SDL VK"; iKey - 100000
End If
If iKey >= 200000 And iKey < &H40000000 Then
Print "QB64 VK"; iKey - 200000
End If
If iKey >= &H40000000 Then 'Unicode values (IME Input mode)
Print "UNICODE "; iKey - &H40000000; "0x" + Hex$(iKey - &H40000000) + " ...";
in$ = MKL$(iKey - &H40000000) + MKL$(0)
Print in$ + in$ + in$;
Print
End If
End If
Loop Until iKey = 27
_KeyClear
KeyHitTest1$ = ""
End Function ' KeyHitTest1$
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN DEBUGGING ROUTINES #DEBUGGING
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Sub DebugPrint (s$)
If m_bTesting = TRUE Then
_Echo s$
'ReDim arrLines$(0)
'dim delim$ : delim$ = Chr$(13)
'split MyString, delim$, arrLines$()
End If
End Sub ' DebugPrint
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END DEBUGGING ROUTINES @DEBUGGING
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN GENERAL PURPOSE ROUTINES #GEN
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' /////////////////////////////////////////////////////////////////////////////
' Convert a value to string and trim it (because normal Str$ adds spaces)
Function cstr$ (myValue)
'cstr$ = LTRIM$(RTRIM$(STR$(myValue)))
cstr$ = _Trim$(Str$(myValue))
End Function ' cstr$
' /////////////////////////////////////////////////////////////////////////////
' Convert a Long value to string and trim it (because normal Str$ adds spaces)
Function cstrl$ (myValue As Long)
cstrl$ = _Trim$(Str$(myValue))
End Function ' cstrl$
' /////////////////////////////////////////////////////////////////////////////
' Convert a Single value to string and trim it (because normal Str$ adds spaces)
Function cstrs$ (myValue As Single)
''cstr$ = LTRIM$(RTRIM$(STR$(myValue)))
cstrs$ = _Trim$(Str$(myValue))
End Function ' cstrs$
' /////////////////////////////////////////////////////////////////////////////
' Convert an unsigned Long value to string and trim it (because normal Str$ adds spaces)
Function cstrul$ (myValue As _Unsigned Long)
cstrul$ = _Trim$(Str$(myValue))
End Function ' cstrul$
' /////////////////////////////////////////////////////////////////////////////
' Scientific notation - QB64 Wiki
' https://www.qb64.org/wiki/Scientific_notation
' Example: A string function that displays extremely small or large exponential decimal values.
Function DblToStr$ (n#)
value$ = UCase$(LTrim$(Str$(n#)))
Xpos% = InStr(value$, "D") + InStr(value$, "E") 'only D or E can be present
If Xpos% Then
expo% = Val(Mid$(value$, Xpos% + 1))
If Val(value$) < 0 Then
sign$ = "-": valu$ = Mid$(value$, 2, Xpos% - 2)
Else valu$ = Mid$(value$, 1, Xpos% - 1)
End If
dot% = InStr(valu$, "."): L% = Len(valu$)
If expo% > 0 Then add$ = String$(expo% - (L% - dot%), "0")
If expo% < 0 Then min$ = String$(Abs(expo%) - (dot% - 1), "0"): DP$ = "."
For n = 1 To L%
If Mid$(valu$, n, 1) <> "." Then num$ = num$ + Mid$(valu$, n, 1)
Next
Else DblToStr$ = value$: Exit Function
End If
DblToStr$ = _Trim$(sign$ + DP$ + min$ + num$ + add$)
End Function ' DblToStr$
Function FormatNumber$ (myValue, iDigits As Integer)
Dim strValue As String
strValue = DblToStr$(myValue) + String$(iDigits, " ")
If myValue < 1 Then
If myValue < 0 Then
strValue = Replace$(strValue, "-.", "-0.")
ElseIf myValue > 0 Then
strValue = "0" + strValue
End If
End If
FormatNumber$ = Left$(strValue, iDigits)
End Function ' FormatNumber$
' /////////////////////////////////////////////////////////////////////////////
' From: Bitwise Manipulations By Steven Roman
' http://www.romanpress.com/Articles/Bitwise_R/Bitwise.htm
' Returns the 8-bit binary representation
' of an integer iInput where 0 <= iInput <= 255
Function GetBinary$ (iInput1 As Integer)
Dim sResult As String
Dim iLoop As Integer
Dim iInput As Integer: iInput = iInput1
sResult = ""
If iInput >= 0 And iInput <= 255 Then
For iLoop = 1 To 8
sResult = LTrim$(RTrim$(Str$(iInput Mod 2))) + sResult
iInput = iInput \ 2
'If iLoop = 4 Then sResult = " " + sResult
Next iLoop
End If
GetBinary$ = sResult
End Function ' GetBinary$
' /////////////////////////////////////////////////////////////////////////////
' wonderfully inefficient way to read if a bit is set
' ival = GetBit256%(int we are comparing, int containing the bits we want to read)
' See also: GetBit256%, SetBit256%
Function GetBit256% (iNum1 As Integer, iBit1 As Integer)
Dim iResult As Integer
Dim sNum As String
Dim sBit As String
Dim iLoop As Integer
Dim bContinue As Integer
'DIM iTemp AS INTEGER
Dim iNum As Integer: iNum = iNum1
Dim iBit As Integer: iBit = iBit1
iResult = FALSE
bContinue = TRUE
If iNum < 256 And iBit <= 128 Then
sNum = GetBinary$(iNum)
sBit = GetBinary$(iBit)
For iLoop = 1 To 8
If Mid$(sBit, iLoop, 1) = "1" Then
'if any of the bits in iBit are false, return false
If Mid$(sNum, iLoop, 1) = "0" Then
iResult = FALSE
bContinue = FALSE
Exit For
End If
End If
Next iLoop
If bContinue = TRUE Then
iResult = TRUE
End If
End If
GetBit256% = iResult
End Function ' GetBit256%
' /////////////////////////////////////////////////////////////////////////////
' From: Bitwise Manipulations By Steven Roman
' http://www.romanpress.com/Articles/Bitwise_R/Bitwise.htm
' Returns the integer that corresponds to a binary string of length 8
Function GetIntegerFromBinary% (sBinary1 As String)
Dim iResult As Integer
Dim iLoop As Integer
Dim strBinary As String
Dim sBinary As String: sBinary = sBinary1
iResult = 0
strBinary = Replace$(sBinary, " ", "") ' Remove any spaces
For iLoop = 0 To Len(strBinary) - 1
iResult = iResult + 2 ^ iLoop * Val(Mid$(strBinary, Len(strBinary) - iLoop, 1))
Next iLoop
GetIntegerFromBinary% = iResult
End Function ' GetIntegerFromBinary%
Function IsOdd% (n)
If n Mod 2 = 1 Then
IsOdd% = TRUE
Else
IsOdd% = FALSE
End If
End Function ' IsOdd%
' /////////////////////////////////////////////////////////////////////////////
' By sMcNeill from https://www.qb64.org/forum/index.php?topic=896.0
Function IsNum% (text$)
Dim a$
Dim b$
a$ = _Trim$(text$)
b$ = _Trim$(Str$(Val(text$)))
If a$ = b$ Then
IsNum% = TRUE
Else
IsNum% = FALSE
End If
End Function ' IsNum%
' /////////////////////////////////////////////////////////////////////////////
' Re: Does a Is Number function exist in QB64?
' https://www.qb64.org/forum/index.php?topic=896.15
' MWheatley
' « Reply #18 on: January 01, 2019, 11:24:30 AM »
' returns 1 if string is an integer, 0 if not
Function IsNumber (text$)
Dim i As Integer
IsNumber = 1
For i = 1 To Len(text$)
If Asc(Mid$(text$, i, 1)) < 45 Or Asc(Mid$(text$, i, 1)) >= 58 Then
IsNumber = 0
Exit For
ElseIf Asc(Mid$(text$, i, 1)) = 47 Then
IsNumber = 0
Exit For
End If
Next i
End Function ' IsNumber
' /////////////////////////////////////////////////////////////////////////////
' Split and join strings
' https://www.qb64.org/forum/index.php?topic=1073.0
'Combine all elements of in$() into a single string with delimiter$ separating the elements.
Function join$ (in$(), delimiter$)
Dim result$
Dim i As Long
result$ = in$(LBound(in$))
For i = LBound(in$) + 1 To UBound(in$)
result$ = result$ + delimiter$ + in$(i)
Next i
join$ = result$
End Function ' join$
' /////////////////////////////////////////////////////////////////////////////
' ABS was returning strange values with type LONG
' so I created this which does not.
Function LongABS& (lngValue As Long)
If Sgn(lngValue) = -1 Then
LongABS& = 0 - lngValue
Else
LongABS& = lngValue
End If
End Function ' LongABS&
' /////////////////////////////////////////////////////////////////////////////
' Writes sText to a debug file in the EXE folder.
' Debug file is named the same thing as the program EXE name with ".txt" at the end.
' For example the program "C:\QB64\MyProgram.BAS" running as
' "C:\QB64\MyProgram.EXE" would have an output file "C:\QB64\MyProgram.EXE.txt".
' If the file doesn't exist, it is created, otherwise it is appended to.
Sub DebugPrintFile (sText As String)
Dim sFileName As String
Dim sError As String
Dim sOut As String
sFileName = m_ProgramPath$ + m_ProgramName$ + ".txt"
sError = ""
If _FileExists(sFileName) = FALSE Then
sOut = ""
sOut = sOut + "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++" + Chr$(13) + Chr$(10)
sOut = sOut + "PROGRAM : " + m_ProgramName$ + Chr$(13) + Chr$(10)
sOut = sOut + "RUN DATE: " + CurrentDateTime$ + Chr$(13) + Chr$(10)
sOut = sOut + "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++" + Chr$(13) + Chr$(10)
sError = PrintFile$(sFileName, sOut, FALSE)
End If
If Len(sError) = 0 Then
sError = PrintFile$(sFileName, sText, TRUE)
End If
If Len(sError) <> 0 Then
Print CurrentDateTime$ + " DebugPrintFile FAILED: " + sError
End If
End Sub ' DebugPrintFile
Function IntPadRight$ (iValue As Integer, iWidth As Integer)
IntPadRight$ = Left$(_Trim$(Str$(iValue)) + String$(iWidth, " "), iWidth)
End Function ' IntPadRight$
Function PrintFile$ (sFileName As String, sText As String, bAppend As Integer)
'x = 1: y = 2: z$ = "Three"
Dim sError As String: sError = ""
If Len(sError) = 0 Then
If (bAppend = TRUE) Then
If _FileExists(sFileName) Then
Open sFileName For Append As #1 ' opens an existing file for appending
Else
sError = "Error in PrintFile$ : File not found. Cannot append."
End If
Else
Open sFileName For Output As #1 ' opens and clears an existing file or creates new empty file
End If
End If
If Len(sError) = 0 Then
' WRITE places text in quotes in the file
'WRITE #1, x, y, z$
'WRITE #1, sText
' PRINT does not put text inside quotes
Print #1, sText
Close #1
'PRINT "File created with data. Press a key!"
'K$ = INPUT$(1) 'press a key
'OPEN sFileName FOR INPUT AS #2 ' opens a file to read it
'INPUT #2, a, b, c$
'CLOSE #2
'PRINT a, b, c$
'WRITE a, b, c$
End If
PrintFile$ = sError
End Function ' PrintFile$
' /////////////////////////////////////////////////////////////////////////////
' Does a _PrintString at the specified row+column.
' iRow and iCol are 0-based.
Sub PrintString (iRow As Integer, iCol As Integer, MyString As String)
Dim iX As Integer
Dim iY As Integer
iX = _FontWidth * iCol
iY = _FontHeight * iRow ' (iRow + 1)
_PrintString (iX, iY), MyString
End Sub ' PrintString
' /////////////////////////////////////////////////////////////////////////////
' Does a _PrintString at the specified row+column.
' iRow and iCol are 1-based.
Sub PrintString1 (iRow As Integer, iCol As Integer, MyString As String)
Dim iX As Integer
Dim iY As Integer
iX = _FontWidth * (iCol - 1)
iY = _FontHeight * (iRow - 1)
_PrintString (iX, iY), MyString
End Sub ' PrintString1
' /////////////////////////////////////////////////////////////////////////////
' Generate random value between Min and Max.
Function RandomNumber% (Min%, Max%)
Dim NumSpread%
' SET RANDOM SEED
'Randomize ' Initialize random-number generator.
Randomize Timer
' GET RANDOM # Min%-Max%
'RandomNumber = Int((Max * Rnd) + Min) ' generate number
NumSpread% = (Max% - Min%) + 1
RandomNumber% = Int(Rnd * NumSpread%) + Min% ' GET RANDOM # BETWEEN Max% AND Min%
Sub RandomNumberTest
Dim iCols As Integer: iCols = 10
Dim iRows As Integer: iRows = 20
Dim iLoop As Integer
Dim iX As Integer
Dim iY As Integer
Dim sError As String
Dim sFileName As String
Dim sText As String
Dim bAppend As Integer
Dim iMin As Integer
Dim iMax As Integer
Dim iNum As Integer
Dim iErrorCount As Integer
Dim sInput$
Input "Press <ENTER> to continue", sInput$
End Sub ' RandomNumberTest
' /////////////////////////////////////////////////////////////////////////////
' FROM: String Manipulation
' found at abandoned, outdated and now likely malicious qb64 dot net website
' http://www.qb64.[net]/forum/index_topic_5964-0/
'
'SUMMARY:
' Purpose: A library of custom functions that transform strings.
' Author: Dustinian Camburides (dustinian@gmail.com)
' Platform: QB64 (www.qb64.org)
' Revision: 1.6
' Updated: 5/28/2012
'SUMMARY:
'[Replace$] replaces all instances of the [Find] sub-string with the [Add] sub-string within the [Text] string.
'INPUT:
'Text: The input string; the text that's being manipulated.
'Find: The specified sub-string; the string sought within the [Text] string.
'Add: The sub-string that's being added to the [Text] string.
Function Replace$ (Text1 As String, Find1 As String, Add1 As String)
' VARIABLES:
Dim Text2 As String
Dim Find2 As String
Dim Add2 As String
Dim lngLocation As Long ' The address of the [Find] substring within the [Text] string.
Dim strBefore As String ' The characters before the string to be replaced.
Dim strAfter As String ' The characters after the string to be replaced.
' INITIALIZE:
' MAKE COPIESSO THE ORIGINAL IS NOT MODIFIED (LIKE ByVal IN VBA)
Text2 = Text1
Find2 = Find1
Add2 = Add1
lngLocation = InStr(1, Text2, Find2)
' PROCESSING:
' While [Find2] appears in [Text2]...
While lngLocation
' Extract all Text2 before the [Find2] substring:
strBefore = Left$(Text2, lngLocation - 1)
' Extract all text after the [Find2] substring:
strAfter = Right$(Text2, ((Len(Text2) - (lngLocation + Len(Find2) - 1))))
Print "ReplaceTest finished."
End Sub ' ReplaceTest
' /////////////////////////////////////////////////////////////////////////////
' https://www.qb64.org/forum/index.php?topic=3605.0
' Quote from: SMcNeill on Today at 03:53:48 PM
'
' Sometimes, you guys make things entirely too complicated.
' There ya go! Three functions to either round naturally,
' always round down, or always round up, to whatever number of digits you desire.
' EDIT: Modified to add another option to round scientific,
' since you had it's description included in your example.
Function Round## (num##, digits%)
Round## = Int(num## * 10 ^ digits% + .5) / 10 ^ digits%
End Function
Function RoundUp## (num##, digits%)
RoundUp## = _Ceil(num## * 10 ^ digits%) / 10 ^ digits%
End Function
Function RoundDown## (num##, digits%)
RoundDown## = Int(num## * 10 ^ digits%) / 10 ^ digits%
End Function
Function Round_Scientific## (num##, digits%)
Round_Scientific## = _Round(num## * 10 ^ digits%) / 10 ^ digits%
End Function
Function RoundUpDouble# (num#, digits%)
RoundUpDouble# = _Ceil(num# * 10 ^ digits%) / 10 ^ digits%
End Function
Function RoundUpSingle! (num!, digits%)
RoundUpSingle! = _Ceil(num! * 10 ^ digits%) / 10 ^ digits%
End Function
' /////////////////////////////////////////////////////////////////////////////
' fantastically inefficient way to set a bit
' example use: arrMaze(iX, iY) = SetBit256%(arrMaze(iX, iY), cS, FALSE)
' See also: GetBit256%, SetBit256%
' newint=SetBit256%(oldint, int containing the bits we want to set, value to set them to)
Function SetBit256% (iNum1 As Integer, iBit1 As Integer, bVal1 As Integer)
Dim sNum As String
Dim sBit As String
Dim sVal As String
Dim iLoop As Integer
Dim strResult As String
Dim iResult As Integer
Dim iNum As Integer: iNum = iNum1
Dim iBit As Integer: iBit = iBit1
Dim bVal As Integer: bVal = bVal1
If iNum < 256 And iBit <= 128 Then
sNum = GetBinary$(iNum)
sBit = GetBinary$(iBit)
If bVal = TRUE Then
sVal = "1"
Else
sVal = "0"
End If
strResult = ""
For iLoop = 1 To 8
If Mid$(sBit, iLoop, 1) = "1" Then
strResult = strResult + sVal
Else
strResult = strResult + Mid$(sNum, iLoop, 1)
End If
Next iLoop
iResult = GetIntegerFromBinary%(strResult)
Else
iResult = iNum
End If
SetBit256% = iResult
End Function ' SetBit256%
' /////////////////////////////////////////////////////////////////////////////
' Scientific notation - QB64 Wiki
' https://www.qb64.org/wiki/Scientific_notation
' Example: A string function that displays extremely small or large exponential decimal values.
Function SngToStr$ (n!)
value$ = UCase$(LTrim$(Str$(n!)))
Xpos% = InStr(value$, "D") + InStr(value$, "E") 'only D or E can be present
If Xpos% Then
expo% = Val(Mid$(value$, Xpos% + 1))
If Val(value$) < 0 Then
sign$ = "-": valu$ = Mid$(value$, 2, Xpos% - 2)
Else valu$ = Mid$(value$, 1, Xpos% - 1)
End If
dot% = InStr(valu$, "."): L% = Len(valu$)
If expo% > 0 Then add$ = String$(expo% - (L% - dot%), "0")
If expo% < 0 Then min$ = String$(Abs(expo%) - (dot% - 1), "0"): DP$ = "."
For n = 1 To L%
If Mid$(valu$, n, 1) <> "." Then num$ = num$ + Mid$(valu$, n, 1)
Next
Else SngToStr$ = value$: Exit Function
End If
SngToStr$ = _Trim$(sign$ + DP$ + min$ + num$ + add$)
End Function ' SngToStr$
' /////////////////////////////////////////////////////////////////////////////
' Split and join strings
' https://www.qb64.org/forum/index.php?topic=1073.0
'
' FROM luke, QB64 Developer
' Date: February 15, 2019, 04:11:07 AM »
'
' Given a string of words separated by spaces (or any other character),
' splits it into an array of the words. I've no doubt many people have
' written a version of this over the years and no doubt there's a million
' ways to do it, but I thought I'd put mine here so we have at least one
' version. There's also a join function that does the opposite
' array -> single string.
'
' Code is hopefully reasonably self explanatory with comments and a little demo.
' Note, this is akin to Python/JavaScript split/join, PHP explode/implode.
'Split in$ into pieces, chopping at every occurrence of delimiter$. Multiple consecutive occurrences
'of delimiter$ are treated as a single instance. The chopped pieces are stored in result$().
'
'delimiter$ must be one character long.
'result$() must have been REDIMmed previously.
' Modified to handle multi-character delimiters
Sub split (in$, delimiter$, result$())
Dim start As Integer
Dim finish As Integer
Dim iDelimLen As Integer
ReDim result$(-1)
iDelimLen = Len(delimiter$)
start = 1
Do
'While Mid$(in$, start, 1) = delimiter$
While Mid$(in$, start, iDelimLen) = delimiter$
'start = start + 1
start = start + iDelimLen
If start > Len(in$) Then
Exit Sub
End If
Wend
finish = InStr(start, in$, delimiter$)
If finish = 0 Then
finish = Len(in$) + 1
End If
ReDim _Preserve result$(0 To UBound(result$) + 1)
result$(UBound(result$)) = Mid$(in$, start, finish - start)
start = finish + 1
Loop While start <= Len(in$)
End Sub ' split
Function StrJustifyRight$ (sValue As String, iWidth As Integer)
StrJustifyRight$ = Right$(String$(iWidth, " ") + sValue, iWidth)
End Function ' StrJustifyRight$
Function StrJustifyLeft$ (sValue As String, iWidth As Integer)
StrJustifyLeft$ = Left$(sValue + String$(iWidth, " "), iWidth)
End Function ' StrJustifyLeft$
Function StrJustifyCenter$ (sValue As String, iWidth As Integer)
Dim iLen0 As Integer
Dim iLen1 As Integer
Dim iLen2 As Integer
Dim iExtra As Integer
iLen0 = Len(sValue)
If iWidth = iLen0 Then
' no extra space: return unchanged
StrJustifyCenter$ = sValue
ElseIf iWidth > iLen0 Then
If IsOdd%(iWidth) Then
iWidth = iWidth - 1
End If
' center
iExtra = iWidth - iLen0
iLen1 = iExtra \ 2
iLen2 = iLen1 + (iExtra Mod 2)
StrJustifyCenter$ = String$(iLen1, " ") + sValue + String$(iLen2, " ")
Else
' string is too long: truncate
StrJustifyCenter$ = Left$(sValue, iWidth)
End If
End Function ' StrJustifyCenter$
Function TrueFalse$ (myValue)
If myValue = TRUE Then
TrueFalse$ = "TRUE"
Else
TrueFalse$ = "FALSE"
End If
End Function ' TrueFalse$
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END GENERAL PURPOSE ROUTINES
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Finally, if you want to map game controllers too, this lets you map gamepad + keyboard input for upto 8 players, and saves the settings out to a config file. There are a both "GUI" and "no GUI" examples, I would recommend looking at the "no GUI" (search in the code for Function MapInput1$) because the my code for that is kind of convoluted. The routines MapInput1$, TestMappings1$, LoadMappings1$, SaveMappings1$ should have what you want to map keyboard and game controllers. (Eventually I'll finish version 3 which simplifies & removes the GUI code.0
' Basic Input Mapper, Barebones Octo edition.
' Version 2.00 by madscijr
' CHANGE LOG:
' Date Who What
' 12/16/2020 madscijr detect keys 0.70
' 02/17/2021 madscijr basic game controller test
' 01/08/2022 madscijr input mapping v1.0
' keyboard + game controllers
' text menu driven (no GUI)
' 01/21/2022 madscijr input mapping v2.0 with simple GUI
' DESCRIPTION:
' A way to map input controls (gamepad + keyboard)
' load/save mapping to a file, and read the input,
' that you can use in your own games.
' END TEXT GUI CONSTANTS
' ################################################################################################################################################################
' #UDT #TYPES = USER DEFINED TYPES
' UDT TO HOLD THE INFO FOR A PLAYER
Type PlayerType
x As Integer ' player x position
y As Integer ' player y position
c As Integer ' character to display on screen
xOld As Integer
yOld As Integer
' control buffer
moveX As Integer
moveY As Integer
moveUp As Integer
moveDown As Integer
moveLeft As Integer
moveRight As Integer
button1 As Integer
button2 As Integer
button3 As Integer
button4 As Integer
' control previous move
'lastMoveX As Integer
'lastMoveY As Integer
lastMoveUp As Integer
lastMoveDown As Integer
lastMoveLeft As Integer
lastMoveRight As Integer
lastButton1 As Integer
lastButton2 As Integer
lastButton3 As Integer
lastButton4 As Integer
'repeat As Integer
End Type ' PlayerType
' UDT TO HOLD THE INFO FOR A GAME CONTROLLER
Type ControllerType
buttonCount As Integer
axisCount As Integer
End Type ' ControllerType
' UDT TO HOLD THE INFO FOR A GAME CONTROLLER
Type ControlInputType
device As Integer
typ As Integer ' cInputKey, cInputButton, cInputAxis
code As Integer
value As Integer
repeat As Integer
End Type ' ControlInputType
' UDT TO HOLD COLOR CODE INFO
Type ColorType
name As String
value As _Unsigned Long
End Type ' ColorType
' UDT TO HOLD TEXT GUI
Type ScreenAreaType
name As String
typ As Integer ' cTextGuiSection, cTextGuiButton
item As String
player As Integer
x1 As Integer
y1 As Integer
x2 As Integer
y2 As Integer
'index as integer
End Type ' ScreenAreaType
' DEFINES CLICKABLE BUTTON BOUNDARIES
Type TextButtonType
item As String
typ As Integer ' cInputUp, cInputDown, cInputLeft, cInputRight, cInputButton1, cInputButton2, cInputButton3, cInputButton4
x1 As Integer
y1 As Integer
x2 As Integer
y2 As Integer
'index as integer
End Type ' TextButtonType
' DEFINES TEXT LABELS
Type TextLabelType
item As String ' "Section", "Up", "Down"
name As String ' "caption", "type", "device", "code", "repeat", "value"
row As Integer ' row (relative to section y1)
column As Integer ' column (relative to section x1)
width As Integer ' needed for cJustifyRight, cJustifyCenter
justify As Integer ' cJustifyLeft, cJustifyRight, cJustifyCenter, cJustifyUnknown
caption As String ' holds the label text
fgcolor As _Unsigned Long
bgcolor As _Unsigned Long
'index as integer
End Type ' TextLabelType
' DEFINES TEXT FIELDS
Type TextFieldType
item As String ' "Section", "Up", "Down"
name As String ' "caption", "type", "device", "code", "repeat", "value"
row As Integer ' row (relative to section y1)
column As Integer ' column (relative to section x1)
width As Integer ' pad values to this width
justify As Integer ' cJustifyLeft, cJustifyRight, cJustifyCenter, cJustifyUnknown
value As String ' holds the formatted value as text
fgcolor As _Unsigned Long
bgcolor As _Unsigned Long
'index as integer
End Type ' TextFieldType
' FOR TEXT SCREEN
Type TextCellType
value As String
fgColor As _Unsigned Long
bgcolor As _Unsigned Long
End Type ' TextCellType
' ################################################################################################################################################################
' #VARS = GLOBAL VARIABLES
' ENABLE / DISABLE DEBUG CONSOLE
Dim Shared m_bTesting As Integer: m_bTesting = TRUE
' BASIC PROGRAM METADATA
Dim Shared m_ProgramPath$: m_ProgramPath$ = Left$(Command$(0), _InStrRev(Command$(0), "\"))
Dim Shared m_ProgramName$: m_ProgramName$ = Mid$(Command$(0), _InStrRev(Command$(0), "\") + 1)
Dim Shared m_VersionInfo$: m_VersionInfo$ = "2.00"
' GAME CONTROLLER MAPPING
Dim Shared m_ControlMapFileName$: m_ControlMapFileName$ = Left$(m_ProgramName$, _InStrRev(m_ProgramName$, ".")) + "map.txt"
ReDim Shared m_arrControlMap(1 To 8, 1 To 8) As ControlInputType ' holds control mapping for each player (player #, direction)
ReDim Shared m_arrController(1 To 8) As ControllerType ' holds info for each game controller
ReDim Shared m_arrButtonCode(1 To 99) As Integer ' Long
ReDim Shared m_arrButtonKey(1 To 99) As String
ReDim Shared m_arrButtonKeyDesc(0 To 512) As String
ReDim Shared m_arrButtonKeyShortDesc(0 To 512) As String
Dim Shared m_bInitialized As Integer: m_bInitialized = FALSE
Dim Shared m_bHaveMapping As Integer: m_bHaveMapping = FALSE
' USE TO GLOBALLY ENABLE/DISABLE REPEATING INPUT PER FUNCTION
' To enable override set m_bRepeatOverride=TRUE,
' otherwise this can be configured for each individual controller
' when you map the functions.
Dim Shared m_bRepeatOverride As Integer: m_bRepeatOverride = TRUE
Dim Shared m_bRepeatUp As Integer: m_bRepeatUp = TRUE
Dim Shared m_bRepeatDown As Integer: m_bRepeatDown = TRUE
Dim Shared m_bRepeatLeft As Integer: m_bRepeatLeft = FALSE
Dim Shared m_bRepeatRight As Integer: m_bRepeatRight = FALSE
Dim Shared m_bRepeatButton1 As Integer: m_bRepeatButton1 = TRUE
Dim Shared m_bRepeatButton2 As Integer: m_bRepeatButton2 = TRUE
Dim Shared m_bRepeatButton3 As Integer: m_bRepeatButton3 = FALSE
Dim Shared m_bRepeatButton4 As Integer: m_bRepeatButton4 = FALSE
' VARIABLES FOR GRAPHIC PRINTING ROUTINES
Dim Shared m_NumColumns As Integer: m_NumColumns = 1
Dim Shared m_PrintRow As Integer: m_PrintRow = 0
Dim Shared m_PrintCol As Integer: m_PrintCol = 0
Dim Shared m_StartRow As Integer: m_StartRow = 0
Dim Shared m_EndRow As Integer: m_EndRow = 0
Dim Shared m_StartCol As Integer: m_StartCol = 0
Dim Shared m_EndCol As Integer: m_EndCol = 0
' VARIABLES FOR TEXT GUI
ReDim Shared m_arrScreenArea(-1) As ScreenAreaType
ReDim Shared m_arrButton(-1) As TextButtonType
ReDim Shared m_arrTextLabel(-1) As TextLabelType
ReDim Shared m_arrTextField(-1) As TextFieldType
' DEMO GAME / TESTING
ReDim Shared m_arrPlayer(1 To 8) As PlayerType ' holds info for each player
' =============================================================================
' LOCAL VARIABLES
Dim in$
' ****************************************************************************************************************************************************************
' ACTIVATE DEBUGGING WINDOW
If m_bTesting = TRUE Then
$Console
_Delay 4
_Console On
_Echo "Started " + m_ProgramName$
_Echo "Debugging on..."
End If
' ****************************************************************************************************************************************************************
' =============================================================================
' START THE MAIN ROUTINE
main
' ****************************************************************************************************************************************************************
' DEACTIVATE DEBUGGING WINDOW
If m_bTesting = TRUE Then
_Console Off
End If
' ****************************************************************************************************************************************************************
System ' return control to the operating system
End
If in$ = "1" Then
result$ = TestJoysticks1$
ElseIf in$ = "2" Then
result$ = LoadMappings1$
If Len(result$) = 0 Then result$ = "Loaded mappings."
ElseIf in$ = "3" Then
result$ = ViewMappings2$
ElseIf in$ = "4" Then
DumpControllerMap1
ElseIf in$ = "5" Then
result$ = EditMappings1$
ElseIf in$ = "6" Then
result$ = ResetMapping1$
ElseIf in$ = "7" Then
result$ = MapInput1$
ElseIf in$ = "8" Then
result$ = MapInput2$
ElseIf in$ = "9" Then
result$ = TestMappings1$
ElseIf in$ = "10" Then
result$ = SaveMappings1$
End If
If Len(result$) > 0 Then
Print result$
End If
Loop Until in$ = "q"
' RETURN TO TEXT SCREEN
Screen 0
End Sub ' main
' /////////////////////////////////////////////////////////////////////////////
' Just a little test to verify _DEFAULTCOLOR and _BACKGROUNDCOLOR work.
Function DetectColor1$
Dim sResult As String: sResult = ""
ReDim arrColor(-1) As ColorType
Dim ScreenArray(1 To 48, 1 To 128) As String
Dim iRow As Integer
Dim iCol As Integer
Dim iForeColor As _Unsigned Long
Dim iBackColor As _Unsigned Long
Dim iY As Integer
Dim iX As Integer
Dim in$
Dim iLoop1 As Integer
Dim iLoop2 As Integer
Dim sInfo As String
Dim sNext As String
Dim sData As String
Dim sTest As String
Dim iMaxLen As Integer
Dim sValue1 As String
Dim sValue2 As String
' INITIALIZE
'AddColors arrColor()
'StringToArray ScreenArray(), GetMap$
ReDim _Preserve arrColor(1 To UBound(arrColor) + 1) As ColorType
arrColor(UBound(arrColor)).name = "cRed"
arrColor(UBound(arrColor)).value = cRed
ReDim _Preserve arrColor(1 To UBound(arrColor) + 1) As ColorType
arrColor(UBound(arrColor)).name = "cWhite"
arrColor(UBound(arrColor)).value = cWhite
ReDim _Preserve arrColor(1 To UBound(arrColor) + 1) As ColorType
arrColor(UBound(arrColor)).name = "cBlue"
arrColor(UBound(arrColor)).value = cBlue
' GET LEN
iMaxLen = 0
For iLoop1 = LBound(arrColor) To UBound(arrColor)
If Len(arrColor(iLoop1).name) > iMaxLen Then
iMaxLen = Len(arrColor(iLoop1).name)
End If
Next iLoop1
' SET UP SCREEN
Screen _NewImage(1280, 1024, 32): _ScreenMove 0, 0
'Screen _NewImage(1024, 768, 32): _ScreenMove 0, 0
' DISPLAY GRAPHICALLY
Cls
'For iY = LBound(ScreenArray, 1) To UBound(ScreenArray, 1)
' For iX = LBound(ScreenArray, 2) To UBound(ScreenArray, 2)
' iRow = iY - 1: iCol = iX - 1
' Color cRed, cBlack
' PrintString iRow, iCol, ScreenArray(iY, iX)
' Next iX
'Next iY
iRow = 0
iCol = 0
For iLoop1 = LBound(arrColor) To UBound(arrColor)
For iLoop2 = LBound(arrColor) To UBound(arrColor)
If iLoop1 <> iLoop2 Then
iCol = 0
iForeColor = arrColor(iLoop1).value
iBackColor = arrColor(iLoop2).value
'sInfo = "Color " + _Trim$(Str$(iForeColor)) + ", " + _Trim$(Str$(iBackColor))
sInfo = "Color " + _
GetColorName$(arrColor(), iForeColor, _Trim$(Str$(iForeColor))) + _
", " + _
GetColorName$(arrColor(), iBackColor, _Trim$(Str$(iBackColor)))
Color cWhite, cBlack: PrintString iRow, iCol, sInfo
Function GetColorName$ (arrColor() As ColorType, ColorValue As _Unsigned Long, DefaultName As String)
Dim sResult As String
Dim iLoop As Long
sResult = DefaultName
For iLoop = LBound(arrColor) To UBound(arrColor)
If arrColor(iLoop).value = ColorValue Then
sResult = arrColor(iLoop).name
Exit For
End If
Next iLoop
GetColorName$ = sResult
End Function ' GetColorName$
' /////////////////////////////////////////////////////////////////////////////
' TODO: get keyboard input working
' TODO: get continuous movement working for digital joysticks
' TODO: adjust analog joystick sensitivity
Function TestMappings1$
Dim sResult As String: sResult = ""
Dim sError As String: sError = ""
Dim iDeviceCount As Integer
Dim iDevice As Integer
Dim iNumControllers As Integer
Dim iController As Integer
Dim iValue As Integer
Dim iWhichInput As Integer
Dim arrButton(32, 16) As Integer ' number of buttons on the joystick
Dim arrButtonNew(32, 16) As Integer ' tracks when to initialize values
Dim arrAxis(32, 16) As Double ' number of axis on the joystick
Dim arrAxisNew(32, 16) As Integer ' tracks when to initialize values
Dim iCols As Integer
Dim iRows As Integer
Dim iPlayer As Integer
Dim iNextY As Integer
Dim iNextX As Integer
Dim iNextC As Integer
Dim iMinX As Integer
Dim iMaxX As Integer
Dim iMinY As Integer
Dim iMaxY As Integer
Dim bHaveInput As Integer
Dim bFinished As Integer
Dim bFoundWho As Integer
Dim bRepeat As Integer
Dim in$
' MAKE SURE WE HAVE MAPPING
If m_bHaveMapping = TRUE Then
' INITIALIZE
InitKeyboardButtonCodes
iCols = _Width(0) \ _FontWidth
iRows = _Height(0) \ _FontHeight
iMinX = 1: iMaxX = iCols
iMinY = 1: iMaxY = iRows
Cls
PrintStringCR1 10, 20, "Test control mapping:"
PrintStringCR1 10, 22, "1. Directional controls move letters around screen."
PrintStringCR1 10, 23, "2. Buttons make sounds."
PrintStringCR1 10, 25, "Press <ESC> to exit."
' INITIALIZE PLAYER COORDINATES AND SCREEN CHARACTERS
iNextY = 1
iNextX = -3
iNextC = 64
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
iNextX = iNextX + 4
If iNextX >= iMaxX Then
iNextX = iMinX
iNextY = iNextY + 4
If iNextY > iMaxY Then
iNextY = iMinY
End If
End If
iNextC = iNextC + 1
m_arrPlayer(iPlayer).x = iNextX
m_arrPlayer(iPlayer).y = iNextY
m_arrPlayer(iPlayer).c = iNextC
m_arrPlayer(iPlayer).xOld = iNextX
m_arrPlayer(iPlayer).yOld = iNextY
' COUNT # OF JOYSTICKS
' TODO: find out the right way to count joysticks
If Len(sError) = 0 Then
' D= _DEVICES ' MUST be read in order for other 2 device functions to work!
iDeviceCount = _Devices ' Find the number of devices on someone's system
If iDeviceCount > 2 Then
' LIMIT # OF DEVICES, IF THERE IS A LIMIT DEFINED
iNumControllers = iDeviceCount - 2
If cMaxControllers > 0 Then
If iNumControllers > cMaxControllers Then
iNumControllers = cMaxControllers
End If
End If
Else
' ONLY 2 FOUND (KEYBOARD, MOUSE)
'sError = "No game controllers found."
iNumControllers = 0
End If
End If
' INITIALIZE CONTROLLER DATA
If Len(sError) = 0 Then
For iController = 1 To iNumControllers
m_arrController(iController).buttonCount = cMaxButtons
m_arrController(iController).axisCount = cMaxAxis
For iLoop = 1 To cMaxButtons
arrButtonNew(iController, iLoop) = TRUE
Next iLoop
For iLoop = 1 To cMaxAxis
arrAxisNew(iController, iLoop) = TRUE
Next iLoop
Next iController
End If
' INITIALIZE CONTROLLER INPUT
If Len(sError) = 0 Then
_KeyClear: _Delay 1
For iController = 1 To iNumControllers
iDevice = iController + 2
While _DeviceInput(iDevice) ' clear and update the device buffer
For iLoop = 1 To _LastButton(iDevice)
If (iLoop > cMaxButtons) Then Exit For
m_arrController(iController).buttonCount = iLoop
arrButton(iController, iLoop) = FALSE
Next iLoop
For iLoop = 1 To _LastAxis(iDevice) ' this loop checks all my axis
If (iLoop > cMaxAxis) Then Exit For
m_arrController(iController).axisCount = iLoop
arrAxis(iController, iLoop) = 0
Next iLoop
Wend ' clear and update the device buffer
Next iController
End If
' GET INPUT AND MOVE PLAYERS AROUND ON SCREEN
_KeyClear: _Delay 1
bFinished = FALSE
Do
' Clear control buffer for players
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
m_arrPlayer(iPlayer).moveUp = FALSE
m_arrPlayer(iPlayer).moveDown = FALSE
m_arrPlayer(iPlayer).moveLeft = FALSE
m_arrPlayer(iPlayer).moveRight = FALSE
m_arrPlayer(iPlayer).button1 = FALSE
m_arrPlayer(iPlayer).button2 = FALSE
m_arrPlayer(iPlayer).button3 = FALSE
m_arrPlayer(iPlayer).button4 = FALSE
Next iPlayer
' -----------------------------------------------------------------------------
' BEGIN CHECK FOR CONTROLLER INPUT
If iNumControllers > 0 Then
For iController = 1 To iNumControllers
iDevice = iController + 2
' Check all devices
While _DeviceInput(iDevice)
Wend ' clear and update the device buffer
' Check each button
For iLoop = 1 To _LastButton(iDevice)
If (iLoop > cMaxButtons) Then Exit For
' update button array to indicate if a button is up or down currently.
'if TRUE=TRUE then
If _ButtonChange(iLoop) Then
iValue = _Button(iLoop)
If iValue <> arrButton(iController, iLoop) Then
' *****************************************************************************
' PRESSED BUTTON
' BEGIN find who this is mapped for
bFoundWho = FALSE
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If m_arrControlMap(iPlayer, iWhichInput).device = iDevice Then
If m_arrControlMap(iPlayer, iWhichInput).typ = cInputButton Then
If m_arrControlMap(iPlayer, iWhichInput).code = iLoop Then
'if m_arrControlMap(iPlayer, iWhichInput).value = iValue then
bFoundWho = TRUE
Select Case iWhichInput
Case cInputUp:
m_arrPlayer(iPlayer).moveUp = TRUE
Case cInputDown:
m_arrPlayer(iPlayer).moveDown = TRUE
Case cInputLeft:
m_arrPlayer(iPlayer).moveLeft = TRUE
Case cInputRight:
m_arrPlayer(iPlayer).moveRight = TRUE
Case cInputButton1:
m_arrPlayer(iPlayer).button1 = TRUE
Case cInputButton2:
m_arrPlayer(iPlayer).button2 = TRUE
Case cInputButton3:
m_arrPlayer(iPlayer).button3 = TRUE
Case cInputButton4:
m_arrPlayer(iPlayer).button4 = TRUE
Case Else:
'(IGNORE)
End Select
Exit For
'end if
End If
End If
End If
Next iWhichInput
If bFoundWho = TRUE Then Exit For
Next iPlayer
' END find who this is mapped for
End If
End If
Next iLoop
' Check each axis
For iLoop = 1 To _LastAxis(iDevice)
If (iLoop > cMaxAxis) Then Exit For
dblNextAxis = _Axis(iLoop)
dblNextAxis = RoundUpDouble#(dblNextAxis, 3)
' I like to give a little "jiggle" resistance to my controls, as I have an old joystick
' which is prone to always give minute values and never really center on true 0.
' A value of 1 means my axis is pushed fully in one direction.
' A value greater than 0.1 means it's been partially pushed in a direction (such as at a 45 degree diagional angle).
' A value of less than 0.1 means we count it as being centered. (As if it was 0.)
' Set sensitivity:
'These are way too sensitive for analog:
'IF ABS(_AXIS(iLoop)) <= 1 AND ABS(_AXIS(iLoop)) >= .1 THEN
'IF ABS(dblNextAxis) <= 1 AND ABS(dblNextAxis) >= .01 THEN
'IF ABS(dblNextAxis) <= 1 AND ABS(dblNextAxis) >= .001 THEN
''For digital input, we'll use a big picture:
'IF ABS(dblNextAxis) <= 1 AND ABS(dblNextAxis) >= 0.75 THEN
If Abs(dblNextAxis) <= 1 And Abs(dblNextAxis) >= 0.5 Then
' WE WANT CONTINUOUS MOVEMENT (DISABLE FOR NOT)
'if TRUE=TRUE then
If dblNextAxis <> arrAxis(iController, iLoop) Then
' *****************************************************************************
' MOVED STICK
' convert to a digital value
If dblNextAxis < 0 Then
iValue = -1
Else
iValue = 1
End If
' BEGIN find who this is mapped for
bFoundWho = FALSE
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If m_arrControlMap(iPlayer, iWhichInput).device = iDevice Then
If m_arrControlMap(iPlayer, iWhichInput).typ = cInputAxis Then
If m_arrControlMap(iPlayer, iWhichInput).code = iLoop Then
If m_arrControlMap(iPlayer, iWhichInput).value = iValue Then
bFoundWho = TRUE
Select Case iWhichInput
Case cInputUp:
m_arrPlayer(iPlayer).moveUp = TRUE
Case cInputDown:
m_arrPlayer(iPlayer).moveDown = TRUE
Case cInputLeft:
m_arrPlayer(iPlayer).moveLeft = TRUE
Case cInputRight:
m_arrPlayer(iPlayer).moveRight = TRUE
Case cInputButton1:
m_arrPlayer(iPlayer).button1 = TRUE
Case cInputButton2:
m_arrPlayer(iPlayer).button2 = TRUE
Case cInputButton3:
m_arrPlayer(iPlayer).button3 = TRUE
Case cInputButton4:
m_arrPlayer(iPlayer).button4 = TRUE
Case Else:
'(IGNORE)
End Select
Exit For
End If
End If
End If
End If
Next iWhichInput
If bFoundWho = TRUE Then Exit For
Next iPlayer
' END find who this is mapped for
End If
End If
Next iLoop
Next iController
End If
' END CHECK FOR CONTROLLER INPUT
' -----------------------------------------------------------------------------
' -----------------------------------------------------------------------------
' BEGIN CHECK FOR KEYBOARD INPUT #1
'_KEYCLEAR: _DELAY 1
While _DeviceInput(1): Wend ' clear and update the keyboard buffer
' Detect changed key state
iDevice = 1 ' keyboard
For iLoop = LBound(m_arrButtonCode) To UBound(m_arrButtonCode)
iCode = m_arrButtonCode(iLoop)
If _Button(iCode) <> FALSE Then
' *****************************************************************************
' PRESSED KEYBOARD
'PRINT "PRESSED " + m_arrButtonKey(iLoop)
' BEGIN find who this is mapped for
bFoundWho = FALSE
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If m_arrControlMap(iPlayer, iWhichInput).device = iDevice Then
If m_arrControlMap(iPlayer, iWhichInput).typ = cInputKey Then
'if m_arrControlMap(iPlayer, iWhichInput).code = iLoop then
If m_arrControlMap(iPlayer, iWhichInput).code = iCode Then
'if m_arrControlMap(iPlayer, iWhichInput).value = iValue then
bFoundWho = TRUE
Select Case iWhichInput
Case cInputUp:
m_arrPlayer(iPlayer).moveUp = TRUE
Case cInputDown:
m_arrPlayer(iPlayer).moveDown = TRUE
Case cInputLeft:
m_arrPlayer(iPlayer).moveLeft = TRUE
Case cInputRight:
m_arrPlayer(iPlayer).moveRight = TRUE
Case cInputButton1:
m_arrPlayer(iPlayer).button1 = TRUE
Case cInputButton2:
m_arrPlayer(iPlayer).button2 = TRUE
Case cInputButton3:
m_arrPlayer(iPlayer).button3 = TRUE
Case cInputButton4:
m_arrPlayer(iPlayer).button4 = TRUE
Case Else:
'(IGNORE)
End Select
Exit For
'end if
End If
End If
End If
Next iWhichInput
If bFoundWho = TRUE Then Exit For
Next iPlayer
' END find who this is mapped for
End If
Next iLoop
' END CHECK FOR KEYBOARD INPUT #1
' -----------------------------------------------------------------------------
' NOW DRAW PLAYERS ON SCREEN
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
' -----------------------------------------------------------------------------
' BEGIN UPDATE MOVEMENT CONTROL STATES
' If repeating keys are disabled then
' disable until the key has been released
If m_arrControlMap(iPlayer, cInputUp).repeat = FALSE Then
If m_arrPlayer(iPlayer).moveUp = TRUE Then
If m_arrPlayer(iPlayer).lastMoveUp = TRUE Then
m_arrPlayer(iPlayer).moveUp = FALSE
End If
Else
m_arrPlayer(iPlayer).lastMoveUp = FALSE
End If
End If
If m_arrControlMap(iPlayer, cInputDown).repeat = FALSE Then
If m_arrPlayer(iPlayer).moveDown = TRUE Then
If m_arrPlayer(iPlayer).lastMoveDown = TRUE Then
m_arrPlayer(iPlayer).moveDown = FALSE
End If
Else
m_arrPlayer(iPlayer).lastMoveDown = FALSE
End If
End If
If m_arrControlMap(iPlayer, cInputLeft).repeat = FALSE Then
If m_arrPlayer(iPlayer).moveLeft = TRUE Then
If m_arrPlayer(iPlayer).lastMoveLeft = TRUE Then
m_arrPlayer(iPlayer).moveLeft = FALSE
End If
Else
m_arrPlayer(iPlayer).lastMoveLeft = FALSE
End If
End If
If m_arrControlMap(iPlayer, cInputRight).repeat = FALSE Then
If m_arrPlayer(iPlayer).moveRight = TRUE Then
If m_arrPlayer(iPlayer).lastMoveRight = TRUE Then
m_arrPlayer(iPlayer).moveRight = FALSE
End If
Else
m_arrPlayer(iPlayer).lastMoveRight = FALSE
End If
End If
' END UPDATE MOVEMENT CONTROL STATES
' -----------------------------------------------------------------------------
' -----------------------------------------------------------------------------
' BEGIN MOVEMENT ACTIONS
If m_arrPlayer(iPlayer).moveUp = TRUE Then
m_arrPlayer(iPlayer).moveY = -1
m_arrPlayer(iPlayer).lastMoveUp = TRUE
End If
If m_arrPlayer(iPlayer).moveDown = TRUE Then
m_arrPlayer(iPlayer).moveY = 1
m_arrPlayer(iPlayer).lastMoveDown = TRUE
End If
If m_arrPlayer(iPlayer).moveLeft = TRUE Then
m_arrPlayer(iPlayer).moveX = -1
m_arrPlayer(iPlayer).lastMoveLeft = TRUE
End If
If m_arrPlayer(iPlayer).moveRight = TRUE Then
m_arrPlayer(iPlayer).moveX = 1
m_arrPlayer(iPlayer).lastMoveRight = TRUE
End If
' END MOVEMENT ACTIONS
' -----------------------------------------------------------------------------
' -----------------------------------------------------------------------------
' BEGIN MOVEMENT
' MOVE RIGHT/LEFT
m_arrPlayer(iPlayer).x = m_arrPlayer(iPlayer).x + m_arrPlayer(iPlayer).moveX
If m_arrPlayer(iPlayer).x < iMinX Then
m_arrPlayer(iPlayer).x = m_arrPlayer(iPlayer).xOld ' iMinX
ElseIf m_arrPlayer(iPlayer).x > iMaxX Then
m_arrPlayer(iPlayer).x = m_arrPlayer(iPlayer).xOld ' iMaxX
End If
' MOVE UP/DOWN
m_arrPlayer(iPlayer).y = m_arrPlayer(iPlayer).y + m_arrPlayer(iPlayer).moveY
If m_arrPlayer(iPlayer).y < iMinY Then
m_arrPlayer(iPlayer).y = m_arrPlayer(iPlayer).yOld ' iMinY
ElseIf m_arrPlayer(iPlayer).y > iMaxY Then
m_arrPlayer(iPlayer).y = m_arrPlayer(iPlayer).yOld ' iMaxY
End If
' END MOVEMENT
' -----------------------------------------------------------------------------
' -----------------------------------------------------------------------------
' BEGIN UPDATE BUTTON STATES
' If repeating keys are disabled then
' disable until the key has been released
'if m_bRepeatButton1 = FALSE then
If m_arrControlMap(iPlayer, cInputButton1).repeat = FALSE Then
If m_arrPlayer(iPlayer).button1 = TRUE Then
If m_arrPlayer(iPlayer).lastButton1 = TRUE Then
m_arrPlayer(iPlayer).button1 = FALSE
End If
Else
m_arrPlayer(iPlayer).lastButton1 = FALSE
End If
End If
If m_arrControlMap(iPlayer, cInputButton2).repeat = FALSE Then
If m_arrPlayer(iPlayer).button2 = TRUE Then
If m_arrPlayer(iPlayer).lastButton2 = TRUE Then
m_arrPlayer(iPlayer).button2 = FALSE
End If
Else
m_arrPlayer(iPlayer).lastButton2 = FALSE
End If
End If
If m_arrControlMap(iPlayer, cInputButton3).repeat = FALSE Then
If m_arrPlayer(iPlayer).button3 = TRUE Then
If m_arrPlayer(iPlayer).lastButton3 = TRUE Then
m_arrPlayer(iPlayer).button3 = FALSE
End If
Else
m_arrPlayer(iPlayer).lastButton3 = FALSE
End If
End If
If m_arrControlMap(iPlayer, cInputButton4).repeat = FALSE Then
If m_arrPlayer(iPlayer).button4 = TRUE Then
If m_arrPlayer(iPlayer).lastButton4 = TRUE Then
m_arrPlayer(iPlayer).button4 = FALSE
End If
Else
m_arrPlayer(iPlayer).lastButton4 = FALSE
End If
End If
' END UPDATE BUTTON STATES
' -----------------------------------------------------------------------------
' -----------------------------------------------------------------------------
' BEGIN BUTTON ACTIONS
If m_arrPlayer(iPlayer).button1 = TRUE Then
MakeSound iPlayer, 1
m_arrPlayer(iPlayer).lastButton1 = TRUE
End If
If m_arrPlayer(iPlayer).button2 = TRUE Then
MakeSound iPlayer, 2
m_arrPlayer(iPlayer).lastButton2 = TRUE
End If
If m_arrPlayer(iPlayer).button3 = TRUE Then
MakeSound iPlayer, 3
m_arrPlayer(iPlayer).lastButton3 = TRUE
End If
If m_arrPlayer(iPlayer).button4 = TRUE Then
MakeSound iPlayer, 4
m_arrPlayer(iPlayer).lastButton4 = TRUE
End If
' END BUTTON ACTIONS
' -----------------------------------------------------------------------------
Next iPlayer
_Limit 30
Loop Until _KeyHit = 27 ' ESCAPE to quit
_KeyClear: _Delay 1
sResult = sError
Else
sResult = "No mapping loaded. Please load a mapping or map keys."
End If
TestMappings1$ = sResult
End Function ' TestMappings1$
Sub MakeSound (iPlayer As Integer, iButton As Integer)
Dim note%
If iPlayer < 1 Then
iPlayer = 1
ElseIf iPlayer > 8 Then
iPlayer = 8
End If
If iButton < 1 Then
iButton = 1
ElseIf iButton > 4 Then
iButton = 4
End If
note% = iPlayer * 100 + (iButton * 25)
If note% > 4186 Then
note% = 4186
End If
Sound note%, .75
End Sub ' MakeSound
' /////////////////////////////////////////////////////////////////////////////
' V2 prints in 2 columns.
' A total kludge!
Sub PrintControllerMap2
Dim RoutineName As String:: RoutineName = "PrintControllerMap2"
Dim iPlayer As Integer
Dim iWhichInput As Integer
Dim iCount As Integer
Dim sLine As String
Dim iHalf As Integer
Dim sColumn1 As String: sColumn1 = ""
Dim sColumn2 As String: sColumn2 = ""
ReDim arrColumn1(-1) As String
ReDim arrColumn2(-1) As String
Dim iLoop As Integer
Dim iColWidth As Integer: iColWidth = 60
Dim sValue As String
Dim in$
' INITIALIZE
InitKeyboardButtonCodes
' START OUTPUT
Print "Controller mapping:"
'Print "Player# Input Device# Type Code Value"
' 1 button #2 x unknown x x
' 9 11 9 9 18 9
' 12345678912345678901123456789123456789123456789012345678123456789
' 12345678901234567890123456789012345678901234567890123456789012345678901234567890
' 00000000011111111112222222222333333333344444444445555555555666666666677777777778
If m_bHaveMapping = TRUE Then
' THIS IS A LAZY WAY TO GET 2 COLUMNS!
iHalf = UBound(m_arrControlMap, 1) / 2
sLine = "Player Input Device# Type Code Value Rep"
sColumn1 = sColumn1 + StrPadRight$(sLine, iColWidth) + Chr$(13)
sLine = "----------------------------------------------------------"
sColumn1 = sColumn1 + StrPadRight$(sLine, iColWidth) + Chr$(13)
For iPlayer = LBound(m_arrControlMap, 1) To iHalf
iCount = 0
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ) <> "unknown" Then
iCount = iCount + 1
End If
Next iWhichInput
If iCount > 0 Then
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ) <> "unknown" Then
sLine = IntPadRight$(iPlayer, 8)
sLine = sLine + StrPadRight$(InputToString$(iWhichInput), 10)
sLine = sLine + IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).device, 9)
sLine = sLine + StrPadRight$(InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ), 9)
'sLine = sLine + IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).code, 9)
If m_arrControlMap(iPlayer, iWhichInput).typ = cInputKey Then
sValue = GetKeyboardButtonCodeShortText$(m_arrControlMap(iPlayer, iWhichInput).code)
sValue = StrPadRight$(sValue, 13)
Else
sValue = IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).code, 13)
End If
sLine = sLine + sValue
'Print sLine
sLine = StrPadRight$(sLine, iColWidth)
sColumn2 = sColumn2 + sLine + Chr$(13)
End If
Next iWhichInput
Else
sLine = IntPadRight$(iPlayer, 9) + "(NONE)"
'Print sLine
sLine = StrPadRight$(sLine, iColWidth)
sColumn2 = sColumn2 + sLine + Chr$(13)
End If
Next iPlayer
split sColumn1, Chr$(13), arrColumn1()
split sColumn2, Chr$(13), arrColumn2()
If UBound(arrColumn1) > UBound(arrColumn2) Then
iCount = UBound(arrColumn1)
Else
iCount = UBound(arrColumn2)
End If
For iLoop = 0 To iCount
sLine = ""
If UBound(arrColumn1) >= iLoop Then
sLine = sLine + arrColumn1(iLoop)
Else
sLine = sLine + String$(iColWidth, " ")
End If
sLine = sLine + " "
If UBound(arrColumn2) >= iLoop Then
sLine = sLine + arrColumn2(iLoop)
Else
sLine = sLine + String$(iColWidth, " ")
End If
Print sLine
Next iLoop
Else
Print "No mapping loaded. Please load a mapping or map keys."
End If
End Sub ' PrintControllerMap2
' /////////////////////////////////////////////////////////////////////////////
' Original (simple) routine
Sub PrintControllerMap1
Dim RoutineName As String:: RoutineName = "PrintControllerMap1"
Dim iPlayer As Integer
Dim iWhichInput As Integer
Dim sLine As String
Dim iCount As Integer
Dim in$
' INITIALIZE
InitKeyboardButtonCodes
' OUTPUT MAPPING
Print "Controller mapping:"
Print "Player# Input Device# Type Code Value"
' 1 button #2 x unknown x x
' 9 11 9 9 9 9
' 12345678912345678901123456789123456789123456789123456789
' 12345678901234567890123456789012345678901234567890123456789012345678901234567890
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
iCount = 0
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ) <> "unknown" Then
iCount = iCount + 1
End If
Next iWhichInput
If iCount > 0 Then
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ) <> "unknown" Then
sLine = IntPadRight$(iPlayer, 9)
sLine = sLine + StrPadRight$(InputToString$(iWhichInput), 11)
sLine = sLine + IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).device, 9)
sLine = sLine + StrPadRight$(InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ), 9)
sLine = sLine + IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).code, 9)
sLine = sLine + IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).value, 9)
Print sLine
End If
Next iWhichInput
Else
sLine = IntPadRight$(iPlayer, 9) + "(NONE)"
Print sLine
End If
Next iPlayer
End Sub ' PrintControllerMap1
' /////////////////////////////////////////////////////////////////////////////
' Simple routine
' enables debugging, prints to debug window
' when done disables debugging (if it was disabled to begin with)
Sub DumpControllerMap1
Dim RoutineName As String:: RoutineName = "DumpControllerMap1"
Dim iPlayer As Integer
Dim iWhichInput As Integer
Dim sLine As String
Dim iCount As Integer
Dim in$
Dim bTesting As Integer
' ENABLE DEEBUGGING (IF NOT ENABLED)
bTesting = m_bTesting
' ACTIVATE DEBUGGING WINDOW (IF NOT ACTIVATED)
If m_bTesting = FALSE Then
m_bTesting = TRUE
$Console
_Delay 4
_Console On
_Echo "Started " + m_ProgramName$
_Echo "Debugging on..."
End If
' INITIALIZE
InitKeyboardButtonCodes
' OUTPUT MAPPING
DebugPrint "Controller mapping:"
DebugPrint "Player# Input Device# Type Code Value"
' 1 button #2 x unknown x x
' 9 11 9 9 9 9
' 12345678912345678901123456789123456789123456789123456789
' 12345678901234567890123456789012345678901234567890123456789012345678901234567890
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
iCount = 0
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ) <> "unknown" Then
iCount = iCount + 1
End If
Next iWhichInput
If iCount > 0 Then
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ) <> "unknown" Then
sLine = IntPadRight$(iPlayer, 9)
sLine = sLine + StrPadRight$(InputToString$(iWhichInput), 11)
sLine = sLine + IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).device, 9)
sLine = sLine + StrPadRight$(InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ), 9)
sLine = sLine + IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).code, 9)
sLine = sLine + IntPadRight$(m_arrControlMap(iPlayer, iWhichInput).value, 9)
DebugPrint sLine
End If
Next iWhichInput
Else
sLine = IntPadRight$(iPlayer, 9) + "(NONE)"
DebugPrint sLine
End If
Next iPlayer
' WAIT FOR USER
Cls
Print "Controller mapping written to console window."
Input "PRESS <ENTER> TO CONTINUE"; in$
' DEACTIVATE DEBUGGING WINDOW (IF IT WAS NOT ACTIVATED BEFORE)
If bTesting = FALSE Then
m_bTesting = FALSE
Function ViewMappings2$
' INITIALIZE
InitKeyboardButtonCodes
PrintControllerMap2
Print
Input "PRESS <ENTER> TO CONTINUE", in$
Print
ViewMappings2$ = ""
End Function ' ViewMappings2$
' /////////////////////////////////////////////////////////////////////////////
' TODO: test this
Function EditMappings1$
Dim RoutineName As String: RoutineName = "EditMappings1$"
Dim in$
Dim iPlayer As Integer
Dim iWhichInput As Integer
Dim iDevice As Integer
Dim iType As Integer
Dim iCode As Integer
Dim iValue As Integer
Dim iRepeat As Integer
Dim iItem As Integer
Dim sResult As String: sResult = ""
Dim bContinue1 As Integer: bContinue1 = TRUE
Dim bContinue2 As Integer: bContinue2 = TRUE
Dim bContinue3 As Integer: bContinue3 = TRUE
Dim bContinue4 As Integer: bContinue4 = TRUE
' INITIALIZE
InitKeyboardButtonCodes
' EDIT
Do
PrintControllerMap2
Print "To edit a mapping, enter a player number: " _
"1-" + cstr$(cMaxPlayers) + ", " + _
cstr$(cMaxPlayers+1) + ") or q to exit."
Input "Edit mapping for player"; in$
If IsNum%(in$) Then
iPlayer = Val(in$)
If iPlayer > 0 And iPlayer <= cMaxPlayers Then
bContinue2 = TRUE
Do
Print "Editing mappings for player " + cstr$(iPlayer) + "."
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
'Print right$(" " + cstr$(iWhichInput), 2) + ". " + InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ)
Print Right$(" " + cstr$(iWhichInput), 2) + ". " + InputToString$(iWhichInput)
Next iWhichInput
Input "Type # of control to edit or q to quit editing player"; in$
If IsNum%(in$) Then
iWhichInput = Val(in$)
If iWhichInput >= LBound(m_arrControlMap, 2) And m_arrControlMap <= UBound(m_arrControlMap, 2) Then
bContinue3 = TRUE
Do
Print "Settings for " + InputToString$(iWhichInput) + ":"
Print "1. Device # : " + cstr$(m_arrControlMap(iPlayer, iWhichInput).device)
Print "2. Device type : " + InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ)
If m_arrControlMap(iPlayer, iWhichInput).typ = cInputKey Then
Print "3. Input code : " + GetKeyboardButtonCodeText$(m_arrControlMap(iPlayer, iWhichInput).code) + _
" (" + _Trim$(Str$(m_arrControlMap(iPlayer, iWhichInput).code)) + ")"
Else
Print "3. Input code : " + _Trim$(Str$(m_arrControlMap(iPlayer, iWhichInput).code))
End If
Print "4. Input value : " + _Trim$(Str$(m_arrControlMap(iPlayer, iWhichInput).value))
Print "5. Enable repeat: " + TrueFalse$(m_arrControlMap(iPlayer, iWhichInput).repeat)
Input "Change item? (1-5 or q to quit editing control)"; in$
If IsNum%(in$) Then
iItem = Val(in$)
Select Case iItem
Case 1:
Print "Change the device number."
Input "Type a new device #, 0 for none (disabled), or blank to leave it unchanged"; in$
If IsNum%(in$) Then
iDevice = Val(in$)
m_arrControlMap(iPlayer, iWhichInput).device = iDevice
Print "Updated device number. Remember to save mappings when done."
Else
Print "(No change.)"
End If
Case 2:
bContinue4 = TRUE
Do
Print "Change the device type."
Print cstr$(cInputKey) + "=keyboard"
Print cstr$(cInputButton) + "=game controller button"
Print cstr$(cInputAxis) + "=game controller joystick/axis"
Print cstr$(cInputNone) + "=none"
Input "Device type or blank to leave it unchanged"; in$
If IsNum%(in$) Then
iType = Val(in$)
if iType=cInputKey or iType=cInputButton or _
iType=cInputAxis or iType=cInputNone then
m_arrControlMap(iPlayer, iWhichInput).typ = iType
Print "Updated device type. Remember to save mappings when done."
bContinue4 = FALSE: Exit Do
Else
Print "Please choose one of the listed values."
End If
Else
Print "(No change.)"
bContinue4 = FALSE: Exit Do
End If
Loop Until bContinue4 = FALSE
Case 3:
Print "Change the input code."
Input "Type a new input code, or blank to leave it unchanged"; in$
If IsNum%(in$) Then
iCode = Val(in$)
m_arrControlMap(iPlayer, iWhichInput).code = iCode
Print "Updated input code. Remember to save mappings when done."
Else
Print "(No change.)"
End If
Case 4:
Print "Change the input value."
Input "Type a new input value, or blank to leave it unchanged"; in$
If IsNum%(in$) Then
iValue = Val(in$)
m_arrControlMap(iPlayer, iWhichInput).value = iValue
Print "Updated input value. Remember to save mappings when done."
Else
Print "(No change.)"
End If
Case 5:
Print "Change the repeat setting."
Input "Type 1 to enable or 0 to disable, or blank to leave it unchanged"; in$
If IsNum%(in$) Then
iRepeat = Val(in$)
If iRepeat = 0 Then
m_arrControlMap(iPlayer, iWhichInput).repeat = FALSE
Print "Repeat disabled. Remember to save mappings when done."
ElseIf iRepeat = 1 Then
m_arrControlMap(iPlayer, iWhichInput).repeat = TRUE
Print "Repeat enabled. Remember to save mappings when done."
Else
Print "(No change.)"
End If
Else
Print "(No change.)"
End If
Case Else:
Print "Please choose a number between 1 and 4."
End Select
Else
bContinue3 = FALSE: Exit Do
End If
Loop Until bContinue3 = FALSE
Else
Print "Please choose a number between " + cstr$(LBound(m_arrControlMap, 2)) + " and " + cstr$(UBound(m_arrControlMap, 2)) + "."
End If
Else
bContinue2 = FALSE: Exit Do
End If
Loop Until bContinue2 = FALSE
If bContinue1 = FALSE Then Exit Do
Else
Print "Please choose a number between 1 and " + cstr$(cMaxPlayers) + "."
End If
Else
If Len(sResult) = 0 Then sResult = "(Cancelled.)"
bContinue1 = FALSE: Exit Do
End If
Loop Until bContinue1 = FALSE
_KeyClear: _Delay 1
EditMappings1$ = sResult
End Function ' EditMappings1$
Function ResetMapping1$
Dim RoutineName As String: RoutineName = "ResetMapping1$"
Dim in$
Dim iPlayer As Integer
Dim sResult As String: sResult = ""
' INITIALIZE
InitKeyboardButtonCodes
' RESET
Do
PrintControllerMap2
Print "To delete mapping, enter a player number: " _
"1-" + cstr$(cMaxPlayers) + ", " + _
cstr$(cMaxPlayers+1) + " for all, or 0 to exit."
Input "Delete mapping for player? "; iPlayer
If iPlayer > 0 And iPlayer <= cMaxPlayers Then
Print "Delete mappings for player " + cstr$(iPlayer) + "."
Input "Delete (y/n)"; in$: in$ = LCase$(_Trim$(in$))
If in$ = "y" Then
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ) <> "unknown" Then
m_arrControlMap(iPlayer, iWhichInput).device = 0
m_arrControlMap(iPlayer, iWhichInput).typ = 0
m_arrControlMap(iPlayer, iWhichInput).code = 0
m_arrControlMap(iPlayer, iWhichInput).value = 0
m_arrControlMap(iPlayer, iWhichInput).repeat = 0 ' GetGlobalInputRepeatSetting%(iWhichInput)
End If
Next iWhichInput
sResult = "Mappings deleted for player " + cstr$(iPlayer) + "."
Print sResult
End If
ElseIf iPlayer = (cMaxPlayers + 1) Then
Input "Delete all mappings (y/n)"; in$: in$ = LCase$(_Trim$(in$))
If in$ = "y" Then
For iPlayer = 1 To cMaxPlayers
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
If InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ) <> "unknown" Then
m_arrControlMap(iPlayer, iWhichInput).device = 0
m_arrControlMap(iPlayer, iWhichInput).typ = 0
m_arrControlMap(iPlayer, iWhichInput).code = 0
m_arrControlMap(iPlayer, iWhichInput).value = 0
m_arrControlMap(iPlayer, iWhichInput).repeat = 0 ' GetGlobalInputRepeatSetting%(iWhichInput)
End If
Next iWhichInput
Next iPlayer
sResult = "All mappings deleted."
Print sResult
End If
Else
If Len(sResult) = 0 Then sResult = "(Cancelled.)"
Exit Do
End If
Loop
ResetMapping1$ = sResult
End Function ' ResetMapping1$
' Usage:
' Dim StringArray(1 To 48, 1 To 128) As String
' StringToArray StringArray(), GetMap$
' version 2 with indexed array(row, columm)
Sub StringToArray (MyArray() As String, MyString As String)
Dim sDelim As String
ReDim arrLines(0) As String
Dim iRow As Integer
Dim iCol As Integer
Dim sChar As String
Dim iDim1 As Integer
Dim iDim2 As Integer
Dim iIndex1 As Integer
Dim iIndex2 As Integer
iDim1 = LBound(MyArray, 1)
iDim2 = LBound(MyArray, 2)
sDelim = Chr$(13)
split MyString, sDelim, arrLines()
For iRow = LBound(arrLines) To UBound(arrLines)
If iRow <= UBound(MyArray, 1) Then
For iCol = 1 To Len(arrLines(iRow))
If iCol <= UBound(MyArray, 2) Then
sChar = Mid$(arrLines(iRow), iCol, 1)
If Len(sChar) > 1 Then
sChar = Left$(sChar, 1)
Else
If Len(sChar) = 0 Then
sChar = "."
End If
End If
iIndex1 = iRow + iDim1
iIndex2 = (iCol - 1) + iDim2
MyArray(iIndex1, iIndex2) = sChar
'DebugPrint "MyArray(" + cstr$(iIndex1) + ", " + cstr$(iIndex2) + " = " + chr$(34) + sChar + chr$(34)
Else
' Exit if out of bounds
Exit For
End If
Next iCol
Else
' Exit if out of bounds
Exit For
End If
Next iRow
End Sub ' StringToArray
' /////////////////////////////////////////////////////////////////////////////
' Size of array:
'
' Resolution Cols Rows
' 1024 x 768 128 48
' 48 total available # of rows
' -2 rows for title
' -1 row for headings
' -1 for player #1 info
' -1 for player #2 info
' -1 for player #3 info
' -1 for player #4 info
' -- --------------------------
' 41 rows available
Function ArrayToString$ (MyArray( 1 To 32 , 1 To 32) As String)
Dim MyString As String
Dim iY As Integer
Dim iX As Integer
Dim sLine As String
MyString = ""
For iY = LBound(MyArray, 1) To UBound(MyArray, 1)
sLine = ""
For iX = LBound(MyArray, 2) To UBound(MyArray, 2)
sLine = sLine + MyArray(iY, iX)
Next iX
MyString = MyString + sLine + Chr$(13)
Next iY
ArrayToString$ = MyString
End Function ' ArrayToString$
' /////////////////////////////////////////////////////////////////////////////
' Looks in MyArray for the first element
' whose .item matches sItem and .name matches sName
' and returns the index, or MyArray's lbound-1 if not found.
Function GetTextLabel% (MyArray() As TextLabelType, sItem As String, sName As String)
Dim iResult As Integer: iResult = LBound(MyArray) - 1
Dim iLoop As Integer
For iLoop = LBound(MyArray) To UBound(MyArray)
If MyArray(iLoop).item = sItem Then
If MyArray(iLoop).name = sName Then
iResult = iLoop
Exit For
End If
End If
Next iLoop
GetTextLabel% = iResult
End Function ' GetTextLabel%
' /////////////////////////////////////////////////////////////////////////////
' Looks in MyArray for the first element
' whose .item matches sItem and .name matches sName
' and returns the index, or MyArray's lbound-1 if not found.
Function GetTextField% (MyArray() As TextFieldType, sItem As String, sName As String)
Dim iResult As Integer: iResult = LBound(MyArray) - 1
Dim iLoop As Integer
For iLoop = LBound(MyArray) To UBound(MyArray)
If MyArray(iLoop).item = sItem Then
If MyArray(iLoop).name = sName Then
iResult = iLoop
Exit For
End If
End If
Next iLoop
GetTextField% = iResult
End Function ' GetTextField%
' /////////////////////////////////////////////////////////////////////////////
' The following must be initialized and populated before calling:
' ReDim arrColor(-1) As ColorType
' Dim MapArray(1 To 48, 1 To 128) As String ' FOR SCREEN 1024 x 768: 128 x 48
' ReDim ScreenArray(1 To 48, 1 To 128) As TextCellType ' FOR SCREEN 1024 x 768: 128 x 48
' This was an experiment in rolling your own "GUI",
' - what a pain it turned out to be
' - next time maybe we would use InForm and be done with it!
' TODO:
' * clean up and remove all the variable definitions left over
' from the main routine MapInput2$ this was moved out of.
Sub UpdateDisplayMapInput2( _
arrColor() As ColorType, _
MapArray() As String, _
ScreenArray() As TextCellType _
)
Dim RoutineName As String: RoutineName = "UpdateDisplayMapInput2"
Dim iDeviceCount As Integer ' count # of devices connected to system (keyboard, mouse, game controllers)
Dim iPlayer As Integer ' same as iController, which of the 8 controllers
Dim iWhichInput As Integer ' one of: cInputUp, cInputDown, cInputLeft, cInputRight, cInputButton1, cInputButton2, cInputButton3, cInputButton4
Dim iCount As Integer
Dim sResult As String
Dim sError As String
Dim sLine As String
Dim iForeColor As Integer
Dim iBackColor As Integer
Dim sName As String
Dim sItem As String
Dim iLoop As Integer
Dim iX As Integer
Dim iY As Integer
Dim iCol As Integer
Dim iRow As Integer
Dim iStartX As Integer
Dim iStartY As Integer
Dim iEndX As Integer
Dim iEndY As Integer
Dim iLabel As Integer
Dim iField As Integer
Dim iButton As Integer
Dim iWidth As Integer
Dim bContinue As Integer
Dim sValue As String
Dim iNextWidth As Integer
Dim iType As Integer
Dim iIndex As Integer
Dim in$
' MOUSE VARIABLES
Dim iX1 As Integer: iX1 = 0
Dim iY1 As Integer: iY1 = 0
Dim iOldX1 As Integer: iOldX1 = 0
Dim iOldY1 As Integer: iOldY1 = 0
Dim bLeftClick As Integer: bLeftClick = FALSE
Dim bOldLeftClick As Integer: bOldLeftClick = FALSE
Dim sZone1 As String ' text description of which controller
Dim sZone2 As String ' text description of which input
Dim iMapPlayer As Integer ' like iController, which of the 8 controllers
' WHICH PORTION OF THE SCREEN USER CLICKED ON
Dim iTempX As Integer
Dim iTempY As Integer
Dim iTextX As Integer
Dim iTextY As Integer
Dim iOffsetX As Integer
Dim iOffsetY As Integer
' BOUNDARIES OF BUTTON CLICKED ON
Dim iMapX1 As Integer
Dim iMapX2 As Integer
Dim iMapY1 As Integer
Dim iMapY2 As Integer
' FOR MAPPING CONTROLLER INPUT
Dim arrButton(32, 16) As Integer ' number of buttons on the joystick
Dim arrButtonNew(32, 16) As Integer ' tracks when to initialize values
Dim arrAxis(32, 16) As Double ' number of axis on the joystick
Dim arrAxisNew(32, 16) As Integer ' tracks when to initialize values
'Dim iPlayer As Integer
Dim iController As Integer ' to loop through devices
Dim iDevice As Integer
Dim iValue As Integer
Dim bMoveNext As Integer
'Dim iLoop As Integer
Dim iCode As Integer
Dim bHaveInput As Integer
'Dim iWhichInput As Integer
Dim iNextInput As Integer
Dim sPrompt As String
Dim dblNextAxis As Double
Dim ExitX As Integer
Dim ExitY As Integer
' DISPLAY BORDERS
Cls
For iRow = LBound(MapArray, 1) To UBound(MapArray, 1)
For iCol = LBound(MapArray, 2) To UBound(MapArray, 2)
Color cRed, cBlack
PrintString2 iRow, iCol, MapArray(iRow, iCol), ScreenArray()
Next iCol
Next iRow
' DISPLAY CONTROL MAPPINGS
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
' Get top left screen coordinates for this controller
iStartX = m_arrScreenArea(iPlayer).x1
iStartY = m_arrScreenArea(iPlayer).y1
iEndX = m_arrScreenArea(iPlayer).x2
iEndY = m_arrScreenArea(iPlayer).y2
iWidth = (iEndX - iStartX) + 1
sItem = "Section"
iIndex = GetTextLabel%(m_arrTextLabel(), sItem, "caption")
If iIndex >= LBound(m_arrTextLabel) Then
iRow = (iStartY + m_arrTextLabel(iIndex).row) - 1
iCol = (iStartX + m_arrTextLabel(iIndex).column) - 1
sValue = m_arrTextLabel(iIndex).caption
sValue = Replace$(sValue, "{p}", cstr$(iPlayer))
iNextWidth = m_arrTextLabel(iIndex).width
If iNextWidth > 0 Then
Select Case m_arrTextLabel(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
ElseIf iNextWidth < 0 Then
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
Else
' (IGNORE)
End If
End If
' POPULATE EACH INPUT FOR THIS PLAYER/CONTROLLER:
' up,down,left,right,button #1,button #2,button #3,button #4
' iWhichInput is one of: cInputUp, cInputDown, cInputLeft, cInputRight, cInputButton1, cInputButton2, cInputButton3, cInputButton4
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN ITEM
' get value to match against m_arrTextField(iField).item
sItem = InputToItem$(iWhichInput)
'InputToString$(iWhichInput) returns up,down,left,right,button #1,button #2,button #3,button #4
' find the layout for each field for this input
' m_arrTextField(iField).name = type,device,code,repeat,value
' END ITEM
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN CAPTION
' LABEL
iIndex = GetTextLabel%(m_arrTextLabel(), sItem, "caption")
If iIndex >= LBound(m_arrTextLabel) Then
iRow = (iStartY + m_arrTextLabel(iIndex).row) - 1
iCol = (iStartX + m_arrTextLabel(iIndex).column) - 1
sValue = m_arrTextLabel(iIndex).caption
sValue = Replace$(sValue, "{p}", cstr$(iPlayer))
iNextWidth = m_arrTextLabel(iIndex).width
If iNextWidth > 0 Then
Select Case m_arrTextLabel(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
ElseIf iNextWidth < 0 Then
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
Else
' (IGNORE)
End If
End If
' END CAPTION
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN TYPE
' values are cInputNone,cInputKey,cInputButton,cInputAxis
' InputTypeToString$ returns none,key,button,axis,unknown
' VALUE
iIndex = GetTextField%(m_arrTextField(), sItem, "type")
If iIndex >= LBound(m_arrTextField) Then
iRow = (iStartY + m_arrTextField(iIndex).row) - 1
iCol = (iStartX + m_arrTextField(iIndex).column) - 1
sValue = InputTypeToString$(m_arrControlMap(iPlayer, iWhichInput).typ)
iNextWidth = m_arrTextField(iIndex).width
Select Case m_arrTextField(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextField(iIndex).fgcolor, m_arrTextField(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
End If
' LABEL
iIndex = GetTextLabel%(m_arrTextLabel(), sItem, "type")
If iIndex >= LBound(m_arrTextLabel) Then
iRow = (iStartY + m_arrTextLabel(iIndex).row) - 1
iCol = (iStartX + m_arrTextLabel(iIndex).column) - 1
sValue = m_arrTextLabel(iIndex).caption
sValue = Replace$(sValue, "{p}", cstr$(iPlayer))
iNextWidth = m_arrTextLabel(iIndex).width
If iNextWidth > 0 Then
Select Case m_arrTextLabel(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
ElseIf iNextWidth < 0 Then
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
Else
' (IGNORE)
End If
End If
' END TYPE
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN DEVICE
' VALUE
iIndex = GetTextField%(m_arrTextField(), sItem, "device")
If iIndex >= LBound(m_arrTextField) Then
iRow = (iStartY + m_arrTextField(iIndex).row) - 1
iCol = (iStartX + m_arrTextField(iIndex).column) - 1
sValue = cstr$(m_arrControlMap(iPlayer, iWhichInput).device)
iNextWidth = m_arrTextField(iIndex).width
Select Case m_arrTextField(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextField(iIndex).fgcolor, m_arrTextField(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
End If
' LABEL
iIndex = GetTextLabel%(m_arrTextLabel(), sItem, "device")
If iIndex >= LBound(m_arrTextLabel) Then
iRow = (iStartY + m_arrTextLabel(iIndex).row) - 1
iCol = (iStartX + m_arrTextLabel(iIndex).column) - 1
sValue = m_arrTextLabel(iIndex).caption
sValue = Replace$(sValue, "{p}", cstr$(iPlayer))
iNextWidth = m_arrTextLabel(iIndex).width
If iNextWidth > 0 Then
Select Case m_arrTextLabel(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
ElseIf iNextWidth < 0 Then
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
Else
' (IGNORE)
End If
End If
' END DEVICE
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN CODE
' VALUE
iIndex = GetTextField%(m_arrTextField(), sItem, "code")
If iIndex >= LBound(m_arrTextField) Then
iRow = (iStartY + m_arrTextField(iIndex).row) - 1
iCol = (iStartX + m_arrTextField(iIndex).column) - 1
If m_arrControlMap(iPlayer, iWhichInput).typ = cInputKey Then
sValue = GetKeyboardButtonCodeText$(m_arrControlMap(iPlayer, iWhichInput).code)
Else
sValue = cstr$(m_arrControlMap(iPlayer, iWhichInput).code)
End If
iNextWidth = m_arrTextField(iIndex).width
Select Case m_arrTextField(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextField(iIndex).fgcolor, m_arrTextField(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
End If
' LABEL
iIndex = GetTextLabel%(m_arrTextLabel(), sItem, "code")
If iIndex >= LBound(m_arrTextLabel) Then
iRow = (iStartY + m_arrTextLabel(iIndex).row) - 1
iCol = (iStartX + m_arrTextLabel(iIndex).column) - 1
sValue = m_arrTextLabel(iIndex).caption
sValue = Replace$(sValue, "{p}", cstr$(iPlayer))
iNextWidth = m_arrTextLabel(iIndex).width
If iNextWidth > 0 Then
Select Case m_arrTextLabel(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
ElseIf iNextWidth < 0 Then
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
Else
' (IGNORE)
End If
End If
' END CODE
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN VALUE
' VALUE
iIndex = GetTextField%(m_arrTextField(), sItem, "value")
If iIndex >= LBound(m_arrTextField) Then
iRow = (iStartY + m_arrTextField(iIndex).row) - 1
iCol = (iStartX + m_arrTextField(iIndex).column) - 1
sValue = cstr$(m_arrControlMap(iPlayer, iWhichInput).value)
iNextWidth = m_arrTextField(iIndex).width
Select Case m_arrTextField(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextField(iIndex).fgcolor, m_arrTextField(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
End If
' LABEL
iIndex = GetTextLabel%(m_arrTextLabel(), sItem, "value")
If iIndex >= LBound(m_arrTextLabel) Then
iRow = (iStartY + m_arrTextLabel(iIndex).row) - 1
iCol = (iStartX + m_arrTextLabel(iIndex).column) - 1
sValue = m_arrTextLabel(iIndex).caption
sValue = Replace$(sValue, "{p}", cstr$(iPlayer))
iNextWidth = m_arrTextLabel(iIndex).width
If iNextWidth > 0 Then
Select Case m_arrTextLabel(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
ElseIf iNextWidth < 0 Then
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
Else
' (IGNORE)
End If
End If
' END VALUE
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN REPEAT
' VALUE
iIndex = GetTextField%(m_arrTextField(), sItem, "repeat")
If iIndex >= LBound(m_arrTextField) Then
iRow = (iStartY + m_arrTextField(iIndex).row) - 1
iCol = (iStartX + m_arrTextField(iIndex).column) - 1
sValue = IIFSTR$(m_arrControlMap(iPlayer, iWhichInput).repeat, "Y", "N")
iNextWidth = m_arrTextField(iIndex).width
Select Case m_arrTextField(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextField(iIndex).fgcolor, m_arrTextField(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
End If
' LABEL
iIndex = GetTextLabel%(m_arrTextLabel(), sItem, "repeat")
If iIndex >= LBound(m_arrTextLabel) Then
iRow = (iStartY + m_arrTextLabel(iIndex).row) - 1
iCol = (iStartX + m_arrTextLabel(iIndex).column) - 1
sValue = m_arrTextLabel(iIndex).caption
sValue = Replace$(sValue, "{p}", cstr$(iPlayer))
iNextWidth = m_arrTextLabel(iIndex).width
If iNextWidth > 0 Then
Select Case m_arrTextLabel(iIndex).justify
Case cJustifyLeft:
sValue = StrJustifyLeft$(sValue, iNextWidth)
Case cJustifyRight:
sValue = StrJustifyRight$(sValue, iNextWidth)
Case cJustifyCenter:
sValue = StrJustifyCenter$(sValue, iNextWidth)
Case Else:
' (DO NOTHING)
End Select
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
ElseIf iNextWidth < 0 Then
Color m_arrTextLabel(iIndex).fgcolor, m_arrTextLabel(iIndex).bgcolor
PrintString2 iRow, iCol, sValue, ScreenArray()
Else
' (IGNORE)
End If
End If
' END REPEAT
' +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Function MapInput2$
Dim RoutineName As String: RoutineName = "MapInput2$"
Dim iDeviceCount As Integer ' count # of devices connected to system (keyboard, mouse, game controllers)
Dim iPlayer As Integer ' same as iController, which of the 8 controllers
Dim iWhichInput As Integer ' one of: cInputUp, cInputDown, cInputLeft, cInputRight, cInputButton1, cInputButton2, cInputButton3, cInputButton4
Dim bFinished As Integer
Dim iCount As Integer
Dim sResult As String
Dim sError As String: sError = ""
Dim sLine As String
ReDim arrColor(-1) As ColorType
Dim iForeColor As Integer
Dim iBackColor As Integer
' VARIABLES FOR SCREEN
' 1024 x 768: 128 x 48
Dim MapArray(1 To 48, 1 To 128) As String
ReDim ScreenArray(1 To 48, 1 To 128) As TextCellType
Dim sName As String
Dim sItem As String
Dim iLoop As Integer
Dim iX As Integer
Dim iY As Integer
Dim iCol As Integer
Dim iRow As Integer
Dim iStartX As Integer
Dim iStartY As Integer
Dim iEndX As Integer
Dim iEndY As Integer
Dim iLabel As Integer
Dim iField As Integer
Dim iButton As Integer
Dim iWidth As Integer
Dim bContinue As Integer
Dim sValue As String
Dim iNextWidth As Integer
Dim iType As Integer
Dim iIndex As Integer
Dim in$
' MOUSE VARIABLES
Dim iX1 As Integer: iX1 = 0
Dim iY1 As Integer: iY1 = 0
Dim iOldX1 As Integer: iOldX1 = 0
Dim iOldY1 As Integer: iOldY1 = 0
Dim bLeftClick As Integer: bLeftClick = FALSE
Dim bOldLeftClick As Integer: bOldLeftClick = FALSE
Dim sZone1 As String ' text description of which controller
Dim sZone2 As String ' text description of which input
Dim iMapPlayer As Integer ' like iController, which of the 8 controllers
' WHICH PORTION OF THE SCREEN USER CLICKED ON
Dim iTempX As Integer
Dim iTempY As Integer
Dim iTextX As Integer
Dim iTextY As Integer
Dim iOffsetX As Integer
Dim iOffsetY As Integer
' BOUNDARIES OF BUTTON CLICKED ON
Dim iMapX1 As Integer
Dim iMapX2 As Integer
Dim iMapY1 As Integer
Dim iMapY2 As Integer
' FOR MAPPING CONTROLLER INPUT
Dim arrButton(32, 16) As Integer ' number of buttons on the joystick
Dim arrButtonNew(32, 16) As Integer ' tracks when to initialize values
Dim arrAxis(32, 16) As Double ' number of axis on the joystick
Dim arrAxisNew(32, 16) As Integer ' tracks when to initialize values
'Dim iPlayer As Integer
Dim iController As Integer ' to loop through devices
Dim iDevice As Integer
Dim iValue As Integer
Dim bMoveNext As Integer
'Dim iLoop As Integer
Dim iCode As Integer
Dim bHaveInput As Integer
'Dim iWhichInput As Integer
Dim iNextInput As Integer
Dim sCaption As String
Dim sPrompt As String
Dim CloseX As Integer
Dim CloseY As Integer
Dim dblNextAxis As Double
Dim bHitEsc As Integer
Dim bHitEnter As Integer
Dim bMapMode As Integer ' if TRUE then look for control mapping input
Dim ExitX As Integer
Dim ExitY As Integer
' =============================================================================
' INITIALIZE
If Len(sError) = 0 Then
InitKeyboardButtonCodes
AddColors arrColor()
StringToArray MapArray(), GetMap$
SetupScreenAreas
SetupButtons
SetupTextLabels
SetupTextFields
End If
' =============================================================================
' MAKE SURE WE HAVE DEVICES
If Len(sError) = 0 Then
' 1 is the keyboard
' 2 is the mouse
' 3 is the joystick
' unless someone has a strange setup with multiple mice/keyboards/ect...
' In that case, you can use _DEVICE$(i) to look for "KEYBOARD", "MOUSE", "JOYSTICK", if necessary.
' I've never actually found it necessary, but I figure it's worth mentioning, just in case...
iDeviceCount = _Devices ' Find the number of devices on someone's system
'If iDeviceCount < 3 Then
If iDeviceCount < 1 Then
sError = "Enough devices not found."
End If
End If
' =============================================================================
' COUNT # OF JOYSTICKS
' TODO: find out the right way to count joysticks
If Len(sError) = 0 Then
' D= _DEVICES ' MUST be read in order for other 2 device functions to work!
iDeviceCount = _Devices ' Find the number of devices on someone's system
If iDeviceCount > 2 Then
' LIMIT # OF DEVICES, IF THERE IS A LIMIT DEFINED
iNumControllers = iDeviceCount - 2
If cMaxControllers > 0 Then
If iNumControllers > cMaxControllers Then
iNumControllers = cMaxControllers
End If
End If
Else
' ONLY 2 FOUND (KEYBOARD, MOUSE)
'sError = "No game controllers found."
iNumControllers = 0
End If
End If
' =============================================================================
' INITIALIZE CONTROLLER DATA
If Len(sError) = 0 Then
For iController = 1 To iNumControllers
m_arrController(iController).buttonCount = cMaxButtons
m_arrController(iController).axisCount = cMaxAxis
For iLoop = 1 To cMaxButtons
arrButtonNew(iController, iLoop) = TRUE
Next iLoop
For iLoop = 1 To cMaxAxis
arrAxisNew(iController, iLoop) = TRUE
Next iLoop
Next iController
End If
' =============================================================================
' INITIALIZE CONTROLLER INPUT
If Len(sError) = 0 Then
_KeyClear: _Delay 1
For iController = 1 To iNumControllers
iDevice = iController + 2
While _DeviceInput(iDevice) ' clear and update the device buffer
For iLoop = 1 To _LastButton(iDevice)
If (iLoop > cMaxButtons) Then Exit For
m_arrController(iController).buttonCount = iLoop
arrButton(iController, iLoop) = FALSE
Next iLoop
For iLoop = 1 To _LastAxis(iDevice) ' this loop checks all my axis
If (iLoop > cMaxAxis) Then Exit For
m_arrController(iController).axisCount = iLoop
arrAxis(iController, iLoop) = 0
Next iLoop
Wend ' clear and update the device buffer
Next iController
End If
' =============================================================================
' INIT SCREEN
If Len(sError) = 0 Then
Screen _NewImage(1024, 768, 32): _ScreenMove 0, 0
UpdateDisplayMapInput2 arrColor(), MapArray(), ScreenArray()
End If
' =============================================================================
' MAIN LOOP
If Len(sError) = 0 Then
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN MOUSE #MOUSE
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
bFinished = FALSE
bHitEsc = FALSE
bHitEnter = FALSE
bMapMode = FALSE
CloseX = 0
CloseY = 0
_MouseHide ' hide OS mouse pointer
Do
' READ MOUSE
Do While _MouseInput: Loop
' ERASE CURSOR
If iOldX1 <> iX1 Or iOldY1 <> iY1 Then
If iOldX1 > 0 And iOldY1 > 0 Then
iTextY = iOldY1 \ _FontHeight
iTextX = iOldX1 \ _FontWidth
Color ScreenArray(iTextY, iTextX).fgColor, ScreenArray(iTextY, iTextX).bgcolor
PrintString1 iTextY, iTextX, ScreenArray(iTextY, iTextX).value
End If
End If
' SAVE OLD POSITION
iOldY1 = iY1
iOldX1 = iX1
iTempY = iOldY1 \ _FontHeight
iTempX = iOldX1 \ _FontWidth
' LEFT CLICK
bLeftClick = _MouseButton(1)
If bLeftClick Then
If bOldLeftClick = FALSE Then
' (CLICK ACTION HERE)
' IS SELECTING A CONTROL TO MAP, OR MAPPING A CONTROL?
If CloseX = 0 Or CloseY = 0 Then
' DID THEY CLOSE?
If (iTextY = ExitY) And (iTextX >= ExitX) Then
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
' USER CLICKED EXIT SCREEN
bFinished = TRUE
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
Else
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
' BEGIN USER IS SELECTING A CONTROL TO MAP
' WHICH CONTROLLER?
For iIndex = LBound(m_arrScreenArea) To UBound(m_arrScreenArea)
If iTextY >= m_arrScreenArea(iIndex).y1 Then
If iTextY <= m_arrScreenArea(iIndex).y2 Then
If iTextX >= m_arrScreenArea(iIndex).x1 Then
If iTextX <= m_arrScreenArea(iIndex).x2 Then
iMapPlayer = m_arrScreenArea(iIndex).player
sZone1 = m_arrScreenArea(iIndex).item
iOffsetX = m_arrScreenArea(iIndex).x1 - 1
iOffsetY = m_arrScreenArea(iIndex).y1 - 1
Exit For
End If
End If
End If
End If
Next iIndex
' WHICH BUTTON?
For iIndex = LBound(m_arrButton) To UBound(m_arrButton)
If iTextY >= m_arrButton(iIndex).y1 + iOffsetY Then
If iTextY <= m_arrButton(iIndex).y2 + iOffsetY Then
If iTextX >= m_arrButton(iIndex).x1 + iOffsetX Then
If iTextX <= m_arrButton(iIndex).x2 + iOffsetX Then
End If
End If
End If
End If
Next iIndex
' END USER IS SELECTING A CONTROL TO MAP
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
End If
Else
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
' BEGIN USER CLICKED TO CANCEL MAP MODE
If iTextY = ExitY And iTextX = ExitX Then
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
' USER CLICKED EXIT SCREEN
bFinished = TRUE
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
ElseIf iTextY = CloseY And iTextX = CloseX Then
' USER CLICKED "CANCEL"
End If
' END USER CLICKED TO CANCEL MAP MODE
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
End If
bOldLeftClick = TRUE
End If
Else
' USER RELEASED MOUSE BUTTON
bOldLeftClick = FALSE
sZone1 = ""
sZone2 = ""
iOffsetX = 0
iOffsetY = 0
End If
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
' BEGIN LOOK FOR MAPPING INPUT
If CloseX > 0 And CloseY > 0 Then
' =============================================================================
' BEGIN LOOK FOR NEXT INPUT
bMoveNext = FALSE
' -----------------------------------------------------------------------------
' BEGIN CHECK FOR CONTROLLER INPUT
For iController = 1 To iNumControllers
iDevice = iController + 2
' Check all devices
While _DeviceInput(iDevice)
' Check each button
If bMoveNext = FALSE Then
For iLoop = 1 To _LastButton(iDevice)
If (iLoop > cMaxButtons) Then Exit For
'm_arrController(iController).buttonCount = iLoop
' update button array to indicate if a button is up or down currently.
If _ButtonChange(iLoop) Then
iValue = _Button(iLoop)
If iValue <> arrButton(iController, iLoop) Then
' *****************************************************************************
' PRESSED BUTTON
' make sure this isn't already mapped
bHaveInput = TRUE
If iWhichInput > LBound(m_arrControlMap, 2) Then
' is input unique?
For iNextInput = LBound(m_arrControlMap, 2) To iWhichInput - 1
If m_arrControlMap(iMapPlayer, iNextInput).device = iDevice Then
If m_arrControlMap(iMapPlayer, iNextInput).typ = cInputButton Then
If m_arrControlMap(iMapPlayer, iNextInput).code = iLoop Then
If m_arrControlMap(iMapPlayer, iNextInput).value = iValue Then
bHaveInput = FALSE
End If
End If
End If
End If
Next iNextInput
End If
If bHaveInput Then
m_arrControlMap(iMapPlayer, iWhichInput).device = iDevice
m_arrControlMap(iMapPlayer, iWhichInput).typ = cInputButton
m_arrControlMap(iMapPlayer, iWhichInput).code = iLoop
m_arrControlMap(iMapPlayer, iWhichInput).value = iValue
bMoveNext = TRUE
Exit For
End If
End If
End If
Next iLoop
End If
' Check each axis
If bMoveNext = FALSE Then
For iLoop = 1 To _LastAxis(iDevice)
If (iLoop > cMaxAxis) Then Exit For
'm_arrController(iController).axisCount = iLoop
' I like to give a little "jiggle" resistance to my controls, as I have an old joystick
' which is prone to always give minute values and never really center on true 0.
' A value of 1 means my axis is pushed fully in one direction.
' A value greater than 0.1 means it's been partially pushed in a direction (such as at a 45 degree diagional angle).
' A value of less than 0.1 means we count it as being centered. (As if it was 0.)
'These are way too sensitive for analog:
'IF ABS(_AXIS(iLoop)) <= 1 AND ABS(_AXIS(iLoop)) >= .1 THEN
'IF ABS(dblNextAxis) <= 1 AND ABS(dblNextAxis) >= .01 THEN
'IF ABS(dblNextAxis) <= 1 AND ABS(dblNextAxis) >= .001 THEN
'For digital input, we'll use a big picture:
If Abs(dblNextAxis) <= 1 And Abs(dblNextAxis) >= 0.75 Then
If dblNextAxis <> arrAxis(iController, iLoop) Then
' *****************************************************************************
' MOVED STICK
' convert to a digital value
If dblNextAxis < 0 Then
iValue = -1
Else
iValue = 1
End If
' make sure this isn't already mapped
bHaveInput = TRUE
If iWhichInput > LBound(m_arrControlMap, 2) Then
' is input unique?
For iNextInput = LBound(m_arrControlMap, 2) To iWhichInput - 1
If m_arrControlMap(iMapPlayer, iNextInput).device = iDevice Then
If m_arrControlMap(iMapPlayer, iNextInput).typ = cInputAxis Then
If m_arrControlMap(iMapPlayer, iNextInput).code = iLoop Then
If m_arrControlMap(iMapPlayer, iNextInput).value = iValue Then
bHaveInput = FALSE
End If
End If
End If
End If
Next iNextInput
End If
If bHaveInput Then
m_arrControlMap(iMapPlayer, iWhichInput).device = iDevice
m_arrControlMap(iMapPlayer, iWhichInput).typ = cInputAxis
m_arrControlMap(iMapPlayer, iWhichInput).code = iLoop
m_arrControlMap(iMapPlayer, iWhichInput).value = iValue
bMoveNext = TRUE
Exit For
End If
End If
End If
Next iLoop
End If
Wend ' clear and update the device buffer
Next iController
' END CHECK FOR CONTROLLER INPUT
' -----------------------------------------------------------------------------
' -----------------------------------------------------------------------------
' BEGIN CHECK FOR KEYBOARD INPUT #1
If bMoveNext = FALSE Then
'_KEYCLEAR: _DELAY 1
While _DeviceInput(1): Wend ' clear and update the keyboard buffer
' Detect changed key state
For iLoop = LBound(m_arrButtonCode) To UBound(m_arrButtonCode)
iCode = m_arrButtonCode(iLoop)
If _Button(iCode) <> FALSE Then
' *****************************************************************************
' PRESSED KEYBOARD
'PRINT "PRESSED " + m_arrButtonKey(iLoop)
' make sure this isn't already mapped
bHaveInput = TRUE
If iWhichInput > LBound(m_arrControlMap, 2) Then
' is input unique?
For iNextInput = LBound(m_arrControlMap, 2) To iWhichInput - 1
If m_arrControlMap(iMapPlayer, iNextInput).device = 1 Then ' .device 1 = keyboard
If m_arrControlMap(iMapPlayer, iNextInput).typ = cInputKey Then
If m_arrControlMap(iMapPlayer, iNextInput).code = iCode Then
'if m_arrControlMap(iMapPlayer, iNextInput).value = TRUE then
bHaveInput = FALSE
'end if
End If
End If
End If
Next iNextInput
End If
' REFRESH SCREEN
UpdateDisplayMapInput2 arrColor(), MapArray(), ScreenArray()
End If
' END LOOK FOR NEXT INPUT
' =============================================================================
End If
' END LOOK FOR MAPPING INPUT
' @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
_Limit 30
Loop Until bFinished
_MouseShow "default": _Delay 0.5
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END MOUSE @MOUSE
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
End If
' CLEAR KEYBOARD BUFFER
_KeyClear
' CLEAR SCREEN
Color cWhite, cBlack
Cls
' DONE
MapInput2$ = sResult
End Function ' MapInput2$
' /////////////////////////////////////////////////////////////////////////////
' Displays a popup with a close button "X" in the upper right,
' with text in MyString.
' Returns the x,y position of the close button in parameters CloseX, CloseY.
Sub MapInputPopup( _
sCaption2 As String, fgCaptionColor As _Unsigned Long, bgCaptionColor As _Unsigned Long, _
sText As String, fgTextColor As _Unsigned Long, bgTextColor As _Unsigned Long, _
CloseX As Integer, CloseY As Integer, ScreenArray() As TextCellType _
)
ReDim arrLines(-1) As String
Dim sCaption As String
Dim iCols As Integer
Dim iRows As Integer
Dim iColCount As Integer
Dim iRowCount As Integer
Dim iLoopRows As Integer
Dim iWidth As Integer
Dim x1 As Integer
Dim x2 As Integer
Dim y1 As Integer
Dim y2 As Integer
Dim iY As Integer
' Figure out window size
sCaption = sCaption2
iCols = _Width(0) \ _FontWidth
iRows = _Height(0) \ _FontHeight
iColCount = Len(sCaption) + 1
iRowCount = 0
split sText, Chr$(13), arrLines()
For iLoopRows = LBound(arrLines) To UBound(arrLines)
iRowCount = iRowCount + 1
If Len(arrLines(iLoopRows)) > iColCount Then
iColCount = Len(arrLines(iLoopRows))
End If
Next iLoopRows
' Draw window as long as there is text
If iColCount > 0 Then
' Make sure popup is not wider than screen
If iColCount > iCols Then iColCount = iCols
If iRowCount > iRows Then iRowCount = iRows
If Len(sCaption) > (iCols - 1) Then sCaption = Left$(sCaption, iCols - 1)
' Draw the close button
Color fgTextColor, bgTextColor
'PrintString1 y1, x2, "X"
PrintString2 y1, x2, "X", ScreenArray()
CloseX = x2: CloseY = y1
' Get width
iWidth = (x2 - x1) + 1
' Draw the popup
iY = y1
Color fgTextColor, bgTextColor
For iLoopRows = LBound(arrLines) To UBound(arrLines)
iY = iY + 1: If iY > y2 Then Exit For
PrintString2 iY, x1, Left$(arrLines(iLoopRows) + String$(iWidth, " "), iWidth), ScreenArray()
Next iLoopRows
For iLoopRows = iY To y2
'PrintString1 iY, x1, String$(iColCount, " ")
PrintString2 iLoopRows, x1, String$(iWidth, " "), ScreenArray()
Next iLoopRows
Else
CloseX = 0: CloseY = 0
End If
End Sub ' MapInputPopup
Sub MapInputPrompt (MyString As String, x1 As Integer, x2 As Integer, y1 As Integer, y2 As Integer, fgColor As _Unsigned Long, bgColor As _Unsigned Long)
ReDim arrLines(-1) As String
Dim iLoopRows As Integer
Dim iMaxLen As Integer
Dim iY As Integer
If x2 >= x1 Then
If y2 >= y1 Then
iMaxLen = (x2 - x1) + 1
split MyString, Chr$(13), arrLines()
iLine = 0
iY = y1
For iLoopRows = LBound(arrLines) To UBound(arrLines)
Color fgColor, bgColor
If Len(arrLines(iLoopRows)) > iMaxLen Then
PrintString1 iY, x1, Left$(arrLines(iLoopRows), iMaxLen)
Else
PrintString1 iY, x1, Left$(arrLines(iLoopRows) + String$(iMaxLen, " "), iMaxLen)
End If
iY = iY + 1
If iY > y2 Then
Exit For
End If
Next iLoopRows
For iLoopRows = iY To y2
PrintString1 iY, x1, String$(iMaxLen, " ")
Next iLoopRows
End If
End If
End Sub ' MapInputPrompt
Sub DebugOut (sPrompt As String, iRow As Integer, iColumn As Integer, fgColor As _Unsigned Long, bgColor As _Unsigned Long)
Color fgColor, bgColor
PrintString iRow, iColumn, String$(128, " ")
PrintString iRow, iColumn, sPrompt
End Sub ' DebugOut
Function TestMouseXYButton$
Dim RoutineName As String: RoutineName = "TestMouseXYButton$"
Dim iX1 As Integer: iX1 = 0
Dim iY1 As Integer: iY1 = 0
Dim iOldX1 As Integer: iOldX1 = 0
Dim iOldY1 As Integer: iOldY1 = 0
Dim bLeftClick As Integer: bLeftClick = FALSE
Dim bRightClick As Integer: bRightClick = FALSE
Dim bMiddleClick As Integer: bMiddleClick = FALSE
Dim bOldLeftClick As Integer: bOldLeftClick = FALSE
Dim bOldRightClick As Integer: bOldRightClick = FALSE
Dim bOldMiddleClick As Integer: bOldMiddleClick = FALSE
Dim iX2 As Integer: iX2 = _Width / 2
Dim iY2 As Integer: iY2 = _Height / 2
Dim iOldX2 As Integer: iOldX2 = 0
Dim iOldY2 As Integer: iOldY2 = 0
Dim iColor1 As _Unsigned Long: iColor1 = cRed
Dim iColor2 As _Unsigned Long: iColor2 = cLime
Screen _NewImage(1280, 1024, 32): _ScreenMove 0, 0
Cls
_MouseHide ' hide OS mouse pointer
Do While _MouseInput: Loop
Do
Color cWhite, cBlack
If iOldX1 <> iX1 Or iOldY1 <> iY1 Then _PrintString (iOldX1, iOldY1), " "
If iOldX2 <> iX2 Or iOldY2 <> iY2 Then _PrintString (iOldX2, iOldY2), " "
bLeftClick = _MouseButton(1)
If bLeftClick Then
If bOldLeftClick = FALSE Then
If iColor1 = cOrangeRed Then
iColor1 = cRed
ElseIf iColor1 = cRed Then
iColor1 = cMagenta
Else
iColor1 = cOrangeRed
End If
bOldLeftClick = TRUE
End If
Else
bOldLeftClick = FALSE
End If
bRightClick = _MouseButton(2)
If bRightClick Then
If bOldRightClick = FALSE Then
If iColor2 = cBlue Then
iColor2 = cLime
ElseIf iColor2 = cLime Then
iColor2 = cYellow
Else
iColor2 = cBlue
End If
bOldRightClick = TRUE
End If
Else
bOldRightClick = FALSE
End If
bMiddleClick = _MouseButton(3)
Do While _MouseInput
If bMiddleClick Then
iY2 = iY2 + (_MouseWheel * _FontHeight) ' -1 up, 0 no movement, 1 down
Else
iX2 = iX2 + (_MouseWheel * _FontWidth) ' -1 up, 0 no movement, 1 down
End If
Loop
If iY2 < 1 Then iY2 = 1
If iY2 > (_Height - _FontHeight) Then iY2 = (HEIGHT - _FontHeight)
Loop Until _KeyDown(27)
_KeyClear
_MouseShow "default": _Delay 0.5
Screen 0
TestMouseXYButton$ = ""
End Function ' TestMouseXYButton$
Function MapInput1$
Dim RoutineName As String: RoutineName = "MapInput1$"
Dim in$
Dim iDeviceCount As Integer
Dim iPlayer As Integer
Dim sResult As String
Dim sError As String
' INITIALIZE
InitKeyboardButtonCodes
' SET UP SCREEN
Screen _NewImage(1280, 1024, 32): _ScreenMove 0, 0
' MAKE SURE WE HAVE DEVICES
' 1 is the keyboard
' 2 is the mouse
' 3 is the joystick
' unless someone has a strange setup with multiple mice/keyboards/ect...
' In that case, you can use _DEVICE$(i) to look for "KEYBOARD", "MOUSE", "JOYSTICK", if necessary.
' I've never actually found it necessary, but I figure it's worth mentioning, just in case...
iDeviceCount = _Devices ' Find the number of devices on someone's system
If iDeviceCount > 2 Then
'' Try loading map
'sError = LoadControllerMap1$
'if len(sError) = 0 then
' print "Previous controller mapping loaded."
'else
' print "*******************************************************************************"
' print "There were errors loading the controller mapping file:"
' print sError
' print
' print "Try remapping - a new file will be created."
' print "*******************************************************************************"
'end if
Do
PrintControllerMap2
Print "To edit mapping, enter a player number (1-" + cstr$(cMaxPlayers) + ") or 0 to exit."
Input "Get input for player? "; iPlayer
If iPlayer > 0 And iPlayer <= cMaxPlayers Then
sResult = MapInput1b$(iPlayer)
If Len(sResult) = 0 Then
Print "Remember to save mappings when done."
Else
Print sResult
End If
Else
sResult = "(Cancelled.)"
Exit Do
End If
Loop
Else
sResult = "No controller devices found."
Input "PRESS <ENTER> TO CONTINUE", in$
End If
MapInput1$ = sResult
End Function ' MapInput1$
' /////////////////////////////////////////////////////////////////////////////
' Detect controls
' THIS VERSION SUPPORTS UPTO 8 JOYSTICKS, WITH UPTO 2 BUTTONS AND 2 AXES EACH
' (THIS IS FOR ATARI 2600 JOYSTICKS)
' The following shared arrays must be declared:
' ReDim Shared m_arrButtonCode(1 To 99) As Long
' ReDim Shared m_arrButtonKey(1 To 99) As String
Function MapInput1b$ (iPlayer As Integer)
Dim RoutineName As String:: RoutineName = "MapInput1b$"
Dim sResult As String: sResult = ""
Dim sError As String: sError = ""
Dim in$
Dim iDeviceCount As Integer
Dim iDevice As Integer
Dim iNumControllers As Integer
Dim iController As Integer
Dim iLoop As Integer
Dim strValue As String
Dim strAxis As String
Dim dblNextAxis As Double
Dim iCount As Long
Dim iValue As Integer
Dim iCode As Integer
Dim arrButton(32, 16) As Integer ' number of buttons on the joystick
Dim arrButtonNew(32, 16) As Integer ' tracks when to initialize values
Dim arrAxis(32, 16) As Double ' number of axis on the joystick
Dim arrAxisNew(32, 16) As Integer ' tracks when to initialize values
'Dim arrInput(1 To 8) As ControlInputType
Dim iWhichInput As Integer
Dim bFinished As Integer
Dim bHaveInput As Integer
Dim bMoveNext As Integer
Dim bCancel As Integer
Dim iNextInput As Integer
' FOR PRINTING OUTPUT
Dim iDigits As Integer ' # digits to display (values are truncated to this length)
Dim iColCount As Integer
Dim iGroupCount As Integer
Dim sLine As String
Dim iCols As Integer
Dim iRows As Integer
Dim iMaxCols As Integer
' INITIALIZE
If Len(sError) = 0 Then
iDigits = 4 ' 11
iColCount = 3
iGroupCount = 0 ' re-initialized at the top of every loop
iCols = _Width(0) \ _FontWidth
iRows = _Height(0) \ _FontHeight
End If
' COUNT # OF JOYSTICKS
' TODO: find out the right way to count joysticks
If Len(sError) = 0 Then
' D= _DEVICES ' MUST be read in order for other 2 device functions to work!
iDeviceCount = _Devices ' Find the number of devices on someone's system
If iDeviceCount > 2 Then
' LIMIT # OF DEVICES, IF THERE IS A LIMIT DEFINED
iNumControllers = iDeviceCount - 2
If cMaxControllers > 0 Then
If iNumControllers > cMaxControllers Then
iNumControllers = cMaxControllers
End If
End If
Else
' ONLY 2 FOUND (KEYBOARD, MOUSE)
sError = "No game controllers found."
End If
End If
' INITIALIZE CONTROLLER DATA
If Len(sError) = 0 Then
For iController = 1 To iNumControllers
m_arrController(iController).buttonCount = cMaxButtons
m_arrController(iController).axisCount = cMaxAxis
For iLoop = 1 To cMaxButtons
arrButtonNew(iController, iLoop) = TRUE
Next iLoop
For iLoop = 1 To cMaxAxis
arrAxisNew(iController, iLoop) = TRUE
Next iLoop
Next iController
End If
' INITIALIZE CONTROLLER INPUT
If Len(sError) = 0 Then
Cls
Print "We will now detect controllers."
Print "Do not touch any keys or game controllers during detection."
Input "Press <ENTER> to begin"; in$
_KeyClear: Print
sLine = "Initializing controllers": Print sLine;
iMaxCols = (iCols - Len(sLine)) - 1
iCount = 0
Do
iCount = iCount + 1
If iCount < iMaxCols Then
Print ".";
Else
Print ".": Print sLine: iCount = 0
End If
For iController = 1 To iNumControllers
iDevice = iController + 2
While _DeviceInput(iDevice) ' clear and update the device buffer
For iLoop = 1 To _LastButton(iDevice)
If (iLoop > cMaxButtons) Then Exit For
m_arrController(iController).buttonCount = iLoop
'IF _BUTTONCHANGE(iLoop) THEN
' arrButton(iController, iLoop) = _BUTTON(iLoop)
'END IF
arrButton(iController, iLoop) = FALSE
Next iLoop
For iLoop = 1 To _LastAxis(iDevice) ' this loop checks all my axis
If (iLoop > cMaxAxis) Then Exit For
m_arrController(iController).axisCount = iLoop
arrAxis(iController, iLoop) = 0
Next iLoop
Wend ' clear and update the device buffer
Next iController
_Limit 30
Loop Until iCount > 60 ' quit after 2 seconds
Print: Print
End If
' WAIT FOR INPUT
If Len(sError) = 0 Then
Cls
Print "Press <ESCAPE> to cancel at any time."
Print
_KeyClear: _Delay 1
bCancel = FALSE
bFinished = FALSE
iLastPressed = 0
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
'print "iWhichInput=" + cstr$(iWhichInput)
Print "Player #" + cstr$(iPlayer) + " press control for " + InputToString$(iWhichInput) + " or ESC to skip: ";
' =============================================================================
' BEGIN LOOK FOR NEXT INPUT
bMoveNext = FALSE
Do
' -----------------------------------------------------------------------------
' BEGIN CHECK FOR CONTROLLER INPUT
For iController = 1 To iNumControllers
iDevice = iController + 2
' Check all devices
While _DeviceInput(iDevice)
' Check each button
If bMoveNext = FALSE Then
For iLoop = 1 To _LastButton(iDevice)
If (iLoop > cMaxButtons) Then Exit For
'm_arrController(iController).buttonCount = iLoop
' update button array to indicate if a button is up or down currently.
If _ButtonChange(iLoop) Then
iValue = _Button(iLoop)
If iValue <> arrButton(iController, iLoop) Then
' *****************************************************************************
' PRESSED BUTTON
' make sure this isn't already mapped
bHaveInput = TRUE
If iWhichInput > LBound(m_arrControlMap, 2) Then
' is input unique?
For iNextInput = LBound(m_arrControlMap, 2) To iWhichInput - 1
If m_arrControlMap(iPlayer, iNextInput).device = iDevice Then
If m_arrControlMap(iPlayer, iNextInput).typ = cInputButton Then
If m_arrControlMap(iPlayer, iNextInput).code = iLoop Then
If m_arrControlMap(iPlayer, iNextInput).value = iValue Then
bHaveInput = FALSE
End If
End If
End If
End If
Next iNextInput
End If
If bHaveInput Then
m_arrControlMap(iPlayer, iWhichInput).device = iDevice
m_arrControlMap(iPlayer, iWhichInput).typ = cInputButton
m_arrControlMap(iPlayer, iWhichInput).code = iLoop
m_arrControlMap(iPlayer, iWhichInput).value = iValue
bMoveNext = TRUE
End If
End If
End If
Next iLoop
End If
' Check each axis
If bMoveNext = FALSE Then
For iLoop = 1 To _LastAxis(iDevice)
If (iLoop > cMaxAxis) Then Exit For
'm_arrController(iController).axisCount = iLoop
' I like to give a little "jiggle" resistance to my controls, as I have an old joystick
' which is prone to always give minute values and never really center on true 0.
' A value of 1 means my axis is pushed fully in one direction.
' A value greater than 0.1 means it's been partially pushed in a direction (such as at a 45 degree diagional angle).
' A value of less than 0.1 means we count it as being centered. (As if it was 0.)
'These are way too sensitive for analog:
'IF ABS(_AXIS(iLoop)) <= 1 AND ABS(_AXIS(iLoop)) >= .1 THEN
'IF ABS(dblNextAxis) <= 1 AND ABS(dblNextAxis) >= .01 THEN
'IF ABS(dblNextAxis) <= 1 AND ABS(dblNextAxis) >= .001 THEN
'For digital input, we'll use a big picture:
If Abs(dblNextAxis) <= 1 And Abs(dblNextAxis) >= 0.75 Then
If dblNextAxis <> arrAxis(iController, iLoop) Then
' *****************************************************************************
' MOVED STICK
' convert to a digital value
If dblNextAxis < 0 Then
iValue = -1
Else
iValue = 1
End If
' make sure this isn't already mapped
bHaveInput = TRUE
If iWhichInput > LBound(m_arrControlMap, 2) Then
' is input unique?
For iNextInput = LBound(m_arrControlMap, 2) To iWhichInput - 1
If m_arrControlMap(iPlayer, iNextInput).device = iDevice Then
If m_arrControlMap(iPlayer, iNextInput).typ = cInputAxis Then
If m_arrControlMap(iPlayer, iNextInput).code = iLoop Then
If m_arrControlMap(iPlayer, iNextInput).value = iValue Then
bHaveInput = FALSE
End If
End If
End If
End If
Next iNextInput
End If
If bHaveInput Then
m_arrControlMap(iPlayer, iWhichInput).device = iDevice
m_arrControlMap(iPlayer, iWhichInput).typ = cInputAxis
m_arrControlMap(iPlayer, iWhichInput).code = iLoop
m_arrControlMap(iPlayer, iWhichInput).value = iValue
bMoveNext = TRUE
End If
End If
End If
Next iLoop
End If
Wend ' clear and update the device buffer
Next iController
' END CHECK FOR CONTROLLER INPUT
' -----------------------------------------------------------------------------
' -----------------------------------------------------------------------------
' BEGIN CHECK FOR KEYBOARD INPUT #1
If bMoveNext = FALSE Then
'_KEYCLEAR: _DELAY 1
While _DeviceInput(1): Wend ' clear and update the keyboard buffer
' Detect changed key state
For iLoop = LBound(m_arrButtonCode) To UBound(m_arrButtonCode)
iCode = m_arrButtonCode(iLoop)
If _Button(iCode) <> FALSE Then
' *****************************************************************************
' PRESSED KEYBOARD
'PRINT "PRESSED " + m_arrButtonKey(iLoop)
' make sure this isn't already mapped
bHaveInput = TRUE
If iWhichInput > LBound(m_arrControlMap, 2) Then
' is input unique?
For iNextInput = LBound(m_arrControlMap, 2) To iWhichInput - 1
If m_arrControlMap(iPlayer, iNextInput).device = 1 Then ' .device 1 = keyboard
If m_arrControlMap(iPlayer, iNextInput).typ = cInputKey Then
If m_arrControlMap(iPlayer, iNextInput).code = iCode Then
'if m_arrControlMap(iPlayer, iNextInput).value = TRUE then
bHaveInput = FALSE
'end if
End If
End If
End If
Next iNextInput
End If
If bHaveInput Then
m_arrControlMap(iPlayer, iWhichInput).device = 1 ' .device 1 = keyboard
m_arrControlMap(iPlayer, iWhichInput).typ = cInputKey
m_arrControlMap(iPlayer, iWhichInput).code = iCode
m_arrControlMap(iPlayer, iWhichInput).value = TRUE
bMoveNext = TRUE
End If
End If
Next iLoop
End If
' END CHECK FOR KEYBOARD INPUT #1
' -----------------------------------------------------------------------------
If bMoveNext = TRUE Then Exit Do
_Limit 30
Loop Until _KeyHit = 27 ' ESCAPE to quit
' END LOOK FOR NEXT INPUT
' =============================================================================
' Only ask user to select repeat if no override.
If m_bRepeatOverride = FALSE Then
Input "Enable repeat (y/n)"; in$: in$ = LCase$(_Trim$(in$))
If in$ = "y" Then
m_arrControlMap(iPlayer, iWhichInput).repeat = TRUE
Else
m_arrControlMap(iPlayer, iWhichInput).repeat = FALSE
End If
Else
m_arrControlMap(iPlayer, iWhichInput).repeat = GetGlobalInputRepeatSetting%(iWhichInput)
End If
Else
Print "(Skipped)"
bCancel = TRUE
bFinished = TRUE
End If
If bFinished = TRUE Then Exit For
Next iWhichInput
End If
If Len(sError) = 0 Then
m_bHaveMapping = TRUE
Else
sResult = "ERRORS: " + sError
End If
_KeyClear: _Delay 1
MapInput1b$ = sResult
End Function ' MapInput1b$
' /////////////////////////////////////////////////////////////////////////////
' Receives which input contstant and returns a text description
Function InputToString$ (iWhich As Integer)
Select Case iWhich
Case cInputUp:
InputToString$ = "up"
Case cInputDown:
InputToString$ = "down"
Case cInputLeft:
InputToString$ = "left"
Case cInputRight:
InputToString$ = "right"
Case cInputButton1:
InputToString$ = "button #1"
Case cInputButton2:
InputToString$ = "button #2"
Case cInputButton3:
InputToString$ = "button #3"
Case cInputButton4:
InputToString$ = "button #4"
Case Else:
InputToString$ = "unknown"
End Select
End Function ' InputToString$
' /////////////////////////////////////////////////////////////////////////////
' Receives which input contstant and returns a text description
' that matches the TextFieldType and TextLabelType ".item" member.
Function InputToItem$ (iWhich As Integer)
Select Case iWhich
Case cInputUp:
InputToItem$ = "Up"
Case cInputDown:
InputToItem$ = "Down"
Case cInputLeft:
InputToItem$ = "Left"
Case cInputRight:
InputToItem$ = "Right"
Case cInputButton1:
InputToItem$ = "Button1"
Case cInputButton2:
InputToItem$ = "Button2"
Case cInputButton3:
InputToItem$ = "Button3"
Case cInputButton4:
InputToItem$ = "Button4"
Case Else:
InputToItem$ = ""
End Select
End Function ' InputToItem$
' /////////////////////////////////////////////////////////////////////////////
' Receives which input contstant and returns its global "repeat" setting
Function GetGlobalInputRepeatSetting% (iWhich As Integer)
Select Case iWhich
Case cInputUp:
GetGlobalInputRepeatSetting% = m_bRepeatUp
Case cInputDown:
GetGlobalInputRepeatSetting% = m_bRepeatDown
Case cInputLeft:
GetGlobalInputRepeatSetting% = m_bRepeatLeft
Case cInputRight:
GetGlobalInputRepeatSetting% = m_bRepeatRight
Case cInputButton1:
GetGlobalInputRepeatSetting% = m_bRepeatButton1
Case cInputButton2:
GetGlobalInputRepeatSetting% = m_bRepeatButton2
Case cInputButton3:
GetGlobalInputRepeatSetting% = m_bRepeatButton3
Case cInputButton4:
GetGlobalInputRepeatSetting% = m_bRepeatButton4
Case Else:
GetGlobalInputRepeatSetting% = FALSE
End Select
End Function ' GetGlobalInputRepeatSetting%
Function InputTypeToString$ (iCode As Integer)
Select Case iCode
Case cInputNone:
InputTypeToString$ = "none"
Case cInputKey:
InputTypeToString$ = "key"
Case cInputButton:
InputTypeToString$ = "button"
Case cInputAxis:
InputTypeToString$ = "axis"
Case Else:
InputTypeToString$ = "unknown"
End Select
End Function ' InputTypeToString$
Function GetKeyboardButtonCodeText$ (iCode As Integer)
Dim sResult As String: sResult = ""
If LBound(m_arrButtonKeyDesc) <= iCode Then
If UBound(m_arrButtonKeyDesc) >= iCode Then
sResult = m_arrButtonKeyDesc(iCode)
End If
End If
If Len(sResult) = 0 Then
sResult = _Trim$(Str$(iCode)) + " (?)"
End If
GetKeyboardButtonCodeText$ = sResult
End Function ' GetKeyboardButtonCodeText$
Function GetKeyboardButtonCodeShortText$ (iCode As Integer)
Dim sResult As String: sResult = ""
If LBound(m_arrButtonKeyDesc) <= iCode Then
If UBound(m_arrButtonKeyDesc) >= iCode Then
sResult = m_arrButtonKeyDesc(iCode)
End If
End If
If Len(sResult) = 0 Then
sResult = _Trim$(Str$(iCode)) + " (?)"
End If
GetKeyboardButtonCodeShortText$ = sResult
End Function ' GetKeyboardButtonCodeText$
' /////////////////////////////////////////////////////////////////////////////
' METHOD v2
' Faster lookup - a dictionary with a hash lookup would be best
' but this is a quick way to do it since the values never change.
' The following shared arrays must be declared:
' ReDim Shared m_arrButtonCode(1 To 99) As Long
' ReDim Shared m_arrButtonKey(1 To 99) As String
' ReDim Shared m_arrButtonKeyDesc(0 To 512) As String
' ReDim Shared m_arrButtonKeyShortDesc(0 To 512) As String
Sub InitKeyboardButtonCodes ()
Dim iLoop As Integer
m_bInitialized = TRUE
End If
End Sub ' InitKeyboardButtonCodes
' /////////////////////////////////////////////////////////////////////////////
' not sure if this works
' Returns TRUE if the F10 key is held down.
' We use _KEYDOWN for this because _BUTTON doesn't detect F10.
' Constant must be declared globally:
' Const c_iKeyDown_F10 = 17408
Function KeydownF10%
Dim iCode As Long
'_KEYCLEAR: _DELAY 1
If _KeyDown(c_iKeyDown_F10) = TRUE Then
KeydownF10% = TRUE
Else
KeydownF10% = FALSE
End If
'_KEYCLEAR
End Function ' KeydownF10%
' /////////////////////////////////////////////////////////////////////////////
' not sure if this works
' Returns TRUE if the left ALT key is held down.
' We use _KEYHIT for this because _BUTTON doesn't detect ALT.
' Constant must be declared globally:
' Const c_iKeyHit_AltLeft = -30764
Function KeyhitAltLeft%
'_KEYCLEAR: _DELAY 1
If _KeyHit = c_iKeyHit_AltLeft Then
KeyhitAltLeft% = TRUE
Else
KeyhitAltLeft% = FALSE
End If
'_KEYCLEAR
End Function ' KeyhitAltLeft%
' /////////////////////////////////////////////////////////////////////////////
' not sure if this works
' Returns TRUE if the right ALT key is held down.
' We use _KEYHIT for this because _BUTTON doesn't detect ALT.
' Constant must be declared globally:
' Const c_iKeyHit_AltRight = -30765
Function KeyhitAltRight%
'_KEYCLEAR: _DELAY 1
If _KeyHit = c_iKeyHit_AltRight Then
KeyhitAltRight% = TRUE
Else
KeyhitAltRight% = FALSE
End If
'_KEYCLEAR
End Function ' KeyhitAltRight%
' /////////////////////////////////////////////////////////////////////////////
' DEVICES Button
' _LASTBUTTON(1) keyboards will normally return 512 buttons. One button is read per loop through all numbers.
' _BUTTONCHANGE(number) returns -1 when pressed, 1 when released and 0 when there is no event since the last read.
' _BUTTON(number) returns -1 when a button is pressed and 0 when released
' Detects most keys (where the codes are documented?)
' However, does not seem to detect:
' Key Use
' --- ---
' F10 Function KeydownF10%
' Left Alt Function KeyhitAltLeft%
' Right Alt Function KeyhitAltRight%
' Print Screen (system API call?)
' Pause/Break (system API call?)
Function KeyPressed% (iCode As Integer)
'_KEYCLEAR: _DELAY 1
While _DeviceInput(1): Wend ' clear and update the keyboard buffer
If _Button(iCode) <> FALSE Then
KeyPressed% = TRUE
Else
KeyPressed% = FALSE
End If
'_KEYCLEAR
End Function ' KeyPressed%
Function TestJoysticks1$
Dim RoutineName As String: RoutineName = "TestJoysticks1$"
Dim iDeviceCount As Integer
Dim sResult As String
' 1 is the keyboard
' 2 is the mouse
' 3 is the joystick
' unless someone has a strange setup with multiple mice/keyboards/ect...
' In that case, you can use _DEVICE$(i) to look for "KEYBOARD", "MOUSE", "JOYSTICK", if necessary.
' I've never actually found it necessary, but I figure it's worth mentioning, just in case...
iDeviceCount = _Devices ' Find the number of devices on someone's system
If iDeviceCount > 2 Then
TestJoysticks1b
sResult = ""
Else
sResult = "No joysticks found."
End If
_KeyClear
TestJoysticks1$ = sResult
End Function ' TestJoysticks1$
' /////////////////////////////////////////////////////////////////////////////
' Reads controllers and displays values on screen.
' Currently this is set up to support up to 8 joysticks,
' with upto 4 buttons and 2 axes each
' Testing with an old USB Logitech RumblePad 2
' and Atari 2600 joysticks plugged into using
' iCode Atari Joystick, Paddle, Driving to USB Adapter 4 ports
' BASED ON CODE BY SMcNeill FROM:
' Simple Joystick Detection and Interaction (Read 316 times)
' https://www.qb64.org/forum/index.php?topic=2160.msg129051#msg129051
' https://forum.qb64.org/index.php?topic=2160.msg129083#msg129083
Sub TestJoysticks1b ()
Dim RoutineName As String:: RoutineName = "TestJoysticks1b"
Dim in$
Dim iDeviceCount As Integer
Dim iDevice As Integer
Dim arrButton(32, 16) As Integer ' number of buttons on the joystick
Dim arrButtonMin(32, 16) As Integer ' stores the minimum value read
Dim arrButtonMax(32, 16) As Integer ' stores the maximum value read
Dim arrAxis(32, 16) As Double ' number of axis on the joystick
Dim arrAxisMin(32, 16) As Double ' stores the minimum value read
Dim arrAxisMax(32, 16) As Double ' stores the maximum value read
Dim arrAxisAvg(32, 16) As Double ' stores the average value read in the last few measurements
Dim arrButtonNew(32, 16) As Integer ' tracks when to initialize values
Dim arrAxisNew(32, 16) As Integer ' tracks when to initialize values
Dim arrController(8) As ControllerType ' holds info for each player
Dim iNumControllers As Integer
Dim iController As Integer
Dim iNextY As Integer
Dim iNextX As Integer
Dim iNextC As Integer
Dim iLoop As Integer
Dim iDigits As Integer ' # digits to display (values are truncated to this length)
Dim strValue As String
Dim strAxis As String
Dim dblNextAxis As Double
'DIM iMeasureCount AS INTEGER
Dim dblAverage As Double
Dim sngAverage As Single
Dim sLine As String
Dim iX As Integer
Dim iY As Integer
Dim iCol As Integer
Dim iRow As Integer
Dim iCols As Integer
Dim iRows As Integer
Dim iColWidth As Integer
Dim iColCount As Integer
Dim iGroupCount As Integer
' SET UP SCREEN
Screen _NewImage(1280, 1024, 32): _ScreenMove 0, 0
' COUNT # OF JOYSTICKS
' D= _DEVICES ' MUST be read in order for other 2 device functions to work!
iDeviceCount = _Devices ' Find the number of devices on someone's system
If iDeviceCount < 3 Then
Cls
Print "NO JOYSTICKS FOUND, EXITING..."
Input "PRESS <ENTER>"; in$
Exit Sub
End If
' BASE # OF PLAYERS ON HOW MANY CONTROLLERS FOUND
iNumControllers = iDeviceCount - 2 ' TODO: find out the right way to count joysticks
If iNumControllers > cMaxControllers Then
iNumControllers = cMaxControllers
End If
' INITIALIZE PLAYER COORDINATES AND SCREEN CHARACTERS
iNextY = 1
iNextX = -3
iNextC = 64
For iController = 1 To iNumControllers
iNextX = iNextX + 4
If iNextX > 80 Then
iNextX = 1
iNextY = iNextY + 4
End If
iNextC = iNextC + 1
arrController(iController).buttonCount = cMaxButtons
arrController(iController).axisCount = cMaxAxis
For iLoop = 1 To cMaxButtons
arrButtonNew(iController, iLoop) = TRUE
Next iLoop
For iLoop = 1 To cMaxAxis
arrAxisNew(iController, iLoop) = TRUE
arrAxisAvg(iController, iLoop) = 0
Next iLoop
Next iController
' CLEAR THE SCREEN
'iMeasureCount = 0
Do
For iController = 1 To iNumControllers
iDevice = iController + 2
While _DeviceInput(iDevice) ' clear and update the device buffer
''IF _DEVICEINPUT = 3 THEN ' this says we only care about joystick input values
' check all the buttons
For iLoop = 1 To _LastButton(iDevice)
If (iLoop > cMaxButtons) Then
Exit For
End If
arrController(iController).buttonCount = iLoop
' update button array to indicate if a button is up or down currently.
If _ButtonChange(iLoop) Then
'' _BUTTON(number) returns -1 when a button is pressed and 0 when released.
''arrButton(iLoop) = NOT arrButton(iLoop)
arrButton(iController, iLoop) = _Button(iLoop)
End If
'' SAVE MINIMUM VALUE
'if arrButton(iController, iLoop) < arrButtonMin(iController, iLoop) then
' arrButtonMin(iController, iLoop) = arrButton(iController, iLoop)
'
' ' INITIALIZE THE MAX TO THE MINIMUM VALUE
' IF arrButtonNew(iController, iLoop) = TRUE THEN
' arrButtonMax(iController, iLoop) = arrButtonMin(iController, iLoop)
' arrButtonNew(iController, iLoop) = FALSE
' END IF
'end if
'
'' SAVE MAXIMUM VALUE
'if arrButton(iController, iLoop) > arrButtonMax(iController, iLoop) then
' arrButtonMax(iController, iLoop) = arrButton(iController, iLoop)
'end if
Next iLoop
For iLoop = 1 To _LastAxis(iDevice) ' this loop checks all my axis
If (iLoop > cMaxAxis) Then
Exit For
End If
arrController(iController).axisCount = iLoop
' I like to give a little "jiggle" resistance to my controls, as I have an old joystick
' which is prone to always give minute values and never really center on true 0.
' A value of 1 means my axis is pushed fully in one direction.
' A value greater than 0.1 means it's been partially pushed in a direction (such as at a 45 degree diagional angle).
' A value of less than 0.1 means we count it as being centered. (As if it was 0.)
'IF ABS(_AXIS(iLoop)) <= 1 AND ABS(_AXIS(iLoop)) >= .1 THEN
dblNextAxis = _Axis(iLoop)
dblNextAxis = RoundUpDouble#(dblNextAxis, 3)
'IF ABS(dblNextAxis) <= 1 AND ABS(dblNextAxis) >= .01 THEN
If Abs(dblNextAxis) <= 1 And Abs(dblNextAxis) >= .001 Then
arrAxis(iController, iLoop) = dblNextAxis
Else
arrAxis(iController, iLoop) = 0
End If
'' SAVE MINIMUM VALUE
'if arrAxis(iController, iLoop) < arrAxisMin(iController, iLoop) then
' arrAxisMin(iController, iLoop) = arrAxis(iController, iLoop)
'
' ' INITIALIZE THE MAX TO THE MINIMUM VALUE
' IF arrAxisNew(iController, iLoop) = TRUE THEN
' arrAxisMax(iController, iLoop) = arrAxisMin(iController, iLoop)
' arrAxisNew(iController, iLoop) = FALSE
' END IF
'end if
'
'' SAVE MAXIMUM VALUE
'if arrAxis(iController, iLoop) > arrAxisMax(iController, iLoop) then
' arrAxisMax(iController, iLoop) = arrAxis(iController, iLoop)
'end if
'
'' ADD CURRENT VALUE TO AVERAGE SUM
'arrAxisAvg(iController, iLoop) = arrAxisAvg(iController, iLoop) + arrAxis(iController, iLoop)
Next iLoop
Wend ' clear and update the device buffer
' And below here is just the simple display routine which displays our values.
' If this was for a game, I'd choose something like arrAxis(1) = -1 for a left arrow style input,
' arrAxis(1) = 1 for a right arrow style input, rather than just using _KEYHIT or INKEY$.
Cls
PrintStringCR1 1, 1, "Game controller test program."
PrintStringCR1 1, 2, "This program is free to use and distribute per GNU GPLv3 license."
PrintStringCR1 1, 3, "Tests up to 4 controllers with 2 axes / 2 buttons each."
PrintStringCR1 1, 4, "Plug in controllers and move them & press buttons."
PrintStringCR1 1, 5, "-------------------------------------------------------------------------------"
iGroupCount = 0
For iController = 1 To iNumControllers
For iLoop = 1 To arrController(iController).axisCount ' A loop for each axis
strAxis = Right$(" " + cstr$(iLoop), 2)
sLine = ""
' display their status to the screen
sLine = sLine + "Player " + cstr$(iController)
_Limit 30
Loop Until _KeyHit = 27 ' ESCAPE to quit
' RETURN TO TEXT SCREEN
Screen 0
End Sub ' TestJoysticks1b
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN FILE FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' /////////////////////////////////////////////////////////////////////////////
' File format is comma-delimited
' containing controller info for one action per line
' where each line contains the following in this order:
' TAB ORDER INFO TYPE DESCRIPTION
' 1 {player #} Integer player # this mapping is for
' 2 {which action} Integer which action this mapping is for (up/down/right/left/button 1/button 2, etc.)
' 3 {device #} Integer number of the device this is mapped to
' 4 {type} Integer type of input (one of: cInputKey, cInputButton, cInputAxis)
' 5 {code} Integer if button the _BUTTON #, if axis the _AXIS #, if keyboard the _BUTTON #
' 6 {value} Integer if axis, the value (-1 or 1), else can be ignored
' 7 {repeat} Integer if TRUE, and repeating keys not controlled by global values (when m_bRepeatOverride=TRUE), controls repeating keys for this control
' These need to be declared globally and populated:
' ReDim Shared m_arrControlMap(1 To 8, 1 To 8) As ControlInputType
' Dim Shared m_ControlMapFileName$: m_ControlMapFileName$ = Left$(m_ProgramName$, _InStrRev(m_ProgramName$, ".")) + "map.txt"
' Dim Shared m_bRepeatOverride As Integer
' If there is an error, returns error message,
' else returns blank string.
Function SaveControllerMap1$
Dim RoutineName As String:: RoutineName = "SaveControllerMap1$"
Dim sResult As String: sResult = ""
Dim sError As String: sError = ""
Dim sFile As String
Dim in$
Dim iPlayer As Integer
Dim iWhichInput As Integer
Dim sLine As String
Dim iCount As Long: iCount = 0
'Dim iError As Long: iError = 0
Dim sDelim As String: sDelim = "," ' CHR$(9)
' Get file name
If Len(m_ControlMapFileName$) = 0 Then
m_ControlMapFileName$ = Left$(m_ProgramName$, _InStrRev(m_ProgramName$, ".")) + "map.txt"
End If
sFile = Mid$(m_ControlMapFileName$, _InStrRev(m_ControlMapFileName$, "\") + 1)
'_KeyClear
'Cls
'Print "SAVE CONTROLLER MAPPING:"
'Print "Default file name is " + Chr$(34) + m_ControlMapFileName$ + Chr$(34) + "."
'Input "Type save file name, or blank for default: ", in$
'in$ = _Trim$(in$)
'If Len(in$) > 0 Then
' m_ControlMapFileName$ = in$
'End If
'sFile = m_ProgramPath$ + m_ControlMapFileName$
' Save mapping to file
Open m_ControlMapFileName$ For Output As #1
For iPlayer = LBound(m_arrControlMap, 1) To UBound(m_arrControlMap, 1)
For iWhichInput = LBound(m_arrControlMap, 2) To UBound(m_arrControlMap, 2)
sLine = ""
Function LoadControllerMap1$
Dim RoutineName As String:: RoutineName = "LoadControllerMap1$"
Dim sResult As String: sResult = ""
Dim sError As String: sError = ""
Dim sNextErr As String
Dim sFile As String
Dim sText As String
Dim iTotal As Long: iTotal = 0
Dim iRead As Long: iRead = 0
Dim iValid As Long: iValid = 0
Dim iBad As Long: iBad = 0
Dim iBlank As Long: iBlank = 0
Dim sLine As String
ReDim arrNextLine(-1) As String
Dim iNumValues As Integer
Dim iAdjust As Integer
Dim iPlayer As Integer
Dim iWhichInput As Integer
Dim iDevice As Integer
Dim iType As Integer
Dim iCode As Integer
Dim iValue As Integer
Dim bRepeat As Integer
'Dim sDebugLine As String
' Get file name
If Len(sError) = 0 Then
If Len(m_ControlMapFileName$) = 0 Then
m_ControlMapFileName$ = Left$(m_ProgramName$, _InStrRev(m_ProgramName$, ".")) + "map.txt"
End If
sFile = Mid$(m_ControlMapFileName$, _InStrRev(m_ControlMapFileName$, "\") + 1)
End If
'' Get file name
'If Len(sError) = 0 Then
' Cls
' If Len(m_ControlMapFileName$) = 0 Then
' m_ControlMapFileName$ = Left$(m_ProgramName$, _InStrRev(m_ProgramName$, ".")) + "map.txt"
' End If
' Print "LOAD CONTROLLER MAPPING:"
' Print "Default file name is " + Chr$(34) + m_ControlMapFileName$ + Chr$(34) + "."
' Input "Type name of file to open, or blank for default: ", in$
' in$ = _Trim$(in$)
' If Len(in$) > 0 Then
' m_ControlMapFileName$ = in$
' End If
' sFile = m_ProgramPath$ + m_ControlMapFileName$
'End If
' Make sure file exists
If Len(sError) = 0 Then
If _FileExists(m_ControlMapFileName$) = FALSE Then
sError = "File not found: " + Chr$(34) + m_ControlMapFileName$ + Chr$(34)
Else
'DebugPrint "Found file: " + chr$(34) + m_ControlMapFileName$ + chr$(34)
End If
End If
' Read data from file
If Len(sError) = 0 Then
'DebugPrint "OPEN m_ControlMapFileName$ FOR BINARY AS #1"
Open m_ControlMapFileName$ For Binary As #1
sText = Space$(LOF(1))
Get #1, , sText
Close #1
iTotal = Len(sText) - Len(Replace$(sText, Chr$(13), ""))
sText = ""
Open m_ControlMapFileName$ For Input As #1
While Not EOF(1)
'INPUT #1, sLine
Line Input #1, sLine ' read entire text file line
iRead = iRead + 1
'DebugPrint "Parsing line " + _Trim$(Str$(iRead)) + _
' " of " + _Trim$(Str$(iTotal))
If Len(sLine) > 0 Then
split sLine, ",", arrNextLine()
'DebugPrint "split into arrNextLine()"
'DebugPrint " lbound =" + _Trim$(Str$(lbound(arrNextLine))) '+ CHR$(10)
'DebugPrint " ubound =" + _Trim$(Str$(ubound(arrNextLine))) '+ CHR$(10)
iNumValues = (UBound(arrNextLine) - LBound(arrNextLine)) + 1
If iNumValues > 5 Then
iAdjust = -1 '- lbound(arrNextLine)
If Len(sNextErr) = 0 Then
If IsNum%(arrNextLine(1 + iAdjust)) = TRUE Then
iPlayer = Val(arrNextLine(1 + iAdjust))
Else
sNextErr = "Error on line " + cstr$(iRead) + ", value 1: not a number"
End If
End If
If Len(sNextErr) = 0 Then
If IsNum%(arrNextLine(2 + iAdjust)) = TRUE Then
iWhichInput = Val(arrNextLine(2 + iAdjust))
Else
sNextErr = "Error on line " + cstr$(iRead) + ", value 2: not a number"
End If
End If
If Len(sNextErr) = 0 Then
If IsNum%(arrNextLine(3 + iAdjust)) = TRUE Then
iDevice = Val(arrNextLine(3 + iAdjust))
Else
sNextErr = "Error on line " + cstr$(iRead) + ", value 3: not a number"
End If
End If
If Len(sNextErr) = 0 Then
If IsNum%(arrNextLine(4 + iAdjust)) = TRUE Then
iType = Val(arrNextLine(4 + iAdjust))
Else
sNextErr = "Error on line " + cstr$(iRead) + ", value 4: not a number"
End If
End If
If Len(sNextErr) = 0 Then
If IsNum%(arrNextLine(5 + iAdjust)) = TRUE Then
iCode = Val(arrNextLine(5 + iAdjust))
Else
sNextErr = "Error on line " + cstr$(iRead) + ", value 5: not a number"
End If
End If
If Len(sNextErr) = 0 Then
If IsNum%(arrNextLine(6 + iAdjust)) = TRUE Then
iValue = Val(arrNextLine(6 + iAdjust))
Else
sNextErr = "Error on line " + cstr$(iRead) + ", value 6: not a number"
End If
End If
' validate iPlayer
If Len(sNextErr) = 0 Then
If iPlayer < LBound(m_arrControlMap, 1) Then
sNextErr = "Player value " + _Trim$(Str$(iPlayer)) + _
" is outside lbound(m_arrControlMap, 1) " + _
" which is " + _Trim$(Str$(lbound(m_arrControlMap, 1))) + "."
ElseIf iPlayer > UBound(m_arrControlMap, 1) Then
sNextErr = "Player value " + _Trim$(Str$(iPlayer)) + _
" is outside ubound(m_arrControlMap, 1) " + _
" which is " + _Trim$(Str$(ubound(m_arrControlMap, 1))) + "."
End If
End If
' validate iWhichInput
If Len(sNextErr) = 0 Then
If iWhichInput < LBound(m_arrControlMap, 2) Then
sNextErr = "WhichInput value " + _Trim$(Str$(iWhichInput)) + _
" is outside lbound(m_arrControlMap, 2) " + _
" which is " + _Trim$(Str$(lbound(m_arrControlMap, 2))) + "."
ElseIf iWhichInput > UBound(m_arrControlMap, 2) Then
sNextErr = "WhichInput value " + _Trim$(Str$(iWhichInput)) + _
" is outside ubound(m_arrControlMap, 2) " + _
" which is " + _Trim$(Str$(ubound(m_arrControlMap, 2))) + "."
End If
End If
' validate repeat setting
If iNumValues > 6 Then
If Len(sNextErr) = 0 Then
If IsNum%(arrNextLine(7 + iAdjust)) = TRUE Then
bRepeat = Val(arrNextLine(7 + iAdjust))
Else
sNextErr = "Error on line " + cstr$(iRead) + ", value 7: not a number"
End If
End If
Else
' get values from global
'if m_bRepeatOverride = TRUE then
bRepeat = GetGlobalInputRepeatSetting%(iWhichInput)
'end if
End If
Else
sNextErr = "Error on line " + cstr$(iRead) + ": detected " + cstr$(iNumValues) + " values, expected 6."
End If
If Len(sNextErr) = 0 Then
iValid = iValid + 1
m_arrControlMap(iPlayer, iWhichInput).device = iDevice
m_arrControlMap(iPlayer, iWhichInput).typ = iType
m_arrControlMap(iPlayer, iWhichInput).code = iCode
m_arrControlMap(iPlayer, iWhichInput).value = iValue
m_arrControlMap(iPlayer, iWhichInput).repeat = bRepeat
Else
iBad = iBad + 1
DebugPrint sNextErr
End If
Else
'DebugPrint " Line is blank: skipped"
iBlank = iBlank + 1
End If ' LEN(sLine) > 0
If Len(sError) = 0 Then
sResult = "Loaded mapping file " + Chr$(34) + sFile + Chr$(34) + "."
m_bHaveMapping = TRUE
Else
sResult = "ERRORS: " + sError
End If
LoadControllerMap1$ = sResult
End Function ' LoadControllerMap1$
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END FILE FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN GRAPHIC PRINTING ROUTINES
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' /////////////////////////////////////////////////////////////////////////////
' A way to automatically print to columns.
Sub ColumnBreak ()
m_PrintRow = m_StartRow
m_PrintCol = m_PrintCol + 1
If m_PrintCol > m_NumColumns Then
'TODO: options for when we go past the last column (stop printing, wrap around)
End If
End Sub ' ColumnBreak
' /////////////////////////////////////////////////////////////////////////////
' A way to automatically print to columns.
Sub InitColumns (iNumColumns As Integer)
Dim iCols As Integer
Dim iRows As Integer
iCols = _Width(0) \ _FontWidth
iRows = _Height(0) \ _FontHeight
If iNumColumns < 1 Or iNumColumns > iCols Then
m_NumColumns = 1
Else
m_NumColumns = iNumColumns
End If
If m_StartRow < 1 Or m_StartRow > iRows Then
m_StartRow = 1
End If
If m_EndRow < m_StartRow Or m_EndRow > iRows Then
m_EndRow = iRows
End If
If m_StartCol < 1 Or m_StartCol > m_NumColumns Then
m_StartCol = 1
End If
If m_EndCol < m_StartCol Or m_EndCol > m_NumColumns Then
m_EndCol = m_NumColumns
End If
m_PrintRow = 1
m_PrintCol = 1
End Sub ' InitColumns
' /////////////////////////////////////////////////////////////////////////////
' A way to automatically print to columns.
' Depends on the following shared variables:
' Dim Shared m_NumColumns As Integer : m_NumColumns = 1
' Dim Shared m_PrintRow As Integer : m_PrintRow = 0
' Dim Shared m_PrintCol As Integer : m_PrintCol = 0
' Dim Shared m_StartRow As Integer : m_StartRow = 0
' Dim Shared m_EndRow As Integer : m_EndRow = 0
' Dim Shared m_StartCol As Integer : m_StartCol = 0
' Dim Shared m_EndCol As Integer : m_EndCol = 0
If m_NumColumns < 1 Or m_NumColumns > iCols Then
m_NumColumns = 1
End If
If m_StartRow < 1 Or m_StartRow > iRows Then
m_StartRow = 1
End If
If m_EndRow < m_StartRow Or m_EndRow > iRows Then
m_EndRow = iRows
End If
If m_StartCol < 1 Or m_StartCol > m_NumColumns Then
m_StartCol = 1
End If
If m_EndCol < m_StartCol Or m_EndCol > m_NumColumns Then
m_EndCol = m_NumColumns
End If
If m_PrintRow < m_StartRow Then
m_PrintRow = m_StartRow
End If
If m_PrintCol < m_StartCol Then
m_PrintCol = m_StartCol
End If
iColWidth = iCols \ m_NumColumns
If m_PrintRow <= m_EndRow Then
If m_PrintCol <= m_EndCol Then
split MyString, Chr$(13), arrLines()
For iRow = 0 To UBound(arrLines)
sLine = Left$(arrLines(iRow), iColWidth)
'TODO: wrap remaining text
iX = _FontWidth * ((m_PrintCol - 1) * iColWidth)
iY = _FontHeight * (m_PrintRow - 1)
_PrintString (iX, iY), sLine
m_PrintRow = m_PrintRow + 1
If m_PrintRow > m_EndRow Then
m_PrintRow = m_StartRow
m_PrintCol = m_PrintCol + 1
If m_PrintCol > m_NumColumns Then
'TODO: options for when we reach the bottom of the last column (stop printing, wrap around)
m_PrintCol = 1
End If
End If
Next iRow
End If
End If
End Sub ' PrintColumn
' /////////////////////////////////////////////////////////////////////////////
' Eliminates the math.
' Text resolution:
' 648 x 480: 80 x 30
' 720 x 480: 90 x 30
' 800 x 600: 100 x 37
' 1024 x 768: 128 x 48
' 1280 x 1024: 160 x 64
' 1920 x 1080: 240 x 67
' 2048 x 1152: 256 x 72 (truncated after 70 rows, 255 columns)
' 3840 x 2160: 480 x135 (truncated after 133 rows, 479 columns)
Sub PrintStringCR1 (iCol As Integer, iRow As Integer, MyString As String)
Dim iCols As Integer
Dim iRows As Integer
Dim iX As Integer
Dim iY As Integer
iCols = _Width(0) \ _FontWidth
iRows = _Height(0) \ _FontHeight
iX = _FontWidth * (iCol - 1)
iY = _FontHeight * (iRow - 1)
_PrintString (iX, iY), MyString
End Sub ' PrintStringCR1
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END GRAPHIC PRINTING ROUTINES
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN DEBUGGING ROUTINES #DEBUGGING
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Sub DebugPrint (s$)
If m_bTesting = TRUE Then
_Echo s$
'ReDim arrLines$(0)
'dim delim$ : delim$ = Chr$(13)
'split MyString, delim$, arrLines$()
End If
End Sub ' DebugPrint
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END DEBUGGING ROUTINES @DEBUGGING
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN GENERAL PURPOSE ROUTINES #GEN
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' /////////////////////////////////////////////////////////////////////////////
' Convert a value to string and trim it (because normal Str$ adds spaces)
Function cstr$ (myValue)
'cstr$ = LTRIM$(RTRIM$(STR$(myValue)))
cstr$ = _Trim$(Str$(myValue))
End Function ' cstr$
' /////////////////////////////////////////////////////////////////////////////
' Convert a Long value to string and trim it (because normal Str$ adds spaces)
Function cstrl$ (myValue As Long)
cstrl$ = _Trim$(Str$(myValue))
End Function ' cstrl$
' /////////////////////////////////////////////////////////////////////////////
' Convert a Single value to string and trim it (because normal Str$ adds spaces)
Function cstrs$ (myValue As Single)
''cstr$ = LTRIM$(RTRIM$(STR$(myValue)))
cstrs$ = _Trim$(Str$(myValue))
End Function ' cstrs$
' /////////////////////////////////////////////////////////////////////////////
' Convert an unsigned Long value to string and trim it (because normal Str$ adds spaces)
Function cstrul$ (myValue As _Unsigned Long)
cstrul$ = _Trim$(Str$(myValue))
End Function ' cstrul$
' /////////////////////////////////////////////////////////////////////////////
' Scientific notation - QB64 Wiki
' https://www.qb64.org/wiki/Scientific_notation
' Example: A string function that displays extremely small or large exponential decimal values.
Function DblToStr$ (n#)
value$ = UCase$(LTrim$(Str$(n#)))
Xpos% = InStr(value$, "D") + InStr(value$, "E") 'only D or E can be present
If Xpos% Then
expo% = Val(Mid$(value$, Xpos% + 1))
If Val(value$) < 0 Then
sign$ = "-": valu$ = Mid$(value$, 2, Xpos% - 2)
Else valu$ = Mid$(value$, 1, Xpos% - 1)
End If
dot% = InStr(valu$, "."): L% = Len(valu$)
If expo% > 0 Then add$ = String$(expo% - (L% - dot%), "0")
If expo% < 0 Then min$ = String$(Abs(expo%) - (dot% - 1), "0"): DP$ = "."
For n = 1 To L%
If Mid$(valu$, n, 1) <> "." Then num$ = num$ + Mid$(valu$, n, 1)
Next
Else DblToStr$ = value$: Exit Function
End If
DblToStr$ = _Trim$(sign$ + DP$ + min$ + num$ + add$)
End Function ' DblToStr$
Function FormatNumber$ (myValue, iDigits As Integer)
Dim strValue As String
strValue = DblToStr$(myValue) + String$(iDigits, " ")
If myValue < 1 Then
If myValue < 0 Then
strValue = Replace$(strValue, "-.", "-0.")
ElseIf myValue > 0 Then
strValue = "0" + strValue
End If
End If
FormatNumber$ = Left$(strValue, iDigits)
End Function ' FormatNumber$
' /////////////////////////////////////////////////////////////////////////////
' From: Bitwise Manipulations By Steven Roman
' http://www.romanpress.com/Articles/Bitwise_R/Bitwise.htm
' Returns the 8-bit binary representation
' of an integer iInput where 0 <= iInput <= 255
Function GetBinary$ (iInput1 As Integer)
Dim sResult As String
Dim iLoop As Integer
Dim iInput As Integer: iInput = iInput1
sResult = ""
If iInput >= 0 And iInput <= 255 Then
For iLoop = 1 To 8
sResult = LTrim$(RTrim$(Str$(iInput Mod 2))) + sResult
iInput = iInput \ 2
'If iLoop = 4 Then sResult = " " + sResult
Next iLoop
End If
GetBinary$ = sResult
End Function ' GetBinary$
' /////////////////////////////////////////////////////////////////////////////
' wonderfully inefficient way to read if a bit is set
' ival = GetBit256%(int we are comparing, int containing the bits we want to read)
' See also: GetBit256%, SetBit256%
Function GetBit256% (iNum1 As Integer, iBit1 As Integer)
Dim iResult As Integer
Dim sNum As String
Dim sBit As String
Dim iLoop As Integer
Dim bContinue As Integer
'DIM iTemp AS INTEGER
Dim iNum As Integer: iNum = iNum1
Dim iBit As Integer: iBit = iBit1
iResult = FALSE
bContinue = TRUE
If iNum < 256 And iBit <= 128 Then
sNum = GetBinary$(iNum)
sBit = GetBinary$(iBit)
For iLoop = 1 To 8
If Mid$(sBit, iLoop, 1) = "1" Then
'if any of the bits in iBit are false, return false
If Mid$(sNum, iLoop, 1) = "0" Then
iResult = FALSE
bContinue = FALSE
Exit For
End If
End If
Next iLoop
If bContinue = TRUE Then
iResult = TRUE
End If
End If
GetBit256% = iResult
End Function ' GetBit256%
' /////////////////////////////////////////////////////////////////////////////
' From: Bitwise Manipulations By Steven Roman
' http://www.romanpress.com/Articles/Bitwise_R/Bitwise.htm
' Returns the integer that corresponds to a binary string of length 8
Function GetIntegerFromBinary% (sBinary1 As String)
Dim iResult As Integer
Dim iLoop As Integer
Dim strBinary As String
Dim sBinary As String: sBinary = sBinary1
iResult = 0
strBinary = Replace$(sBinary, " ", "") ' Remove any spaces
For iLoop = 0 To Len(strBinary) - 1
iResult = iResult + 2 ^ iLoop * Val(Mid$(strBinary, Len(strBinary) - iLoop, 1))
Next iLoop
GetIntegerFromBinary% = iResult
End Function ' GetIntegerFromBinary%
Function IsOdd% (n)
If n Mod 2 = 1 Then
IsOdd% = TRUE
Else
IsOdd% = FALSE
End If
End Function ' IsOdd%
' /////////////////////////////////////////////////////////////////////////////
' By sMcNeill from https://www.qb64.org/forum/index.php?topic=896.0
Function IsNum% (text$)
Dim a$
Dim b$
a$ = _Trim$(text$)
b$ = _Trim$(Str$(Val(text$)))
If a$ = b$ Then
IsNum% = TRUE
Else
IsNum% = FALSE
End If
End Function ' IsNum%
' /////////////////////////////////////////////////////////////////////////////
' Re: Does a Is Number function exist in QB64?
' https://www.qb64.org/forum/index.php?topic=896.15
' MWheatley
' « Reply #18 on: January 01, 2019, 11:24:30 AM »
' returns 1 if string is an integer, 0 if not
Function IsNumber (text$)
Dim i As Integer
IsNumber = 1
For i = 1 To Len(text$)
If Asc(Mid$(text$, i, 1)) < 45 Or Asc(Mid$(text$, i, 1)) >= 58 Then
IsNumber = 0
Exit For
ElseIf Asc(Mid$(text$, i, 1)) = 47 Then
IsNumber = 0
Exit For
End If
Next i
End Function ' IsNumber
' /////////////////////////////////////////////////////////////////////////////
' Split and join strings
' https://www.qb64.org/forum/index.php?topic=1073.0
'Combine all elements of in$() into a single string with delimiter$ separating the elements.
Function join$ (in$(), delimiter$)
Dim result$
Dim i As Long
result$ = in$(LBound(in$))
For i = LBound(in$) + 1 To UBound(in$)
result$ = result$ + delimiter$ + in$(i)
Next i
join$ = result$
End Function ' join$
' /////////////////////////////////////////////////////////////////////////////
' ABS was returning strange values with type LONG
' so I created this which does not.
Function LongABS& (lngValue As Long)
If Sgn(lngValue) = -1 Then
LongABS& = 0 - lngValue
Else
LongABS& = lngValue
End If
End Function ' LongABS&
' /////////////////////////////////////////////////////////////////////////////
' Writes sText to a debug file in the EXE folder.
' Debug file is named the same thing as the program EXE name with ".txt" at the end.
' For example the program "C:\QB64\MyProgram.BAS" running as
' "C:\QB64\MyProgram.EXE" would have an output file "C:\QB64\MyProgram.EXE.txt".
' If the file doesn't exist, it is created, otherwise it is appended to.
Sub DebugPrintFile (sText As String)
Dim sFileName As String
Dim sError As String
Dim sOut As String
sFileName = ProgramPath$ + ProgramName$ + ".txt"
sError = ""
If _FileExists(sFileName) = FALSE Then
sOut = ""
sOut = sOut + "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++" + Chr$(13) + Chr$(10)
sOut = sOut + "PROGRAM : " + ProgramName$ + Chr$(13) + Chr$(10)
sOut = sOut + "RUN DATE: " + CurrentDateTime$ + Chr$(13) + Chr$(10)
sOut = sOut + "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++" + Chr$(13) + Chr$(10)
sError = PrintFile$(sFileName, sOut, FALSE)
End If
If Len(sError) = 0 Then
sError = PrintFile$(sFileName, sText, TRUE)
End If
If Len(sError) <> 0 Then
Print CurrentDateTime$ + " DebugPrintFile FAILED: " + sError
End If
End Sub ' DebugPrintFile
Function IntPadRight$ (iValue As Integer, iWidth As Integer)
IntPadRight$ = Left$(_Trim$(Str$(iValue)) + String$(iWidth, " "), iWidth)
End Function ' IntPadRight$
Function PrintFile$ (sFileName As String, sText As String, bAppend As Integer)
'x = 1: y = 2: z$ = "Three"
Dim sError As String: sError = ""
If Len(sError) = 0 Then
If (bAppend = TRUE) Then
If _FileExists(sFileName) Then
Open sFileName For Append As #1 ' opens an existing file for appending
Else
sError = "Error in PrintFile$ : File not found. Cannot append."
End If
Else
Open sFileName For Output As #1 ' opens and clears an existing file or creates new empty file
End If
End If
If Len(sError) = 0 Then
' WRITE places text in quotes in the file
'WRITE #1, x, y, z$
'WRITE #1, sText
' PRINT does not put text inside quotes
Print #1, sText
Close #1
'PRINT "File created with data. Press a key!"
'K$ = INPUT$(1) 'press a key
'OPEN sFileName FOR INPUT AS #2 ' opens a file to read it
'INPUT #2, a, b, c$
'CLOSE #2
'PRINT a, b, c$
'WRITE a, b, c$
End If
PrintFile$ = sError
End Function ' PrintFile$
' /////////////////////////////////////////////////////////////////////////////
' Does a _PrintString at the specified row+column.
' iRow and iCol are 0-based.
Sub PrintString (iRow As Integer, iCol As Integer, MyString As String)
Dim iX As Integer
Dim iY As Integer
iX = _FontWidth * iCol
iY = _FontHeight * iRow ' (iRow + 1)
_PrintString (iX, iY), MyString
End Sub ' PrintString
' /////////////////////////////////////////////////////////////////////////////
' Does a _PrintString at the specified row+column.
' iRow and iCol are 1-based.
Sub PrintString1 (iRow As Integer, iCol As Integer, MyString As String)
Dim iX As Integer
Dim iY As Integer
iX = _FontWidth * (iCol - 1)
iY = _FontHeight * (iRow - 1)
_PrintString (iX, iY), MyString
End Sub ' PrintString1
' /////////////////////////////////////////////////////////////////////////////
' Does a _PrintString at the specified row+column,
' and saves value and color info to ScreenArray.
Sub PrintString2 (iRow As Integer, iCol As Integer, MyString As String, ScreenArray() As TextCellType)
Dim iX As Integer
Dim iY As Integer
Dim iLoop As Integer
If (iRow >= LBound(ScreenArray, 1)) Then
If (iRow <= UBound(ScreenArray, 1)) Then
If (iCol >= LBound(ScreenArray, 2)) Then
If (iCol <= UBound(ScreenArray, 2)) Then
iX = iCol
iY = iRow
For iLoop = 1 To Len(MyString)
If iX > UBound(ScreenArray, 2) Then
Exit For
Else
ScreenArray(iY, iX).value = Mid$(MyString, iLoop, 1)
ScreenArray(iY, iX).fgColor = _DefaultColor
ScreenArray(iY, iX).bgcolor = _BackgroundColor
iX = iX + 1
End If
Next iLoop
End If
End If
End If
End If
End Sub ' PrintString2
' /////////////////////////////////////////////////////////////////////////////
' Generate random value between Min and Max.
Function RandomNumber% (Min%, Max%)
Dim NumSpread%
' SET RANDOM SEED
'Randomize ' Initialize random-number generator.
Randomize Timer
' GET RANDOM # Min%-Max%
'RandomNumber = Int((Max * Rnd) + Min) ' generate number
NumSpread% = (Max% - Min%) + 1
RandomNumber% = Int(Rnd * NumSpread%) + Min% ' GET RANDOM # BETWEEN Max% AND Min%
Sub RandomNumberTest
Dim iCols As Integer: iCols = 10
Dim iRows As Integer: iRows = 20
Dim iLoop As Integer
Dim iX As Integer
Dim iY As Integer
Dim sError As String
Dim sFileName As String
Dim sText As String
Dim bAppend As Integer
Dim iMin As Integer
Dim iMax As Integer
Dim iNum As Integer
Dim iErrorCount As Integer
Dim sInput$
Input "Press <ENTER> to continue", sInput$
End Sub ' RandomNumberTest
' /////////////////////////////////////////////////////////////////////////////
' FROM: String Manipulation
' found at abandoned, outdated and now likely malicious qb64 dot net website
' http://www.qb64.[net]/forum/index_topic_5964-0/
'
'SUMMARY:
' Purpose: A library of custom functions that transform strings.
' Author: Dustinian Camburides (dustinian@gmail.com)
' Platform: QB64 (www.qb64.org)
' Revision: 1.6
' Updated: 5/28/2012
'SUMMARY:
'[Replace$] replaces all instances of the [Find] sub-string with the [Add] sub-string within the [Text] string.
'INPUT:
'Text: The input string; the text that's being manipulated.
'Find: The specified sub-string; the string sought within the [Text] string.
'Add: The sub-string that's being added to the [Text] string.
Function Replace$ (Text1 As String, Find1 As String, Add1 As String)
' VARIABLES:
Dim Text2 As String
Dim Find2 As String
Dim Add2 As String
Dim lngLocation As Long ' The address of the [Find] substring within the [Text] string.
Dim strBefore As String ' The characters before the string to be replaced.
Dim strAfter As String ' The characters after the string to be replaced.
' INITIALIZE:
' MAKE COPIESSO THE ORIGINAL IS NOT MODIFIED (LIKE ByVal IN VBA)
Text2 = Text1
Find2 = Find1
Add2 = Add1
lngLocation = InStr(1, Text2, Find2)
' PROCESSING:
' While [Find2] appears in [Text2]...
While lngLocation
' Extract all Text2 before the [Find2] substring:
strBefore = Left$(Text2, lngLocation - 1)
' Extract all text after the [Find2] substring:
strAfter = Right$(Text2, ((Len(Text2) - (lngLocation + Len(Find2) - 1))))
Print "ReplaceTest finished."
End Sub ' ReplaceTest
' /////////////////////////////////////////////////////////////////////////////
' https://www.qb64.org/forum/index.php?topic=3605.0
' Quote from: SMcNeill on Today at 03:53:48 PM
'
' Sometimes, you guys make things entirely too complicated.
' There ya go! Three functions to either round naturally,
' always round down, or always round up, to whatever number of digits you desire.
' EDIT: Modified to add another option to round scientific,
' since you had it's description included in your example.
Function Round## (num##, digits%)
Round## = Int(num## * 10 ^ digits% + .5) / 10 ^ digits%
End Function
Function RoundUp## (num##, digits%)
RoundUp## = _Ceil(num## * 10 ^ digits%) / 10 ^ digits%
End Function
Function RoundDown## (num##, digits%)
RoundDown## = Int(num## * 10 ^ digits%) / 10 ^ digits%
End Function
Function Round_Scientific## (num##, digits%)
Round_Scientific## = _Round(num## * 10 ^ digits%) / 10 ^ digits%
End Function
Function RoundUpDouble# (num#, digits%)
RoundUpDouble# = _Ceil(num# * 10 ^ digits%) / 10 ^ digits%
End Function
Function RoundUpSingle! (num!, digits%)
RoundUpSingle! = _Ceil(num! * 10 ^ digits%) / 10 ^ digits%
End Function
' /////////////////////////////////////////////////////////////////////////////
' fantastically inefficient way to set a bit
' example use: arrMaze(iX, iY) = SetBit256%(arrMaze(iX, iY), cS, FALSE)
' See also: GetBit256%, SetBit256%
' newint=SetBit256%(oldint, int containing the bits we want to set, value to set them to)
Function SetBit256% (iNum1 As Integer, iBit1 As Integer, bVal1 As Integer)
Dim sNum As String
Dim sBit As String
Dim sVal As String
Dim iLoop As Integer
Dim strResult As String
Dim iResult As Integer
Dim iNum As Integer: iNum = iNum1
Dim iBit As Integer: iBit = iBit1
Dim bVal As Integer: bVal = bVal1
If iNum < 256 And iBit <= 128 Then
sNum = GetBinary$(iNum)
sBit = GetBinary$(iBit)
If bVal = TRUE Then
sVal = "1"
Else
sVal = "0"
End If
strResult = ""
For iLoop = 1 To 8
If Mid$(sBit, iLoop, 1) = "1" Then
strResult = strResult + sVal
Else
strResult = strResult + Mid$(sNum, iLoop, 1)
End If
Next iLoop
iResult = GetIntegerFromBinary%(strResult)
Else
iResult = iNum
End If
SetBit256% = iResult
End Function ' SetBit256%
' /////////////////////////////////////////////////////////////////////////////
' Scientific notation - QB64 Wiki
' https://www.qb64.org/wiki/Scientific_notation
' Example: A string function that displays extremely small or large exponential decimal values.
Function SngToStr$ (n!)
value$ = UCase$(LTrim$(Str$(n!)))
Xpos% = InStr(value$, "D") + InStr(value$, "E") 'only D or E can be present
If Xpos% Then
expo% = Val(Mid$(value$, Xpos% + 1))
If Val(value$) < 0 Then
sign$ = "-": valu$ = Mid$(value$, 2, Xpos% - 2)
Else valu$ = Mid$(value$, 1, Xpos% - 1)
End If
dot% = InStr(valu$, "."): L% = Len(valu$)
If expo% > 0 Then add$ = String$(expo% - (L% - dot%), "0")
If expo% < 0 Then min$ = String$(Abs(expo%) - (dot% - 1), "0"): DP$ = "."
For n = 1 To L%
If Mid$(valu$, n, 1) <> "." Then num$ = num$ + Mid$(valu$, n, 1)
Next
Else SngToStr$ = value$: Exit Function
End If
SngToStr$ = _Trim$(sign$ + DP$ + min$ + num$ + add$)
End Function ' SngToStr$
' /////////////////////////////////////////////////////////////////////////////
' Split and join strings
' https://www.qb64.org/forum/index.php?topic=1073.0
'
' FROM luke, QB64 Developer
' Date: February 15, 2019, 04:11:07 AM »
'
' Given a string of words separated by spaces (or any other character),
' splits it into an array of the words. I've no doubt many people have
' written a version of this over the years and no doubt there's a million
' ways to do it, but I thought I'd put mine here so we have at least one
' version. There's also a join function that does the opposite
' array -> single string.
'
' Code is hopefully reasonably self explanatory with comments and a little demo.
' Note, this is akin to Python/JavaScript split/join, PHP explode/implode.
'Split in$ into pieces, chopping at every occurrence of delimiter$. Multiple consecutive occurrences
'of delimiter$ are treated as a single instance. The chopped pieces are stored in result$().
'
'delimiter$ must be one character long.
'result$() must have been REDIMmed previously.
' Modified to handle multi-character delimiters
Sub split (in$, delimiter$, result$())
Dim start As Integer
Dim finish As Integer
Dim iDelimLen As Integer
ReDim result$(-1)
iDelimLen = Len(delimiter$)
start = 1
Do
'While Mid$(in$, start, 1) = delimiter$
While Mid$(in$, start, iDelimLen) = delimiter$
'start = start + 1
start = start + iDelimLen
If start > Len(in$) Then
Exit Sub
End If
Wend
finish = InStr(start, in$, delimiter$)
If finish = 0 Then
finish = Len(in$) + 1
End If
ReDim _Preserve result$(0 To UBound(result$) + 1)
result$(UBound(result$)) = Mid$(in$, start, finish - start)
start = finish + 1
Loop While start <= Len(in$)
End Sub ' split
Function StrJustifyRight$ (sValue As String, iWidth As Integer)
StrJustifyRight$ = Right$(String$(iWidth, " ") + sValue, iWidth)
End Function ' StrJustifyRight$
Function StrJustifyLeft$ (sValue As String, iWidth As Integer)
StrJustifyLeft$ = Left$(sValue + String$(iWidth, " "), iWidth)
End Function ' StrJustifyLeft$
Function StrJustifyCenter$ (sValue As String, iWidth As Integer)
Dim iLen0 As Integer
Dim iLen1 As Integer
Dim iLen2 As Integer
Dim iExtra As Integer
iLen0 = Len(sValue)
If iWidth = iLen0 Then
' no extra space: return unchanged
StrJustifyCenter$ = sValue
ElseIf iWidth > iLen0 Then
If IsOdd%(iWidth) Then
iWidth = iWidth - 1
End If
' center
iExtra = iWidth - iLen0
iLen1 = iExtra \ 2
iLen2 = iLen1 + (iExtra Mod 2)
StrJustifyCenter$ = String$(iLen1, " ") + sValue + String$(iLen2, " ")
Else
' string is too long: truncate
StrJustifyCenter$ = Left$(sValue, iWidth)
End If
End Function ' StrJustifyCenter$
Function TrueFalse$ (myValue)
If myValue = TRUE Then
TrueFalse$ = "TRUE"
Else
TrueFalse$ = "FALSE"
End If
End Function ' TrueFalse$
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END GENERAL PURPOSE ROUTINES
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN COLOR ARRAY FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Sub AddColor (ColorValue As _Unsigned Long, ColorName As String, arrColor() As ColorType)
ReDim _Preserve arrColor(0 To UBound(arrColor) + 1) As ColorType
arrColor(UBound(arrColor)).name = ColorName
arrColor(UBound(arrColor)).value = ColorValue
End Sub ' AddColor
AddColor cEmpty, "cEmpty", arrColor()
End Sub ' AddColors
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END COLOR ARRAY FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN COLOR FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Function cRed~& ()
cRed = _RGB32(255, 0, 0)
End Function
Function cOrangeRed~& ()
cOrangeRed = _RGB32(255, 69, 0)
End Function ' cOrangeRed~&
Function cDarkOrange~& ()
cDarkOrange = _RGB32(255, 140, 0)
End Function ' cDarkOrange~&
Function cOrange~& ()
cOrange = _RGB32(255, 165, 0)
End Function ' cOrange~&
Function cGold~& ()
cGold = _RGB32(255, 215, 0)
End Function ' cGold~&
Function cYellow~& ()
cYellow = _RGB32(255, 255, 0)
End Function ' cYellow~&
' LONG-HAIRED FRIENDS OF JESUS OR NOT,
' THIS IS NOT YELLOW ENOUGH (TOO CLOSE TO LIME)
' TO USE FOR OUR COMPLEX RAINBOW SEQUENCE:
Function cChartreuse~& ()
cChartreuse = _RGB32(127, 255, 0)
End Function ' cChartreuse~&
' WE SUBSTITUTE THIS CSS3 COLOR FOR INBETWEEN LIME AND YELLOW:
Function cOliveDrab1~& ()
cOliveDrab1 = _RGB32(192, 255, 62)
End Function ' cOliveDrab1~&
Function cLime~& ()
cLime = _RGB32(0, 255, 0)
End Function ' cLime~&
Function cMediumSpringGreen~& ()
cMediumSpringGreen = _RGB32(0, 250, 154)
End Function ' cMediumSpringGreen~&
Function cCyan~& ()
cCyan = _RGB32(0, 255, 255)
End Function ' cCyan~&
Function cDeepSkyBlue~& ()
cDeepSkyBlue = _RGB32(0, 191, 255)
End Function ' cDeepSkyBlue~&
Function cDodgerBlue~& ()
cDodgerBlue = _RGB32(30, 144, 255)
End Function ' cDodgerBlue~&
Function cSeaBlue~& ()
cSeaBlue = _RGB32(0, 64, 255)
End Function ' cSeaBlue~&
Function cBlue~& ()
cBlue = _RGB32(0, 0, 255)
End Function ' cBlue~&
Function cBluePurple~& ()
cBluePurple = _RGB32(64, 0, 255)
End Function ' cBluePurple~&
Function cDeepPurple~& ()
cDeepPurple = _RGB32(96, 0, 255)
End Function ' cDeepPurple~&
Function cPurple~& ()
cPurple = _RGB32(128, 0, 255)
End Function ' cPurple~&
Function cPurpleRed~& ()
cPurpleRed = _RGB32(128, 0, 192)
End Function ' cPurpleRed~&
Function cDarkRed~& ()
cDarkRed = _RGB32(160, 0, 64)
End Function ' cDarkRed~&
Function cBrickRed~& ()
cBrickRed = _RGB32(192, 0, 32)
End Function ' cBrickRed~&
Function cDarkGreen~& ()
cDarkGreen = _RGB32(0, 100, 0)
End Function ' cDarkGreen~&
Function cGreen~& ()
cGreen = _RGB32(0, 128, 0)
End Function ' cGreen~&
Function cOliveDrab~& ()
cOliveDrab = _RGB32(107, 142, 35)
End Function ' cOliveDrab~&
Function cLightPink~& ()
cLightPink = _RGB32(255, 182, 193)
End Function ' cLightPink~&
Function cHotPink~& ()
cHotPink = _RGB32(255, 105, 180)
End Function ' cHotPink~&
Function cDeepPink~& ()
cDeepPink = _RGB32(255, 20, 147)
End Function ' cDeepPink~&
Function cMagenta~& ()
cMagenta = _RGB32(255, 0, 255)
End Function ' cMagenta~&
Function cBlack~& ()
cBlack = _RGB32(0, 0, 0)
End Function ' cBlack~&
Function cDimGray~& ()
cDimGray = _RGB32(105, 105, 105)
End Function ' cDimGray~&
Function cGray~& ()
cGray = _RGB32(128, 128, 128)
End Function ' cGray~&
Function cDarkGray~& ()
cDarkGray = _RGB32(169, 169, 169)
End Function ' cDarkGray~&
Function cSilver~& ()
cSilver = _RGB32(192, 192, 192)
End Function ' cSilver~&
Function cLightGray~& ()
cLightGray = _RGB32(211, 211, 211)
End Function ' cLightGray~&
Function cGainsboro~& ()
cGainsboro = _RGB32(220, 220, 220)
End Function ' cGainsboro~&
Function cWhiteSmoke~& ()
cWhiteSmoke = _RGB32(245, 245, 245)
End Function ' cWhiteSmoke~&
Function cWhite~& ()
cWhite = _RGB32(255, 255, 255)
'cWhite = _RGB32(254, 254, 254)
End Function ' cWhite~&
Function cDarkBrown~& ()
cDarkBrown = _RGB32(128, 64, 0)
End Function ' cDarkBrown~&
Function cLightBrown~& ()
cLightBrown = _RGB32(196, 96, 0)
End Function ' cLightBrown~&
Function cKhaki~& ()
cKhaki = _RGB32(240, 230, 140)
End Function ' cKhaki~&
Function cEmpty~& ()
'cEmpty~& = -1
cEmpty = _RGB32(0, 0, 0, 0)
End Function ' cEmpty~&
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END COLOR FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++