Welcome, Guest
You have to register before you can post on our site.

Username/Email:
  

Password
  





Search Forums

(Advanced Search)

Forum Statistics
» Members: 496
» Latest member: braveparrot
» Forum threads: 2,846
» Forum posts: 26,669

Full Statistics

Latest Threads
Fun with Ray Casting
Forum: a740g
Last Post: MasterGy
1 hour ago
» Replies: 9
» Views: 150
QBJS - ANSI Draw
Forum: QBJS, BAM, and Other BASICs
Last Post: dbox
3 hours ago
» Replies: 3
» Views: 77
Cautionary tale of open, ...
Forum: General Discussion
Last Post: doppler
9 hours ago
» Replies: 3
» Views: 87
Extended KotD #23 and #24...
Forum: Keyword of the Day!
Last Post: SMcNeill
9 hours ago
» Replies: 0
» Views: 34
Big problem for me.
Forum: General Discussion
Last Post: JRace
Today, 05:11 AM
» Replies: 11
» Views: 187
Virtual Arrays
Forum: Site Suggestions
Last Post: hsiangch_ong
Today, 12:35 AM
» Replies: 8
» Views: 298
QBJS v0.9.0 - Release
Forum: QBJS, BAM, and Other BASICs
Last Post: hsiangch_ong
Today, 12:25 AM
» Replies: 17
» Views: 310
Very basic key mapping de...
Forum: SMcNeill
Last Post: SMcNeill
Yesterday, 11:18 PM
» Replies: 0
» Views: 34
1990's 3D Doom-Like Walls...
Forum: Programs
Last Post: a740g
Yesterday, 06:12 PM
» Replies: 10
» Views: 327
Editor WIP
Forum: bplus
Last Post: aadityap0901
Yesterday, 08:54 AM
» Replies: 12
» Views: 687

 
  beginning of spacewar! / asteroids type game
Posted by: madscijr - 11-01-2022, 09:10 PM - Forum: Works in Progress - No Replies

This started as an old VB6 Asteroids type game by someone named Tassadar that I downloaded from planetsourcecode.com years ago, and converted to QB64. 

It's very easy to understand and modify to make something more interesting. 

The next thing I would like to figure out is how to change how it handles x/y velocity for smoother more natural acceleration / momentum.

Enjoy

Code: (Select All)
_Title "Collisions"

' Original VB6 code for "Collisions" by Tassadar,
' found on PlanetSourceCode.com in 2001.
' Converted to QB64 by madscijr.

' DATE         WHO-DONE-IT   DID-WHAT
' 2001-06-10   Tassadar      created program
' 2022-11-01   madscijr      v0.19, got it working in QB64, added some tweaks

' DONE:
' * get it working in QB64 + tweak various calculations
' * random enemy radius (wider range with each level)
' * added variables for bullet lifespan, bullet radius
' * bullets can wrap around screen
' * bullet-bullet collisions
' * bonus shields at end of round
' * are you lookin' at me??

' TODO:
' * track x/y velocity differently (movement is jerky)
' * option: enemies can shoot each other
' * option: fire button auto-repeat
' * support simple polygon shapes
' * improve collisions
' * support higher resolution upto 4k to fit lots of stuff on screen
' * local multiplayer Spacewar! (upto 16 players)
' * sound effects
' * explosions and stuff
' * menu + customize input + options
' * sun and gravity
' * asteroids
' * enemies can move + enemy AI
' * vary difficulty + scoring based on difficulty
' * limited fuel/ammo
' * player must dock with space station for more fuel/ammo/repair
' * gravity for large asteroids/moons
' * lunar lander mode, players land moons/asteroids to refuel, etc.
' * attack opponents' bases, etc.
' * etc.

' ORIGINAL CREDITS:
'********************************************************************************
'*            This is My Program, I made it to show some trig stuff             *
'* and got a little carried away, but it shows how to do trigonometry anyways   *
'*                                                                              *
'*    You are permitted to do whatever you want with this code                  *
'*        eg. Feel free to modify this code, steal it, etc.                     *
'*               I don't really give a crap!                                    *
'*                                                                              *
'* Programmed by Tassadar                                                       *
'********************************************************************************

' USEFUL CONSTANTS
Const FALSE = 0
Const TRUE = Not FALSE
'Const Pi = 3.14159 ' Pi , bit obvious

Const iFPS = 60 ' Delay between frames (frames per second)

Const SHIP_RADIUS = 10 ' What is the radius of the players ship
Const BULLET_RADIUS = 2 ' What is the radius of the bullets
Const SHIP_ACCEL = 0.05 ' 0.1 ' how fast does it accelerate
Const MAX_SPEED = 12 ' 6 ' what is the ships max speed
Const SHOOT_DELAY = 200 ' Delay between shots for the ship
Const BULLET_SPEED = 9 ' 6 ' Bullet speed - Ship speed + bullet speed = overall bulletspeed
Const BULLET_LIFESPAN = 1 ' # seconds bullet is alive
Const BULLETS_STOP_BULLETS = TRUE

Const TURN_SPEED = 72 ' 36=faster 18=superfast

Const MIN_ENEMY_RADIUS = 6 ' What is the initial minimum radius of the enemy ships
Const MAX_ENEMY_RADIUS = 99 ' What is the initial maximum radius of the enemy ships

Const iMinX = 1
Const iMaxX = 800
Const iMinY = 1
Const iMaxY = 640

Const BULLET_DAMAGE = 5
Const WRAP_BULLET = TRUE
Const BONUS_SHIELDS = 50

Type ShipType
    xPos As Integer ' X co-ordinate of the ship
    yPos As Integer ' Y co-ordinate of the ship
    heading As Single ' which direction is the ship heading
    facing As Single ' which direction is the ship facing
    shields As Integer ' how much shields does the ship have
    speed As Single ' how fast is the ship going
    ShootTime As Long
    ShootCount As Long
End Type ' ShipType

Type EnemyType
    xPos As Integer ' X position of this enemy
    yPos As Integer ' Y position of this enemy
    life As Integer ' How much life does this enemy have
    alive As Integer ' Is this enemy alive
    radius As Integer ' size of enemy ship
End Type ' EnemyType

Type BulletType
    xPos As Integer ' X co-ordinate of this bullet
    yPos As Integer ' Y co-ordinate of this bullet
    heading As Single ' Direction this bullet is heading
    speed As Single ' Speed of this bullet
    alive As Integer ' Is this bullet alive
    kind As String ' What type of bullet is this (Players or enemies)
    lifespan As Long
    lifetime As Long
End Type ' BulletType

'' ENABLE / DISABLE DEBUG CONSOLE
'Dim Shared m_bDebug As Integer: m_bDebug = 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)

' OTHER USEFUL VARIABLES
Dim Shared PI As Single: PI = 4 * Atn(1)
Dim Shared ENGINE_RADIUS As Integer: ENGINE_RADIUS = SHIP_RADIUS * 0.6
Dim Shared FLAME_RADIUS As Integer: FLAME_RADIUS = SHIP_RADIUS * 2

' GAME STATE
Dim Shared m_bGameOver As Integer ' Is the game over
Dim Shared m_bAllDead As Integer ' Are all the enemies dead
Dim Shared m_iLevel As Integer ' Track the level
Dim Shared m_iScore As Integer ' Keeps track of player score
Dim Shared m_iMinEnemyRadius As Integer ' current minimum enemy radius
Dim Shared m_iMaxEnemyRadius As Integer ' current maximum enemy radius

' INPUT VARIABLES
Dim Shared m_bLeftKey As Integer ' Is the LeftKey depressed
Dim Shared m_bRightKey As Integer ' Is the RightKey depressed
Dim Shared m_bUpKey As Integer ' Is the UpKey depressed
Dim Shared m_bDownKey As Integer ' Is the DownKey depressed
Dim Shared m_bShootKey As Integer ' Is the ShootKey depressed
Dim Shared m_bCheatKey As Integer
Dim Shared m_bEscKey As Integer

' GAME OBJECTS
Dim Shared m_Ship As ShipType ' The Players Ship
ReDim Shared m_arrEnemy(-1) As EnemyType ' A nice array of enemies
ReDim Shared m_arrBullet(-1) As BulletType ' A nice array of Bullets

'' ****************************************************************************************************************************************************************
'' ACTIVATE DEBUGGING WINDOW
'If m_bDebug = TRUE Then
'    $Console
'    _Delay 4
'    _Console On
'    _Echo "Started " + m_ProgramName$
'    _Echo "Debugging on..."
'End If
'' ****************************************************************************************************************************************************************

' START THE GAME
main

' FINISHED
System ' return control to the operating system

'' ****************************************************************************************************************************************************************
'' DEACTIVATE DEBUGGING WINDOW
'If m_bDebug = TRUE Then
'    _Console Off
'End If
' ****************************************************************************************************************************************************************

' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN GAME CODE
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

' /////////////////////////////////////////////////////////////////////////////

Sub main ()
    Dim RoutineName As String: RoutineName = "main"

    Screen _NewImage(iMaxX, iMaxY, 32) ' 100 text columns x 40 text rows
    _KeyClear
   
    InitVariables ' Initialize variables
   
    Do ' main game loop
        Cls ' Clear the form
        GetInput
        If m_bEscKey = FALSE Then
            MoveBullets ' Activates the MoveBullets sub
            MoveShip ' Activates the MoveShip sub
            MoveEnemy ' (doesn't do much yet)
            Collisions ' Activates the Collisions sub
        Else
            m_bGameOver = TRUE
        End If
       
        If m_bGameOver = FALSE Then
            Shooting ' Activates the Shooting sub
            DrawEnemy ' Activates the DrawEnemy sub
            DrawBullets ' Activates the DrawBullets sub
            ShowScore ' Display the score, etc.
            DrawShip ' Activates the DrawShip sub
            Respawn ' Activates the Respawn sub
        Else
            If AskPlayAgain% = TRUE Then
                InitVariables
            Else
                Exit Do
            End If
        End If

        ' UPDATE THE SCREEN
        _Display
       
        ' CONTROL GAME SPEED
        _Limit iFPS
    Loop
   
    ' RETURN TO AUTODISPLAY
    _AutoDisplay
   
End Sub ' main

' /////////////////////////////////////////////////////////////////////////////

Function AskPlayAgain%
    Dim bResult As Integer
    Dim in$
   
    Cls
    Print "GAME OVER"
    Print
    Print "Level: " + cstr$(m_iLevel)
    Print "Score: " + cstr$(m_iScore)
    Print
    Do
        Input "Do you wish to try again (y/n) "; in$
        If LCase$(_Trim$(in$)) = "y" Then
            bResult = TRUE
            Exit Do
        ElseIf LCase$(_Trim$(in$)) = "n" Then
            bResult = FALSE
            Exit Do
        Else
            Print
            Print "Please type 'y' or 'n'"
            Print
        End If
    Loop
   
    AskPlayAgain% = bResult
End Function ' AskPlayAgain%

' /////////////////////////////////////////////////////////////////////////////
' Set the initial state for variables

Sub InitVariables ()
    Dim iLoop1 As Integer ' Used for variables
    Dim iSpread As Integer
    Dim iHalf As Integer
    Dim iDivisor As Integer
   
    ' Msgbox telling you how to play
    'MsgBox "Use the arrow keys to fly around" + vbCrLf + "Control to shoot", vbOKOnly, "How To Play"
   
    ' Score
    m_iLevel = 1
    m_iScore = 0
   
    ' Game status
    m_bGameOver = FALSE
   
    ' Enemy min/max radius
    iSpread = MAX_ENEMY_RADIUS - MIN_ENEMY_RADIUS
    iHalf = iSpread / 2
    iDivisor = iSpread / 10
    m_iMinEnemyRadius = iHalf - iDivisor
    If m_iMinEnemyRadius < MIN_ENEMY_RADIUS Then m_iMinEnemyRadius = MIN_ENEMY_RADIUS
    m_iMaxEnemyRadius = iHalf + iDivisor
    If m_iMaxEnemyRadius > MAX_ENEMY_RADIUS Then m_iMaxEnemyRadius = MAX_ENEMY_RADIUS
   
    ' Clear input flags
    m_bLeftKey = FALSE
    m_bRightKey = FALSE
    m_bUpKey = FALSE
    m_bDownKey = FALSE
    m_bShootKey = FALSE
   
    ' Set the starting positions of the ship
    m_Ship.heading = 0
    m_Ship.facing = 0
    m_Ship.shields = 100
    m_Ship.speed = 0
    m_Ship.xPos = RandomNumber%(iMinX, iMaxX) 'Int(Rnd * ScaleWidth + 1)
    m_Ship.yPos = RandomNumber%(iMinY, iMaxY) 'Int(Rnd * ScaleHeight + 1)
    m_Ship.ShootTime = iFPS \ 4
    m_Ship.ShootCount = m_Ship.ShootTime + 1
   
    ' Spawn enemy
    ReDim _Preserve m_arrEnemy(0) As EnemyType
    For iLoop1 = 0 To UBound(m_arrEnemy)
        ' Set the starting position of the enemies
        m_arrEnemy(iLoop1).alive = TRUE
        m_arrEnemy(iLoop1).life = 30
        m_arrEnemy(iLoop1).xPos = m_Ship.xPos
        m_arrEnemy(iLoop1).yPos = m_Ship.yPos
       
        ' choose a random size
        m_arrEnemy(iLoop1).radius = RandomNumber%(m_iMinEnemyRadius, m_iMaxEnemyRadius)
       
        ' Stops the enemy starting on top of the ship
        Do Until GetDist(m_arrEnemy(iLoop1).xPos, m_arrEnemy(iLoop1).yPos, m_Ship.xPos, m_Ship.yPos) > SHIP_RADIUS * 10
            m_arrEnemy(iLoop1).xPos = RandomNumber%(iMinX, iMaxX) 'Int(Rnd * ScaleWidth + 1)
            m_arrEnemy(iLoop1).yPos = RandomNumber%(iMinY, iMaxY) 'Int(Rnd * ScaleHeight + 1)
        Loop
    Next iLoop1
   
    ' RESET BULLETS
    ReDim _Preserve m_arrBullet(-1) As BulletType
End Sub ' InitVariables

' /////////////////////////////////////////////////////////////////////////////
' Detect which keys are pressed

Sub GetInput ()
    While _DeviceInput(1): Wend ' clear and update the keyboard buffer
   
    If _Button(KeyCode_Left%) Then
        m_bLeftKey = TRUE
        m_bRightKey = FALSE
    ElseIf _Button(KeyCode_Right%) Then
        m_bLeftKey = FALSE
        m_bRightKey = TRUE
    Else
        m_bLeftKey = FALSE
        m_bRightKey = FALSE
    End If
   
    If _Button(KeyCode_Up%) Then
        m_bUpKey = TRUE
        m_bDownKey = FALSE
    ElseIf _Button(KeyCode_Down%) Then
        m_bUpKey = FALSE
        m_bDownKey = TRUE
    Else
        m_bUpKey = FALSE
        m_bDownKey = FALSE
    End If

    If _Button(KeyCode_CtrlLeft%) Then
        m_bShootKey = TRUE
    ElseIf _Button(KeyCode_CtrlRight%) Then
        m_bShootKey = TRUE
    Else
        m_bShootKey = FALSE
    End If

    If _Button(KeyCode_1%) Then
        m_bCheatKey = TRUE
    Else
        m_bCheatKey = FALSE
    End If
   
    If _Button(KeyCode_Escape%) Then
        m_bEscKey = TRUE
    Else
        m_bEscKey = FALSE
    End If
   
    ' CLEAR KEYBOARD BUFFER
    _KeyClear

End Sub ' GetInput

' /////////////////////////////////////////////////////////////////////////////
' Check for collisions

' TODO: improve collision checking to handle different shape polygons, etc.

Sub Collisions ()
    Dim iLoop1 As Integer
    Dim iLoop2 As Integer
    Dim in$
   
    ' Check for bullet collisions
    For iLoop1 = 0 To UBound(m_arrBullet)
       
        ' IS THIS BULLET ALIVE?
        If m_arrBullet(iLoop1).alive = TRUE Then
           
            ' CHECK FOR BULLET HIT BULLET
            For iLoop2 = 0 To UBound(m_arrBullet)
                If iLoop2 <> iLoop1 Then
                    If BULLETS_STOP_BULLETS = TRUE Then
                        If GetDist(m_arrBullet(iLoop2).xPos, m_arrBullet(iLoop2).yPos, m_arrBullet(iLoop1).xPos, m_arrBullet(iLoop1).yPos) <= BULLET_RADIUS Then
                            ' BOTH SHOTS DESTROYED
                            m_arrBullet(iLoop1).alive = FALSE ' Destroy the bullet
                            m_arrBullet(iLoop2).alive = FALSE ' Destroy the other bullet
                        End If
                    End If
                End If
            Next iLoop2
        End If
       
        ' IS THIS BULLET STILL ALIVE?
        If m_arrBullet(iLoop1).alive = TRUE Then
           
            ' CHECK ENEMY BULLET
            If m_arrBullet(iLoop1).kind = "ENEMY" Then
                ' Check for collision between bullet and ship
                If GetDist(m_Ship.xPos, m_Ship.yPos, m_arrBullet(iLoop1).xPos, m_arrBullet(iLoop1).yPos) <= SHIP_RADIUS Then
                    m_Ship.shields = m_Ship.shields - BULLET_DAMAGE ' Take Damage
                    m_arrBullet(iLoop1).alive = FALSE ' Destroy the bullet
                End If
            Else
                ' CHECK FOR PLAYER'S BULLET
                If m_arrBullet(iLoop1).kind = "SHIP" Then
                    For iLoop2 = 0 To UBound(m_arrEnemy)
                        ' If the enemy is alive then
                        If m_arrEnemy(iLoop2).alive = TRUE Then
                            ' Check for collision between bullet and enemy
                            If GetDist(m_arrEnemy(iLoop2).xPos, m_arrEnemy(iLoop2).yPos, m_arrBullet(iLoop1).xPos, m_arrBullet(iLoop1).yPos) <= m_arrEnemy(iLoop2).radius Then
                                m_arrEnemy(iLoop2).life = m_arrEnemy(iLoop2).life - BULLET_DAMAGE ' Enemy take damage
                                m_arrBullet(iLoop1).alive = FALSE ' Destroy the bullet
                            End If
                        End If
                    Next iLoop2
                End If
            End If
        End If
    Next iLoop1
   
    ' CHECK FOR SHIP COLLIDING WITH ENEMY
    For iLoop1 = 0 To UBound(m_arrEnemy)
        ' If the enemy is alive then
        If m_arrEnemy(iLoop1).alive = TRUE Then
            ' Check for collision between ship and enemy
            If GetDist(m_Ship.xPos, m_Ship.yPos, m_arrEnemy(iLoop1).xPos, m_arrEnemy(iLoop1).yPos) <= m_arrEnemy(iLoop1).radius Then
                m_arrEnemy(iLoop1).life = 0 ' The enemy has no life/Dead
                m_Ship.shields = 0 ' The ship has no shields/Dead
            End If
           
            ' if the enemy is dead then destroy it, add to score
            If m_arrEnemy(iLoop1).life <= 0 Then
                m_arrEnemy(iLoop1).alive = FALSE
                m_iScore = m_iScore + 10
            End If
        End If
    Next iLoop1
   
    ' IS SHIP DEAD?
    If m_Ship.shields <= 0 Then
        '' Display the message box
        'iLoop1 = MsgBox("You have a score of " & m_iScore & vbCrLf & vbCrLf & "Do you wish to try again?", vbYesNo, "Try Again")
        'Select Case iLoop1
        '    Case vbYes
        '        ' Restart if yes is clicked
        '        InitVariables
        '    Case vbNo
        '        ' End if no clicked
        '        End
        'End Select
        m_bGameOver = TRUE
    End If
End Sub ' Collisions

' /////////////////////////////////////////////////////////////////////////////

Sub Shooting ()
    Dim iLoop1 As Integer ' Used for variables
    Dim iLoop2 As Integer ' Used for variables
    Dim iFreeSpot As Integer
    Dim sngXComp As Single
    Dim sngYComp As Single

    ' DID PLAYER SHOOT?
    If m_bShootKey = TRUE Then
        ' Has the gun cooled down yet (prevent bullet being created every 25 milliseconds)
       
        If m_Ship.ShootCount > m_Ship.ShootTime Then
            m_Ship.ShootCount = 0
           
            iFreeSpot = -1
           
            For iLoop1 = 0 To UBound(m_arrBullet)
                ' Check whether it can use another bullet or not
                If m_arrBullet(iLoop1).alive = FALSE Then
                    ' if so use the dead bullet
                    iFreeSpot = iLoop1
                    Exit For
                End If
            Next iLoop1
            ' if there were no already dead bullets
            If iFreeSpot = -1 Then
                ' create another one
                ReDim _Preserve m_arrBullet(UBound(m_arrBullet) + 1) As BulletType

                ' iFreeSpot is this new bullet
                iFreeSpot = UBound(m_arrBullet)
            End If

            ' Set the properties of this bullet
            m_arrBullet(iFreeSpot).alive = TRUE ' The bullet is alive
            m_arrBullet(iFreeSpot).xPos = m_Ship.xPos ' the bullet is created where the ship is
            m_arrBullet(iFreeSpot).yPos = m_Ship.yPos ' the bullet is created where the ship is
            m_arrBullet(iFreeSpot).kind = "SHIP" ' This is a Ship Bullet
            m_arrBullet(iFreeSpot).lifespan = BULLET_LIFESPAN * iFPS ' # seconds bullet is alive
            m_arrBullet(iFreeSpot).lifetime = 0 ' bullet is brand new
           
            ' Determine the X and Y components of the resultant vector
            sngXComp = m_Ship.speed * Sin(m_Ship.heading) + BULLET_SPEED * Sin(m_Ship.facing)
            sngYComp = m_Ship.speed * Cos(m_Ship.heading) + BULLET_SPEED * Cos(m_Ship.facing)

            ' Determine the resultant speed
            m_arrBullet(iFreeSpot).speed = Sqr(sngXComp ^ 2 + sngYComp ^ 2)

            'Calculate the resultant heading, and adjust for arctangent by adding Pi if necessary
            If Sgn(sngYComp) > 0 Then
                m_arrBullet(iFreeSpot).heading = Atn(sngXComp / sngYComp)
            End If
            If Sgn(sngYComp) < 0 Then
                m_arrBullet(iFreeSpot).heading = Atn(sngXComp / sngYComp) + PI
            End If
        End If
    End If
   
    ' ENEMIES SHOOT
    For iLoop1 = 0 To UBound(m_arrEnemy)
        ' Check whether the enemy is alive
        If m_arrEnemy(iLoop1).alive = TRUE Then
            ' Check whether the enemy will fire or not
            If Int(Rnd * 100 + 1) = 1 Then
                iFreeSpot = -1
               
                For iLoop2 = 0 To UBound(m_arrBullet)
                    ' Check whether the enemy will use an old bullet
                    If m_arrBullet(iLoop2).alive = FALSE Then
                        ' If so iFreeSpot is the old bullet
                        iFreeSpot = iLoop2
                        Exit For
                    End If
                Next iLoop2
               
                ' If there were no free spots then create another bullet
                If iFreeSpot = -1 Then
                    ' Create the new bullet
                    ReDim _Preserve m_arrBullet(UBound(m_arrBullet) + 1) As BulletType
                    ' iFreeSpot is this new bullet
                    iFreeSpot = UBound(m_arrBullet)
                End If
               
                ' Set the properties for this bullet
                m_arrBullet(iFreeSpot).alive = TRUE ' It is Alive!!!
               
                ' Set it so the bullet shoots at the ship
                m_arrBullet(iFreeSpot).heading = GetAngle(m_arrEnemy(iLoop1).xPos, m_arrEnemy(iLoop1).yPos, m_Ship.xPos, m_Ship.yPos)
                m_arrBullet(iFreeSpot).xPos = m_arrEnemy(iLoop1).xPos ' Create the bullet where the enemy is
                m_arrBullet(iFreeSpot).yPos = m_arrEnemy(iLoop1).yPos ' Create the bullet where the enemy is
                m_arrBullet(iFreeSpot).speed = 6 ' Set the bullet speed
                m_arrBullet(iFreeSpot).kind = "ENEMY" ' This is an enemy bullet
                m_arrBullet(iFreeSpot).lifespan = BULLET_LIFESPAN * iFPS ' # seconds bullet is alive
                m_arrBullet(iFreeSpot).lifetime = 0 ' bullet is brand new
               
                ' Move bullet outside of enemy
                m_arrBullet(iFreeSpot).xPos = m_arrBullet(iFreeSpot).xPos + ((m_arrEnemy(iLoop1).radius + 1) * Sin(m_arrBullet(iFreeSpot).heading))
                m_arrBullet(iFreeSpot).yPos = m_arrBullet(iFreeSpot).yPos - ((m_arrEnemy(iLoop1).radius + 1) * Cos(m_arrBullet(iFreeSpot).heading))
               
            End If
        End If
    Next iLoop1
   
End Sub ' Shooting

' /////////////////////////////////////////////////////////////////////////////
' Draw the enemies

Sub DrawEnemy ()
    Dim iLoop1 As Integer ' Used for variables
    Dim iLoop2 As Integer ' Used for variables
    Dim iColor As _Unsigned Long
    Dim iX As Integer
    Dim iY As Integer
    Dim sngHeading As Single
    Dim iRadius As Integer
   
    For iLoop1 = 0 To UBound(m_arrEnemy)
        ' Is this enemy alive
        If m_arrEnemy(iLoop1).alive = TRUE Then
            ' Color based on damage
            If m_arrEnemy(iLoop1).life >= 30 Then
                iColor = cWhite
            ElseIf m_arrEnemy(iLoop1).life > 25 Then
                iColor = cYellow
            ElseIf m_arrEnemy(iLoop1).life > 20 Then
                iColor = cGold
            ElseIf m_arrEnemy(iLoop1).life > 15 Then
                iColor = cOrange
            ElseIf m_arrEnemy(iLoop1).life > 10 Then
                iColor = cDarkOrange
            ElseIf m_arrEnemy(iLoop1).life > 5 Then
                iColor = cOrangeRed
            Else
                iColor = cRed
            End If
           
            ' Draw body
            ' CIRCLE (x, y), radius, color
            'DrawCircleSolid iX, iY, 8, cRed
            Circle (m_arrEnemy(iLoop1).xPos, m_arrEnemy(iLoop1).yPos), m_arrEnemy(iLoop1).radius, iColor
           
            ' Draw "eye"
            sngHeading = GetAngle(m_arrEnemy(iLoop1).xPos, m_arrEnemy(iLoop1).yPos, m_Ship.xPos, m_Ship.yPos)
            iX = m_arrEnemy(iLoop1).xPos
            iY = m_arrEnemy(iLoop1).yPos
            iRadius = m_arrEnemy(iLoop1).radius / 3
            iX = iX + (m_arrEnemy(iLoop1).radius - iRadius) * Sin(sngHeading)
            iY = iY - (m_arrEnemy(iLoop1).radius - iRadius) * Cos(sngHeading)
            Circle (iX, iY), iRadius, iColor
           
        End If
    Next iLoop1
End Sub ' DrawEnemy

' /////////////////////////////////////////////////////////////////////////////
' Move Bullets

Sub MoveBullets ()
    Dim iLoop1 As Integer ' Used for variables
   
    If m_Ship.ShootCount <= m_Ship.ShootTime Then
        m_Ship.ShootCount = m_Ship.ShootCount + 1
    End If
   
    For iLoop1 = 0 To UBound(m_arrBullet)
        ' Is the bullet alive
        If m_arrBullet(iLoop1).alive = TRUE Then
            ' Move the bullets
            m_arrBullet(iLoop1).xPos = m_arrBullet(iLoop1).xPos + (m_arrBullet(iLoop1).speed * Sin(m_arrBullet(iLoop1).heading))
            m_arrBullet(iLoop1).yPos = m_arrBullet(iLoop1).yPos - (m_arrBullet(iLoop1).speed * Cos(m_arrBullet(iLoop1).heading))
           
            ' Did the bullet move off screen horizontally?
            If m_arrBullet(iLoop1).xPos < iMinX Then
                If WRAP_BULLET = TRUE Then
                    m_arrBullet(iLoop1).xPos = iMaxX
                Else
                    m_arrBullet(iLoop1).alive = FALSE ' Destroy the bullet
                End If
            ElseIf m_arrBullet(iLoop1).xPos > iMaxX Then
                If WRAP_BULLET = TRUE Then
                    m_arrBullet(iLoop1).xPos = iMinX
                Else
                    m_arrBullet(iLoop1).alive = FALSE ' Destroy the bullet
                End If
            End If
           
            ' Did the bullet move off screen vertically?
            If m_arrBullet(iLoop1).yPos < iMinY Then
                If WRAP_BULLET = TRUE Then
                    m_arrBullet(iLoop1).yPos = iMaxY
                Else
                    m_arrBullet(iLoop1).alive = FALSE ' Destroy the bullet
                End If
            ElseIf m_arrBullet(iLoop1).yPos > iMaxY Then
                If WRAP_BULLET = TRUE Then
                    m_arrBullet(iLoop1).yPos = iMinY
                Else
                    m_arrBullet(iLoop1).alive = FALSE ' Destroy the bullet
                End If
            End If
           
            ' Time how long bullet stays active
            m_arrBullet(iLoop1).lifetime = m_arrBullet(iLoop1).lifetime + 1
            If m_arrBullet(iLoop1).lifetime > m_arrBullet(iLoop1).lifespan Then
                m_arrBullet(iLoop1).alive = FALSE ' Destroy the bullet
            End If
           
        End If
    Next iLoop1
End Sub ' MoveBullets

' /////////////////////////////////////////////////////////////////////////////
' Draw the bullets

Sub DrawBullets ()
    Dim iLoop1 As Integer ' Used for variables
    For iLoop1 = 0 To UBound(m_arrBullet)
        ' Is the bullet alive
        If m_arrBullet(iLoop1).alive = TRUE Then
            If m_arrBullet(iLoop1).kind = "SHIP" Then
                ' Is this a ship bullet, draw a white bullet
                'Circle (m_arrBullet(iLoop1).xPos, m_arrBullet(iLoop1).yPos), 3, cWhite
                DrawCircleSolid m_arrBullet(iLoop1).xPos, m_arrBullet(iLoop1).yPos, BULLET_RADIUS, cWhite
               
            ElseIf m_arrBullet(iLoop1).kind = "ENEMY" Then
                ' if this is enemy bullet, draw a red bullet
                'Circle (m_arrBullet(iLoop1).xPos, m_arrBullet(iLoop1).yPos), 3, cOrangeRed
                DrawCircleSolid m_arrBullet(iLoop1).xPos, m_arrBullet(iLoop1).yPos, BULLET_RADIUS, cOrangeRed
            End If
        End If
    Next iLoop1
End Sub ' DrawBullets

' /////////////////////////////////////////////////////////////////////////////
' Move the ship
Sub MoveShip ()
    Dim sngXComp As Single
    Dim sngYComp As Single
   
    ' If the left key is pressed then rotate the ship left
    If m_bLeftKey = TRUE Then
        m_Ship.facing = m_Ship.facing - PI / TURN_SPEED
    End If
   
    ' If the Right key is pressed then rotate the ship right
    If m_bRightKey = TRUE Then
        m_Ship.facing = m_Ship.facing + PI / TURN_SPEED
    End If
   
    ' If the up key is pressed then and accelerate it in the direction the ship is facing
    If m_bUpKey = TRUE Then
        ' ****************************************************************************************************************************************************************
        ' ****************************************************************************************************************************************************************
        ' ****************************************************************************************************************************************************************
        ' TODO: fix this to make the movement more natural...
       
        ' Determine the X and Y components of the resultant vector
        sngXComp = m_Ship.speed * Sin(m_Ship.heading) + SHIP_ACCEL * Sin(m_Ship.facing)
        sngYComp = m_Ship.speed * Cos(m_Ship.heading) + SHIP_ACCEL * Cos(m_Ship.facing)
       
        ' Determine the resultant speed
        m_Ship.speed = Sqr(sngXComp ^ 2 + sngYComp ^ 2)
       
        ' Calculate the resultant heading, and adjust for arctangent by adding Pi if necessary
        If Sgn(sngYComp) > 0 Then
            m_Ship.heading = Atn(sngXComp / sngYComp)
        End If
        If Sgn(sngYComp) < 0 Then
            m_Ship.heading = Atn(sngXComp / sngYComp) + PI
        End If
       
        ' ****************************************************************************************************************************************************************
        ' ****************************************************************************************************************************************************************
        ' ****************************************************************************************************************************************************************
    End If
   
    ' If the down key is pressed then and accelerate the ship in the opposite direction it is facing
    If m_bDownKey = TRUE And m_Ship.speed > -MAX_SPEED Then
        ' Determine the X and Y components of the resultant vector
        sngXComp = m_Ship.speed * Sin(m_Ship.heading) - SHIP_ACCEL * Sin(m_Ship.facing)
        sngYComp = m_Ship.speed * Cos(m_Ship.heading) - SHIP_ACCEL * Cos(m_Ship.facing)
       
        ' Determine the resultant speed
        m_Ship.speed = Sqr(sngXComp ^ 2 + sngYComp ^ 2)
       
        ' Calculate the resultant heading, and adjust for actangent by adding Pi if necessary
        If Sgn(sngYComp) > 0 Then
            m_Ship.heading = Atn(sngXComp / sngYComp)
        End If
        If Sgn(sngYComp) < 0 Then
            m_Ship.heading = Atn(sngXComp / sngYComp) + PI
        End If
    End If
   
    ' Did player hit cheat key?
    If m_bCheatKey = TRUE Then
        m_Ship.shields = m_Ship.shields + 10
    End If
   
    ' Don't let the ship go faster then the max speed
    If m_Ship.speed > MAX_SPEED Then
        m_Ship.speed = MAX_SPEED
    End If
   
    ' Move the ship
    m_Ship.xPos = m_Ship.xPos + m_Ship.speed * Sin(m_Ship.heading)
    m_Ship.yPos = m_Ship.yPos - m_Ship.speed * Cos(m_Ship.heading)
   
    ' Keep the ship inside the form
    If m_Ship.xPos < iMinX Then
        m_Ship.xPos = iMaxX
    End If
    If m_Ship.xPos > iMaxX Then
        m_Ship.xPos = iMinX
    End If
    If m_Ship.yPos < iMinY Then
        m_Ship.yPos = iMaxY
    End If
    If m_Ship.yPos > iMaxY Then
        m_Ship.yPos = iMinY
    End If
End Sub ' MoveShip

' /////////////////////////////////////////////////////////////////////////////
' Placeholder

Sub MoveEnemy ()
    Dim iLoop1 As Integer
    'For iLoop1 = 0 To UBound(m_arrEnemy)
    '    ' Check whether the enemy is alive
    '    If m_arrEnemy(iLoop1).alive = TRUE Then
    '    End If
    'Next iLoop1
End Sub ' MoveEnemy

' /////////////////////////////////////////////////////////////////////////////

Sub ShowScore
    ' Draw background
    Color cBlue, cBlue
    PrintAt 0, 0, String$(120, " ")
   
    ' Place the text on the form
    Color cLime, cEmpty
    PrintAt 0, 10, "Shields: " + cstr$(m_Ship.shields) ' LeftPadString$(cstr$(m_Ship.shields), 5, " ")
   
    ' Title displays the players score
    'frmCollision.Caption = "Score: " & m_iScore
    Color cCyan, cEmpty
    PrintAt 0, 40, "Score: " + cstr$(m_iScore) ' LeftPadString$(cstr$(m_iScore), 10, " ")
   
    ' Display the level
    Color cWhite, cEmpty
    PrintAt 0, 70, "Level: " + cstr$(m_iLevel) ' LeftPadString$(cstr$(m_iScore), 10, " ")
   
    ' Show instructions
    Color cRed, cRed
    PrintAt 39, 0, String$(120, " ")
    Color cWhite, cEmpty
    PrintAt 39, 0, "CONTROLS: LEFT/RIGHT = TURN   UP/DOWN = FORWARD/BACK   CTRL=FIRE   1=ADD SHIELD (CHEAT)"
   
End Sub ' ShowScore

' /////////////////////////////////////////////////////////////////////////////
' Draw the ship

Sub DrawShip ()
    Dim intX1 As Integer
    Dim intY1 As Integer
    Dim intX2 As Integer
    Dim intY2 As Integer
    Dim intX3 As Integer
    Dim intY3 As Integer
   
    Dim intX1b As Integer ' engine flame ends here
    Dim intY1b As Integer ' engine flame ends here
    Dim intX2b As Integer ' back wall starts here on left
    Dim intY2b As Integer ' back wall starts here on left
    Dim intX3b As Integer ' back wall starts here on right
    Dim intY3b As Integer ' back wall starts here on right
   
    ' -----------------------------------------------------------------------------
    ' Set the coordinates of the ship
   
    ' front
    intX1 = m_Ship.xPos + SHIP_RADIUS * Sin(m_Ship.facing)
    intY1 = m_Ship.yPos - SHIP_RADIUS * Cos(m_Ship.facing)
   
    ' left rear
    intX2 = m_Ship.xPos + SHIP_RADIUS * Sin(m_Ship.facing + 2 * PI / 3)
    intY2 = m_Ship.yPos - SHIP_RADIUS * Cos(m_Ship.facing + 2 * PI / 3)
   
    ' left rear (3/4 of the way down)
    intX2b = m_Ship.xPos + ENGINE_RADIUS * Sin(m_Ship.facing + 2 * PI / 3)
    intY2b = m_Ship.yPos - ENGINE_RADIUS * Cos(m_Ship.facing + 2 * PI / 3)
   
    ' right rear
    intX3 = m_Ship.xPos + SHIP_RADIUS * Sin(m_Ship.facing + 4 * PI / 3)
    intY3 = m_Ship.yPos - SHIP_RADIUS * Cos(m_Ship.facing + 4 * PI / 3)

    ' right rear (3/4 of the way down)
    intX3b = m_Ship.xPos + ENGINE_RADIUS * Sin(m_Ship.facing + 4 * PI / 3)
    intY3b = m_Ship.yPos - ENGINE_RADIUS * Cos(m_Ship.facing + 4 * PI / 3)
   
    ' rear where engine flames end
    intX1b = m_Ship.xPos - FLAME_RADIUS * Sin(m_Ship.facing)
    intY1b = m_Ship.yPos + FLAME_RADIUS * Cos(m_Ship.facing)
    'm_Ship.facing = m_Ship.facing - Pi / 36
   
    ' -----------------------------------------------------------------------------
    ' Draw the ship
   
    ' Draw the left side
    Line (intX1, intY1)-(intX2, intY2), cWhite
   
    ' Draw the right side
    Line (intX1, intY1)-(intX3, intY3), cWhite
   
    ' Draw the rear / aft side
    If m_bUpKey Then
        ' Draw rear side
        Line (intX2b, intY2b)-(intX3b, intY3b), cSilver
       
        ' Engine is firing
        'Line (intX2b, intY2b)-(intX3b, intY3b), cOrangeRed
       
        ' Draw the flame left side
        Line (intX1b, intY1b)-(intX2b, intY2b), cOrangeRed
       
        ' Draw the flame right side
        Line (intX1b, intY1b)-(intX3b, intY3b), cOrangeRed
       
    Else
        ' Draw rear side
        Line (intX2b, intY2b)-(intX3b, intY3b), cSilver
    End If
   
End Sub ' DrawShip

' /////////////////////////////////////////////////////////////////////////////
' Respawn the enemies if there all dead

Sub Respawn ()
    Dim iLoop1 As Integer ' Used for variables

    ' Check if all enemies are dead
    m_bAllDead = TRUE
    For iLoop1 = 0 To UBound(m_arrEnemy)
        ' If an enemy is alive then
        If m_arrEnemy(iLoop1).alive = TRUE Then
            ' enemies aren't all dead
            m_bAllDead = FALSE
            Exit For
        End If
    Next iLoop1
   
    ' if all dead, respawn and create one more enemy
    ' advance to next level
    If m_bAllDead = TRUE Then
        ' INCREASE LEVEL
        m_iLevel = m_iLevel + 1
       
        ' GIVE PLAYER SOME BONUS SHIELDS
        m_Ship.shields = m_Ship.shields + BONUS_SHIELDS
       
        ' INCREASE THE NUMBER OF ENEMIES
        ReDim _Preserve m_arrEnemy(UBound(m_arrEnemy) + 1) As EnemyType
       
        ' INCREASE VARIETY OF ENEMY SIZES
        If m_iMinEnemyRadius > MIN_ENEMY_RADIUS Then
            m_iMinEnemyRadius = m_iMinEnemyRadius - 1
        End If
        If m_iMaxEnemyRadius < MAX_ENEMY_RADIUS Then
            m_iMaxEnemyRadius = m_iMaxEnemyRadius + 1
        End If
       
        ' SPAWN NEW WAVE OF ENEMIES
        For iLoop1 = 0 To UBound(m_arrEnemy)
            ' Set the starting positions
            m_arrEnemy(iLoop1).alive = TRUE
            m_arrEnemy(iLoop1).life = 30
            m_arrEnemy(iLoop1).xPos = m_Ship.xPos
            m_arrEnemy(iLoop1).yPos = m_Ship.yPos
           
            ' choose a random size
            m_arrEnemy(iLoop1).radius = RandomNumber%(m_iMinEnemyRadius, m_iMaxEnemyRadius)
           
            ' make sure the enemies don't start on the ship
            Do Until GetDist(m_arrEnemy(iLoop1).xPos, m_arrEnemy(iLoop1).yPos, m_Ship.xPos, m_Ship.yPos) > SHIP_RADIUS * 10
                m_arrEnemy(iLoop1).xPos = RandomNumber%(iMinX, iMaxX) 'Int(Rnd * ScaleWidth + 1)
                m_arrEnemy(iLoop1).yPos = RandomNumber%(iMinY, iMaxY) 'Int(Rnd * ScaleHeight + 1)
            Loop
        Next iLoop1
    End If
End Sub ' Respawn

' /////////////////////////////////////////////////////////////////////////////

Function GetDist! (sngX1 As Single, sngY1 As Single, sngX2 As Single, sngY2 As Single) ' As Single
    Dim sngXComp As Single
    Dim sngYComp As Single
   
    ' Set the X componate
    sngXComp = sngX2 - sngX1
   
    ' Set the Y Componate
    sngYComp = sngY1 - sngY2
   
    ' Get the distance between the two objects
    GetDist = Sqr(sngXComp ^ 2 + sngYComp ^ 2)
End Function ' GetDist

' /////////////////////////////////////////////////////////////////////////////

Function GetAngle (sngX1 As Single, sngY1 As Single, sngX2 As Single, sngY2 As Single)
    Dim sngXComp As Single
    Dim sngYComp As Single
   
    ' Set the X componate
    sngXComp = sngX2 - sngX1
   
    ' Set the Y componate
    sngYComp = sngY1 - sngY2
   
    ' Calculate the resultant angle, and adjust for actangent by adding Pi if necessary
    If Sgn(sngYComp) > 0 Then
        GetAngle = Atn(sngXComp / sngYComp)
    End If
    If Sgn(sngYComp) < 0 Then
        GetAngle = Atn(sngXComp / sngYComp) + PI
    End If
End Function ' GetAngle


' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END GAME CODE
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN TRIG FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

' /////////////////////////////////////////////////////////////////////////////
' use angles in degrees units instead of radians (converted inside sub)
' Note this function uses whatever the default type is, better not be some Integer Type.
Function CosD (degrees)
    CosD = Cos(_D2R(degrees))
End Function ' CosD

' /////////////////////////////////////////////////////////////////////////////
' use angles in degrees units instead of radians (converted inside sub)
' Note this function uses whatever the default type is, better not be some Integer Type.
Function SinD (degrees)
    SinD = Sin(_D2R(degrees))
End Function ' SinD

' /////////////////////////////////////////////////////////////////////////////
' use angles in degrees units instead of radians (converted inside sub)
' Note this function uses whatever the default type is, better not be some Integer Type.
Function DAtan2 (x1, y1, x2, y2) ' The angle in degrees a 2nd point (x2, y2)  makes to a first point (x1, y1)
    ' Delta means change between 1 measure and another for example x2 - x1
    deltaX = x2 - x1
    deltaY = y2 - y1
   
    ' To find the angle point(x2, y2) makes to (x1, y1) in Degrees
    ' Take DegreeAngle = DAtan2(y2 - y1, x2 - x1)
    rtn = _R2D(_Atan2(deltaY, deltaX))
    If rtn < 0 Then
        DAtan2 = rtn + 360
    Else
        DAtan2 = rtn
    End If
End Function ' DAtan2

' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END TRIG FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++


' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' BEGIN GENERAL PURPOSE FUNCTIONS
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

' /////////////////////////////////////////////////////////////////////////////
' Integer to string

Function cstr$ (myValue)
    'cstr$ = LTRIM$(RTRIM$(STR$(myValue)))
    cstr$ = _Trim$(Str$(myValue))
End Function ' cstr$

' /////////////////////////////////////////////////////////////////////////////
' Hack function, to be replaced with something better

Function DblToInt% (dblOld As Double)
    Dim dblNew As Double
    Dim sValue As String
    Dim iPos As Integer

    dblNew = RoundDouble#(dblOld, 0)
    'sValue = _Trim$(Str$(dblNew))

    sValue = DblToStr$(dblNew)

    'iPos = InStr(1, sValue, ".")
    'If iPos > 0 Then
    '    DblToInt% = Val(Left$(sValue, iPos - 1))
    'Else
    '    DblToInt% = Val(sValue)
    'End If

    DblToInt% = Val(sValue)
End Function ' DblToInt%

' /////////////////////////////////////////////////////////////////////////////
' 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#)
    Dim result$: result$ = ""
    Dim value$
    Dim Xpos%
    Dim expo%
    Dim sign$
    Dim valu$
    Dim dot%
    Dim L%
    Dim add$
    Dim min$
    Dim DP$
    Dim 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")
        End If
        If expo% < 0 Then
            min$ = String$(Abs(expo%) - (dot% - 1), "0")
            DP$ = "."
        End If
        For n% = 1 To L%
            If Mid$(valu$, n%, 1) <> "." Then
                num$ = num$ + Mid$(valu$, n%, 1)
            End If
        Next n%
        result$ = _Trim$(sign$ + DP$ + min$ + num$ + add$)
    Else
        result$ = value$
    End If

    DblToStr$ = result$
End Function ' DblToStr$

' /////////////////////////////////////////////////////////////////////////////

Function DblRoundedToStr$ (dblValue As Double, intNumPlaces As Integer)
    Dim dblNew As Double
    dblNew = RoundDouble#(dblValue, intNumPlaces)
    DblRoundedToStr$ = DblToStr$(dblNew)
End Function ' DblRoundedToStr$

' /////////////////////////////////////////////////////////////////////////////

Function DoubleABS# (dblValue As Double)
    If Sgn(dblValue) = -1 Then
        DoubleABS# = 0 - dblValue
    Else
        DoubleABS# = dblValue
    End If
End Function ' DoubleABS#

' /////////////////////////////////////////////////////////////////////////////
' bplus: The Gold Standard is even better than THE QB64 CIRCLE sub in this respect!
' https://forum.qb64.org/index.php?topic=1044.135

' from Steve Gold standard
' Renamed fcirc to DrawCircleSolid

' Not as fast as DrawCircleTopLeft but pretty fast.

' Example:
' Screen _NewImage(800, 600, 32)
' _ScreenMove 250, 60
' For r = 250 To 0 Step -60
'     DrawCircleSolid 400, 300, r, _RGBA(255, 255, 255, 100)
' Next r

Sub DrawCircleSolid (CX As Integer, CY As Integer, R As Integer, C As _Unsigned Long)
    Dim Radius As Integer, RadiusError As Integer
    Dim X As Integer, Y As Integer

    Radius = Abs(R)
    RadiusError = -Radius
    X = Radius
    Y = 0

    If Radius = 0 Then PSet (CX, CY), C: Exit Sub

    ' Draw the middle span here so we don't draw it twice in the main loop,
    ' which would be a problem with blending turned on.
    Line (CX - X, CY)-(CX + X, CY), C, BF

    While X > Y
        RadiusError = RadiusError + Y * 2 + 1
        If RadiusError >= 0 Then
            If X <> Y + 1 Then
                Line (CX - Y, CY - X)-(CX + Y, CY - X), C, BF
                Line (CX - Y, CY + X)-(CX + Y, CY + X), C, BF
            End If
            X = X - 1
            RadiusError = RadiusError - X * 2
        End If
        Y = Y + 1
        Line (CX - X, CY - Y)-(CX + X, CY - Y), C, BF
        Line (CX - X, CY + Y)-(CX + X, CY + Y), C, BF
    Wend
End Sub ' DrawCircleSolid

' /////////////////////////////////////////////////////////////////////////////

Function FloatRoundedToStr$ (fValue As _Float, intNumPlaces As Integer)
    Dim fNew As _Float
    fNew = Round##(fValue, intNumPlaces)
    FloatRoundedToStr$ = FloatToStr$(fNew)
End Function ' DblRoundedToStr$

' /////////////////////////////////////////////////////////////////////////////
' 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 FloatToStr$ (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")
        End If
        If expo% < 0 Then
            min$ = String$(Abs(expo%) - (dot% - 1), "0")
            DP$ = "."
        End If
        For n = 1 To L%
            If Mid$(valu$, n, 1) <> "." Then
                num$ = num$ + Mid$(valu$, n, 1)
            End If
        Next n
    Else
        FloatToStr$ = value$
        Exit Function
    End If
    FloatToStr$ = _Trim$(sign$ + DP$ + min$ + num$ + add$)
End Function ' FloatToStr$

' /////////////////////////////////////////////////////////////////////////////
' 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%

' /////////////////////////////////////////////////////////////////////////////
' 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$

' /////////////////////////////////////////////////////////////////////////////

Function LeftPadString$ (myString$, toWidth%, padChar$)
    LeftPadString$ = Right$(String$(toWidth%, padChar$) + myString$, toWidth%)
End Function ' LeftPadString$

' /////////////////////////////////////////////////////////////////////////////
' 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&

' /////////////////////////////////////////////////////////////////////////////
' iRow% and iCol% are 0-based in this version

Sub PrintAt (iRow%, iCol%, sText$)
    '_PrintString (iCol% * 8, iRow% * 16), sText$
    _PrintString (iCol% * 8, iRow% * 16), sText$
    '_PrintString (iCol%, iRow%), sText$
End Sub ' PrintAt

' /////////////////////////////////////////////////////////////////////////////
' 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 RightPadString$ (myString$, toWidth%, padChar$)
    RightPadString$ = Left$(myString$ + String$(toWidth%, padChar$), toWidth%)
End Function ' RightPadString$

' /////////////////////////////////////////////////////////////////////////////
' 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.

' Receives + returns _FLOAT myVar## (-1.18E-4932 to +1.18E+4932)

' =============================================================================
' ROUNDING FUNCTIONS FOR TYPE _FLOAT

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 RoundScientific## (num##, digits%)
    RoundScientific## = _Round(num## * 10 ^ digits%) / 10 ^ digits%
End Function

' =============================================================================
' ROUNDING FUNCTIONS FOR TYPE DOUBLE

Function RoundDouble# (num#, digits%)
    RoundDouble# = Int(num# * 10 ^ digits% + .5) / 10 ^ digits%
End Function

Function RoundUpDouble# (num#, digits%)
    RoundUpDouble# = _Ceil(num# * 10 ^ digits%) / 10 ^ digits%
End Function

Function RoundDownDouble# (num#, digits%)
    RoundDownDouble# = Int(num# * 10 ^ digits%) / 10 ^ digits%
End Function

Function RoundScientificDouble# (num#, digits%)
    RoundScientificDouble# = _Round(num# * 10 ^ digits%) / 10 ^ digits%
End Function

' =============================================================================
' ROUNDING FUNCTIONS FOR TYPE SINGLE

Function RoundSingle! (num!, digits%)
    RoundSingle! = Int(num! * 10 ^ digits% + .5) / 10 ^ digits%
End Function

' NOTE: not sure this one works: when digits%=3, it rounds .31 to .32
Function RoundUpSingle! (num!, digits%)
    RoundUpSingle! = _Ceil(num! * 10 ^ digits%) / 10 ^ digits%
End Function

Function RoundDownSingle! (num!, digits%)
    RoundDownSingle! = Int(num! * 10 ^ digits%) / 10 ^ digits%
End Function

Function RoundScientificSingle! (num!, digits%)
    RoundScientificSingle! = _Round(num! * 10 ^ digits%) / 10 ^ digits%
End Function

' /////////////////////////////////////////////////////////////////////////////

Function SmallestOf3% (i1%, i2%, i3%)
    Dim iMin%
    iMin% = i1%
    If i2% < iMin% Then iMin% = i2%
    If i3% < iMin% Then iMin% = i3%
    SmallestOf3% = iMin%
End Function ' SmallestOf3

' /////////////////////////////////////////////////////////////////////////////

Function SngRoundedToStr$ (sngValue As Single, intNumPlaces As Integer)
    Dim sngNew As Single
    sngNew = RoundSingle!(sngValue, intNumPlaces)
    SngRoundedToStr$ = SngToStr$(sngNew)
End Function ' SngRoundedToStr$

' /////////////////////////////////////////////////////////////////////////////
' Hack function, to be replaced with something better

Function SngToInt% (sngOld As Single)
    Dim sngNew As Single
    Dim sValue As String
    Dim iPos As Integer

    sngNew = RoundSingle!(sngOld, 0)
    'sValue = _Trim$(Str$(sngNew))

    sValue = SngToStr$(sngNew)

    'iPos = InStr(1, sValue, ".")
    'If iPos > 0 Then
    '    SngToInt% = Val(Left$(sValue, iPos - 1))
    'Else
    '    SngToInt% = Val(sValue)
    'End If

    SngToInt% = Val(sValue)
End Function ' SngToInt%

' /////////////////////////////////////////////////////////////////////////////
' 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!)
    Dim result$: result$ = ""
    Dim value$
    Dim Xpos%
    Dim expo%
    Dim sign$
    Dim valu$
    Dim dot%
    Dim L%
    Dim add$
    Dim min$
    Dim DP$
    Dim 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")
        End If
        If expo% < 0 Then
            min$ = String$(Abs(expo%) - (dot% - 1), "0")
            DP$ = "."
        End If
        For n% = 1 To L%
            If Mid$(valu$, n%, 1) <> "." Then
                num$ = num$ + Mid$(valu$, n%, 1)
            End If
        Next n%
        result$ = _Trim$(sign$ + DP$ + min$ + num$ + add$)
    Else
        result$ = value$
    End If

    SngToStr$ = result$
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

' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
' END GENERAL PURPOSE 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~&

' ADDED THIS FOR THE GAUGE COLOR:
Function cSpringGreen~& ()
    cSpringGreen = _RGB32(0, 255, 160)
End Function ' cSpringGreen~&

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
' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

' ################################################################################################################################################################
' 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

' ################################################################################################################################################################
' END KEYBOARD CODE FUNCTIONS
' ################################################################################################################################################################

'' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
'' BEGIN DEBUGGING ROUTINES #DEBUGGING
'' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
'
'Sub DebugPrint (MyString As String)
'    If m_bDebug = TRUE Then
'        '_Echo MyString
'
'        ReDim arrLines(-1) As String
'        Dim iLoop As Integer
'        split MyString, Chr$(13), arrLines()
'        For iLoop = LBound(arrLines) To UBound(arrLines)
'            _Echo arrLines(iLoop)
'        Next iLoop
'    End If
'End Sub ' DebugPrint
'
'' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
'' END DEBUGGING ROUTINES @DEBUGGING
'' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

' ################################################################################################################################################################
' #REFERENCE

' =============================================================================
' SOME USEFUL STUFF FOR REFERENCE:

' Type Name               Type suffix symbol   Minimum value                  Maximum value                Size in Bytes
' ---------------------   ------------------   ----------------------------   --------------------------   -------------
' _BIT                    `                    -1                             0                            1/8
' _BIT * n                `n                   -128                           127                          n/8
' _UNSIGNED _BIT          ~`                   0                              1                            1/8
' _BYTE                   %%                   -128                           127                          1
' _UNSIGNED _BYTE         ~%%                  0                              255                          1
' INTEGER                 %                    -32,768                        32,767                       2
' _UNSIGNED INTEGER       ~%                   0                              65,535                       2
' LONG                    &                    -2,147,483,648                 2,147,483,647                4
' _UNSIGNED LONG          ~&                   0                              4,294,967,295                4
' _INTEGER64              &&                   -9,223,372,036,854,775,808     9,223,372,036,854,775,807    8
' _UNSIGNED _INTEGER64    ~&&                  0                              18,446,744,073,709,551,615   8
' SINGLE                  ! or none            -2.802597E-45                  +3.402823E+38                4
' DOUBLE                  #                    -4.490656458412465E-324        +1.797693134862310E+308      8
' _FLOAT                  ##                   -1.18E-4932                    +1.18E+4932                  32(10 used)
' _OFFSET                 %&                   -9,223,372,036,854,775,808     9,223,372,036,854,775,807    Use LEN
' _UNSIGNED _OFFSET       ~%&                  0                              18,446,744,073,709,551,615   Use LEN
' _MEM                    none                 combined memory variable type  N/A                          Use LEN

' div: int1% = num1% \ den1%
' mod: rem1% = num1% MOD den1%

'#END

Print this item

  Odysee channel for QB64pe
Posted by: SpriggsySpriggs - 11-01-2022, 06:25 PM - Forum: General Discussion - Replies (18)

I've created an Odysee channel for the QB64pe project! Odysee is a blockchain site for sharing videos, livestreams, and articles. The first article I've posted there is the announcement of version 3.4.0! To see the article, follow this link: https://odysee.com/v3.4.0

I hope to also share some videos there later for showcasing new features as they come out. Maybe this could be something like Fellippe was trying to get going a while back. I don't know. I'm not as charismatic as he is. Big Grin

If you all have ideas for videos and things, please let me know. I can't guarantee I'll be quick to get stuff out, though.

Print this item

  QB64 Phoenix Edition v3.4.0 Released!
Posted by: DSMan195276 - 11-01-2022, 05:46 PM - Forum: Announcements - Replies (38)

QB64 Phoenix Edition v3.4.0!
https://github.com/QB64-Phoenix-Edition/...tag/v3.4.0

Enhancements

  • #189, #216, #225, #227, #230 - Added several cross-platform dialogs - @a740g, @mkilgore
    • The commands make a best-effort of determining the way to show the dialog based on what is provided by the system.
    • The follow new commands were added:

Bug Fixes
  • #218, #222 - `setup_osx.command` now works correctly when run from any directory - @mkilgore

Full Changelog: https://github.com/QB64-Phoenix-Edition/...0...v3.4.0

Print this item

  Space Invaders Clone
Posted by: TerryRitchie - 11-01-2022, 05:44 PM - Forum: Programs - Replies (23)

Back in 2011 or so I wrote QB64 Space Invaders which was a sad attempt at Space Invaders. In 2013 I rewrote it to clone the original as much as possible. However, it was buggy and I never released it. I rediscovered the code recently when rewriting the tutorial web site and decided it was time to do this code justice. I completely rewrote the code and I present to you a QB64 Space Invaders game that clones the original 99%. I spent the last 3 weeks working on this (and playing WAY TOO MANY games of Space Invaders working the kinks out. Give it a whirl and let me know what you think.

The ZIP file below contains the source code and all assets (sound files and graphics). The game does create a file upon startup called "si.sav" that is used to save the game options and high score. It only contains two text lines so will never exceed 1KB in size.

You can turn the bezel image on/off, the background image on/off, resize the screen 1x, 2x, 3x, or full screen. Just like the original you can choose the number of shields (3 to 6), the extra ship score (1000 or 1500), and coin required or free play through the use of DIP switches in the options screen (just like found on the motherboard of the original arcade systems). The game is capable of playing 1 or 2 players exactly like the original was as well.

Code: (Select All)
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
'
'                 ÛÛÛÛ     ÛÛÛÛÛÛÛ     ÛÛÛ       ÛÛÛÛ    ÛÛÛÛÛÛÛ               This software has been written for educational purposes only. Under no circumstances is this source
'                ÛÛÛÛÛÛÛ   ÛÛÛÛÛÛÛÛ    ÛÛÛ      ÛÛÛÛÛÛ   ÛÛÛÛÛÛÛ               code or compiled EXE to be sold or otherwise used for a profitable purpose.
'                ÛÛÛÛÛÛÛÛ  ÛÛÛÛÛÛÛÛÛ  ÛÛÛÛÛ     ÛÛÛÛÛÛÛ  ÛÛÛÛÛÛÛ
'                ÛÛÛÛÛÛÛÛ  ÛÛÛÛÛÛÛÛÛ  ÛÛÛÛÛ    ÛÛÛÛÛÛÛÛ  ÛÛÛÛÛÛÛ               Space Invaders was designed by Tomohiro Nishikado and is Copyright Taito Corporation with a license
'                ÛÛÛÛÛÛÛÛ  ÛÛÛ   ÛÛÛ ÛÛÛÛÛÛÛ   ÛÛÛ  ÛÛÛ ÛÛÛ                    to the Midway division of Bally Corporation for production. This software was written to pay homage
'                ÛÛ   ÛÛÛ  ÛÛÛ   ÛÛÛ ÛÛÛÛÛÛÛ   ÛÛÛ  ÛÛÛ ÛÛÛ                    to Mr. Nishikado's outstanding game.
'                ÛÛÛ  ÛÛÛ  ÛÛÛ   ÛÛÛ ÛÛÛÛÛÛÛ   ÛÛÛ  ÛÛÛ ÛÛÛ
'                 ÛÛÛ      ÛÛÛ   ÛÛÛ ÛÛ Û ÛÛ   ÛÛÛ      ÛÛÛ                    This source code has been released as open source which means you are free to use this source code
'                 ÛÛÛÛ     ÛÛÛÛÛÛÛÛÛ ÛÛ Û ÛÛ   ÛÛÛ      ÛÛÛÛÛ                  to learn from and modify without prior consent from the original author. It is requested, however,
'                 ÛÛÛÛÛ    ÛÛÛÛÛÛÛÛ  ÛÛÛÛÛÛÛ  ÛÛÛ      ÛÛÛÛÛÛ                  that if you do make modifications to this source code that the author, version, and any
'                  ÛÛÛÛÛÛ   ÛÛÛÛÛÛ    ÛÛÛÛÛ   ÛÛÛ      ÛÛÛÛÛÛ                  modifications be noted in the area provided below if you plan to redistribute the source code with
'                    ÛÛÛÛÛ  ÛÛÛÛÛÛ    ÛÛÛÛÛ   ÛÛÛ      ÛÛ                      your modifications.
'                      ÛÛÛ  ÛÛÛ       ÛÛ ÛÛ   ÛÛÛ      ÛÛ
'                   ÛÛ  ÛÛ  ÛÛÛ      ÛÛÛ ÛÛÛ  ÛÛÛ ÛÛÛ  ÛÛ                           Author      Version   Date     Modifications
'                   ÛÛ  ÛÛ  ÛÛÛ      ÛÛ   ÛÛ  ÛÛÛ ÛÛÛ ÛÛÛ                      ---------------- ------- -------- ------------------------------------------------------------------
'                   ÛÛÛÛÛÛ  ÛÛÛ     ÛÛÛÛ ÛÛÛÛ ÛÛÛÛÛÛÛ ÛÛÛÛÛÛ                   Terry Ritchie    1.0     06/24/13 Original version written in QB64 v0.954 SDL never released (buggy)
'                   ÛÛÛÛÛÛ  ÛÛÛ      ÛÛÛ ÛÛÛ   ÛÛÛÛÛ  ÛÛÛÛÛÛ                   ---------------- ------- -------- ------------------------------------------------------------------
'                    ÛÛÛÛÛ  ÛÛÛ       ÛÛ ÛÛ     ÛÛÛ   ÛÛÛÛÛÛ                   Terry Ritchie    2.0     10/31/22 Complete rewrite of code to support QB64PE v.3.3.0
'                     ÛÛÛ   ÛÛÛ       ÛÛ ÛÛ     ÛÛÛ   ÛÛÛÛÛÛ                                                     This source code should compile with versions of QB64 2.1 and up
'                                                                                                                The QB64PE logo was created by Pwillard at the QB64PE forum
'                 ÛÛ ÛÛ  ÛÛ ÛÛ  ÛÛ  ÛÛÛ  ÛÛÛÛ  ÛÛÛÛÛ  ÛÛÛ   ÛÛÛ                ---------------- ------- -------- ------------------------------------------------------------------
'                 ÛÛ ÛÛ  ÛÛ ÛÛ  ÛÛ ÛÛÛÛÛ ÛÛÛÛÛ ÛÛ    ÛÛÛÛÛ ÛÛ ÛÛ
'                 ÛÛ ÛÛ  ÛÛ ÛÛ  ÛÛ ÛÛÛÛÛ ÛÛ ÛÛ ÛÛ    ÛÛÛÛÛ ÛÛ
'                 ÛÛ ÛÛÛ ÛÛ ÛÛ  ÛÛ ÛÛ ÛÛ ÛÛ ÛÛ ÛÛ    ÛÛ ÛÛ ÛÛ
'                 ÛÛ ÛÛÛÛÛÛ ÛÛ  ÛÛ ÛÛ ÛÛ ÛÛ ÛÛ ÛÛ    ÛÛ ÛÛ ÛÛ
'                 ÛÛ ÛÛÛÛÛÛ  ÛÛÛÛ  ÛÛÛÛÛ ÛÛ ÛÛ ÛÛÛÛÛ ÛÛÛÛ   ÛÛÛ
'                 ÛÛ ÛÛ ÛÛÛ  ÛÛÛÛ  ÛÛÛÛÛ ÛÛ ÛÛ ÛÛ    ÛÛÛÛÛ    ÛÛ
'                 ÛÛ ÛÛ  ÛÛ  ÛÛÛÛ  ÛÛ ÛÛ ÛÛ ÛÛ ÛÛ    ÛÛ ÛÛ ÛÛ ÛÛ
'                 ÛÛ ÛÛ  ÛÛ   ÛÛ   ÛÛ ÛÛ ÛÛÛÛ  ÛÛ    ÛÛ ÛÛ  ÛÛÛ
'                 ÛÛ ÛÛ  ÛÛ   ÛÛ   ÛÛ ÛÛ ÛÛÛÛ  ÛÛÛÛÛ ÛÛ ÛÛ  ÛÛÛ
'
'                               QB64 adaptation
'                                      by
'                                Terry Ritchie
'                           (quickbasic64@gmail.com)
'                                   06/24/13
'                               Updated 10/31/22
'
'
' I tried to clone the original space invaders game as closely as possible. The deviations from the original are noted below:
' ---------------------------------------------------------------------------------------------------------------------------
' - Centered the high score.
' - UFO intervals are 23 to 27 seconds. Sources I found state every 25 seconds give or take a few seconds but nothing firm.
' - player keyboard instructions on screen for inserting coins, setting options, and exiting the game.
' - An options screen simulating DIP switch settings on the motherboard of arcade machine.
' - Ability to turn the background image on and off (set in options).
' - Ability to display cabinet bezel around the screen (set in options).
' - Ability to change the screen resolution from native 224x248 to 2x, 3x, and full screen (set in options).
' - QB64 and author credit screen added to demo loop.
' - Invader beat sounds are slightly longer in length than the original.
' - Scores are five digits instead of the four found in the original game.
' - Extra ship sound is not exact to the original.
' - The high score is saved between program executions with the ability to reset the high score if desired (reset in options).
' - There is no demo play yet between the different intro screens at startup. Coming in next version.
'
' What I found about the original that was emulated in the game.
' --------------------------------------------------------------
' - The invaders update in a wave pattern like the original due to the slow microprocessor.
' - As the invaders are destroyed the wave pattern increases in speed.
' - The UFO scoring is not random. After 23 shots the UFO will always be 300 points and every 15 shots after that.
' - DIP switch options for setting the number of shields (3 to 6), coin or free play, and extra ship score (1000 or 1500).
' - The player's laser will always be destroyed when hit by a bomb. The bomb will randomly survive the encounter.
' - The demo screen will alternate between spelling PLAY correctly and with an upside down Y. The upside down Y will be carried off and corrected by an invader.
' - The demo screen will alternative between spelling COIN correctly or adding an extra C. The extra C will be destroyed by an invader.
'
' Bug or easter egg?
' ------------------
' There was a "bug" in the original Space Invaders that some considered to be an "Easter Egg". Mr. Nishikado never confirmed this either way. When the invaders are in the row
' directly above the player the player is immune to bombs. I consider this to be a "bug" because the bottom of the bomb was used to detect a collision with the player's ship.
' Since the bomb moves down one pixel before the collision check is done the player's ship becomes immune. Since this was a bug in my view this behavior has not been emulated
' in this version of the game. However, this behavior would be very easy to emulate if desired.
'
' Coding today is easier!
' -----------------------
' Of all the games I've emulated this was surprisingly one of the most difficult to write. It makes me appreciate the complexity of writing software for the late 70's to early
' 80's microprocessors used in arcade games of the time. Those were true programmers crafting excellent games in Assembler on very limited processors with 2K to 4K of RAM and ROM!
'
' Known issues with this version of the game.
' -------------------------------------------
' - When selecting a different screen size or adding/removing the bezel image in options upon exit from the options screen the game window will not recenter on the desktop.
'   I've tried every method I can think of to get this to work. Upon game exit and restart the centering will work fine.
'
' Files included with the game:
' -----------------------------
' File created by the game - si.sav         - saved options and high score
' Support files            - sibeat1.ogg    - heart beat 1 sound
'                          - sibeat2.ogg    - heart beat 2 sound
'                          - sibeat3.ogg    - heart beat 3 sound
'                          - sibeat4.ogg    - heart beat 4 soound
'                          - sicoin.ogg     - coin inserted sound
'                          - siextra.ogg    - extra ship sound
'                          - siidead.ogg    - invader explosion sound
'                          - silaser.ogg    - laser sound
'                          - sipdead.ogg    - ship explosion sound
'                          - siudead.ogg    - UFO explosion sound
'                          - siufofly.ogg   - UFO flying sound
'                          - siicon.bmp     - window icon image
'                          - sisprites.png  - game graphics
'                          - invaders.bas   - QB64 source code
'                                                                                                                                                                  +--------------+
'                                                                                                                                                                  | METACOMMANDS |
'------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
'
OPTION _EXPLICIT '                                         force declaration of all variables
$VERSIONINFO:CompanyName=RitchCraft Creations
$VERSIONINFO:FileDescription=QB64 Space Invaders
$VERSIONINFO:InternalName=invaders.exe
$VERSIONINFO:ProductName=QB64 Space Invaders
$VERSIONINFO:OriginalFilename=invaders.exe
$VERSIONINFO:LegalCopyright=(c)2022 RitchCraft Creations
$VERSIONINFO:FILEVERSION#=2,0,0,0
$VERSIONINFO:PRODUCTVERSION#=2,0,0,0
'                                                                                                                                                                     +-----------+
'                                                                                                                                                                     | CONSTANTS |
'---------------------------------------------------------------------------------------------------------------------------------------------------------------------+-----------+
'
CONST FALSE = 0, TRUE = NOT FALSE '   boolean truth detecors
CONST PLAYER1 = 1 '                   player 1 value
CONST PLAYER2 = 2 '                   player 2 value
CONST BOTHPLAYERS = 3 '               both players value
CONST BLACK = _RGB32(0, 0, 0) '       color constants
CONST WHITE = _RGB32(255, 255, 255)
CONST INGAME = 5 '                    mode settings
CONST INOPTIONS = 1
CONST INCOIN = 2
CONST INSELECT = 3
CONST NEWGAME = -1
CONST NEWLEVEL = 0
CONST CLEARVARIABLES = -1
CONST GODMODE = FALSE '               developer option
'                                                                                                                                                             +-------------------+
'                                                                                                                                                             | TYPE DECLARATIONS |
'-------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------------+
TYPE RECT '                   rectangle definition                                                                                                                         | RECT |
    x1 AS INTEGER '           rectagular coordinates for objects and collision detection                                                                                   +------+
    y1 AS INTEGER
    x2 AS INTEGER '           x1,y1 = upper left    x2,y2 = lower right
    y2 AS INTEGER
END TYPE
'-------------------------------------------------------------------------------------------------------------------------------------------------------------------------+-------+
TYPE PAUSE '                  pause conditions                                                                                                                            | PAUSE |
    Level AS INTEGER '        between levels pause                                                                                                                        +-------+
    Die AS INTEGER '          after player death pause
END TYPE
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
TYPE LASERHIT '               laser hitting objects properties                                                                                                         | LASERHIT |
    Count AS INTEGER '        number of invaders player has hit
    Invader AS INTEGER '      countdown timer after invader hit by laser                                                                                               +----------+
    InvaderX AS INTEGER '     invader hit coordinates for explosion
    InvaderY AS INTEGER
    UFO AS INTEGER '          countdown timer after UFO hit by laser
    UFOX AS INTEGER '         UFO hit coordinate for score text
    Shield AS INTEGER '       countdown timer after shield hit by laser
    ShieldX AS INTEGER '      shield hit coordinates for explosion (mask)
    ShieldY AS INTEGER
    Bomb AS INTEGER '         countdown timer after bomb hit by laser
    BombX AS INTEGER '        bomb hit coordinates for explosion
    BombY AS INTEGER
END TYPE
'-----------------------------------------------------------------------------------------------------------------------------------------------------------------------+---------+
TYPE BOMBHIT '                bomb hitting objects properties                                                                                                           | BOMBHIT |
    Shield AS INTEGER '       countdown timer after shield hit by bomb
    ShieldX AS INTEGER '      shield hit coordinates for explosion (mask)
    ShieldY AS INTEGER
END TYPE
'-------------------------------------------------------------------------------------------------------------------------------------------------------------------------+-------+
TYPE LASER '                  laser properties                                                                                                                            | LASER |
    rect AS RECT '            laser coordinates                                                                                                                           +-------+
    Active AS INTEGER '       laser active (t/f)
    Hit AS LASERHIT '         laser hit something
    Miss AS INTEGER '         laser hit top of screen
    ShotsFired AS INTEGER '   number of laser shots fired (for UFO scoring)
END TYPE
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------------+------+
TYPE SHIP '                   player ship properties                                                                                                                       | SHIP |
    rect AS RECT '            player ship coordinates                                                                                                                      +------+
    Remain AS INTEGER '       player ships remain
    Dead AS INTEGER '         player ship is dead
    DeadX AS INTEGER '        player ship death location
    DeadImage AS INTEGER '    player ship exploding images indicator (-1 or 1)
    Extra AS INTEGER '        extra ship awarded to player (t/f)
END TYPE
'------------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------+
TYPE SOUNDS '                 sounds                                                                                                                                     | SOUNDS |
    InvaderHit AS LONG '      invader hit sound                                   snd.invaderhit                                                                         +--------+
    PlayerHit AS LONG '       player hit sound                                    snd.playerhit
    UFOHit AS LONG '          UFO hit sound                                       snd.ufohit
    UFOFlying AS LONG '       UFO slying sound                                    snd.ufoflying
    Laser AS LONG '           laser firing sound                                  snd.laser
    Coin AS LONG '            coin dropping sound                                 snd.coin
    Beat1 AS LONG '           heartbeat sounds                                    snd.beat1
    Beat2 AS LONG '                                                               snd.beat2
    Beat3 AS LONG '                                                               snd.beat3
    Beat4 AS LONG '                                                               snd.beat4
    Extra AS LONG '           extra ship sound                                    snd.extra
END TYPE
'------------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------+
TYPE IMAGES '                 images                                                                                                                                     | IMAGES |
    UFO AS LONG '             UFO image                                           img.ufo                                                                                +--------+
    InvaderHit AS LONG '      invader explosion                                   img.invaderhit
    BombHit AS LONG '         bomb explosion (bottom of screen and shields)       img.bombhit
    BombHitMask AS LONG '     destroy shields image                               img.bombhitmask
    LaserHit AS LONG '        laser hit explosion (top of screen and shields)     img.laserhit
    LaserHitMask AS LONG '    destroy shields image                               img.laserhitmask
    Shield AS LONG '          shield                                              img.shield
    QB64PE AS LONG '          QB64 Phoenix Edition logo                           img.qb64pe
    DipSwitch AS LONG '       single DIP switch image                             img.dipswitch
END TYPE
'-----------------------------------------------------------------------------------------------------------------------------------------------------------------------+---------+
TYPE OPTIONS '                game options                                                                                                                              | OPTIONS |
    ScreenSize AS INTEGER '   screen size (1 to 4)                                options.screensize                                                                    +---------+
    Shields AS INTEGER '      number of shields (3 to 6)                          options.shields
    ExtraShip AS INTEGER '    extra ship score (1000 or 1500)                     options.extraship
    FreePlay AS INTEGER '     free play (t/f)                                     options.freeplay
    Background AS INTEGER '   show background (t/f)                               options.background
    Bezel AS INTEGER '        show bezel (t/f)                                    options.bezel
    Switches AS INTEGER '     DIP switch settings                                 options.switches
    FullScreen AS INTEGER '   full screen (t/f)                                   options.fullscreen
END TYPE
'-----------------------------------------------------------------------------------------------------------------------------------------------------------------------+---------+
TYPE DISPLAY '                screens                                                                                                                                   | DISPLAY |
    WorkScreen AS LONG '      224x248 game work screen                            display.workscreen                                                                    +---------+
    Screen AS LONG '          screen to stretch work screen onto                  display.screen
    OptionScreen AS LONG '    options (dip switch) screen                         display.optionscreen
    ColorMask AS LONG '       color mask to lay over                              display.colormask
    WorkMask AS LONG '        work mask screen                                    display.workmask
    Bezel AS LONG '           bezel image                                         display.bezel
    Background AS LONG '      background image                                    display.background
    Bez AS RECT '             display screen coordinates within bezel             display.bez
    WithY AS LONG '           Images for intro animations                         display.withy
    WithoutY AS LONG '                                                            display.withouty
    CorrectY AS LONG '                                                            display.correcty
    AddedC AS LONG '                                                              display.addedc
    NormalC AS LONG '                                                             display.normalc
END TYPE
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------------+------+
TYPE GAME '                   game in progress settings                                                                                                                    | GAME |
    Player AS INTEGER '       current player playing game                         game.player                                                                              +------+
    Players AS INTEGER '      number of players playing game                      game.players
    Credits AS INTEGER '      credits inserted into machine                       game.credits
    Pause AS PAUSE '          pause needed between levels                         game.pause.level             (count down timer 120 frames or 2 seconds - PlayGame)
    '                         pause needed between player deaths                  game.pause.die               (count down timer 180 frames or 3 seconds - PlayGame)
    Frame AS INTEGER '        master frame counter                                game.frame
    HighScore AS LONG '       game high score                                     game.highscore
    Landed AS INTEGER '       invaders landed (t/f)                               game.landed
END TYPE
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------------+------+
TYPE BOMB '                   bomb properties                                                                                                                              | BOMB |
    rect AS RECT '            bomb coordinates                                    bomb().rect                                                                              +------+
    Hit AS BOMBHIT '          indicates bomb hit a shield                         bomb().hit.shield            (count down timer 5 frames - DrawShields)
    '                         where the bomb hit the shield                       bomb().hit.shieldx
    '                                                                             bomb().hit.shieldy
    Image AS INTEGER '        bomb image (1 to 3)                                 bomb().image
    Cell AS INTEGER '         bomb image animation cell (1 to 4)                  bomb().cell
    Active AS INTEGER '       bomb currently dropping (t/f)                       bomb().active
    Miss AS INTEGER '         bomb hit bottom of screen                           bomb().miss                  (count down timer 5 frames - DrawBombs)
END TYPE
'---------------------------------------------------------------------------------------------------------------------------------------------------------------------------+-----+
TYPE UFO '                    UFO properties                                                                                                                                | UFO |
    rect AS RECT '            UFO coordinates                                     ufo.rect                                                                                  +-----+
    Dir AS INTEGER '          UFO direction                                       ufo.dir
    Active AS INTEGER '       UFO active                                          ufo.active
    Score AS INTEGER '        UFO score                                           ufo.score
    Pause AS INTEGER '        Time to wait for next UFO                           ufo.pause                    (count down timer 1500 frames +/- 120 - DrawUFO)
END TYPE
'------------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------+
TYPE PLAYER '                 player properties                                                                                                                          | PLAYER |
    Ship AS SHIP '            player ship coordinates                             player().ship.rect                                                                     +--------+
    '                         player ships remaining                              player().ship.remain
    '                         indicates the player's ship was hit                 player().ship.dead
    '                         where on screen the ship was hit                    player().ship.deadx
    '                         player ship exploding images indicator (-1 or 1)    player().ship.deadimage
    '                         indicates if player was awarded extra ship (t/f)    player().ship.extra
    Level AS INTEGER '        player level                                        player().level
    Score AS LONG '           player score                                        player().score
    GameOver AS INTEGER '     player game over (t/f)                              player().gameover
    Laser AS LASER '          player laser coordinates                            player().laser.rect
    '                         indicates if laser if currently flying (t/f)        player().laser.active
    '                         indicates laser hit top of screen                   player().laser.miss          (count down timer 10 frames - DrawLaser)
    '                         indictaes laser hit an invader                      player().laser.hit.invader   (count down timer 5 frames - DrawInvaders)
    '                         location on screen where invader was hit            player().laser.hit.invaderx
    '                                                                             player().laser.hit.invadery
    '                         number of invaders killed (1 to 55)                 player().laser.hit.count
    '                         indicates if laser hit a UFO                        player().laser.hit.ufo       (count down timer 60 frames or 1 second - DrawUFO)
    '                         location on screen where UFO was hit                player().laser.hit.ufox
    '                         indicates that laser hit a shield                   player().laser.hit.shield    (count down timer 5 frames - DrawShields)
    '                         location on shield where laser hit                  player().laser.hit.shieldx
    '                                                                             player().laser.hit.shieldy
    '                         indicates that laser hit a bomb                     player().laser.hit.bomb      (count down timer 5 frames - DrawLaser)
    '                         location on screen where laser hit bomb             player().laser.hit.bombx
    '                                                                             player().laser.hit.bomby
    '                         number of lasers fired by player (for UFO score)    player().laser.shotsfired
    idir AS INTEGER '         invader direction (-2 or 2)                         player().idir
    MaxBombs AS INTEGER '     maximum number of invader bombs allowed (1 to 3)    player().maxbombs
    UFO AS UFO '              UFO saved state in multiplayer game                 player().ufo.rect
    '                         direction of UFO (-1 or 1)                          player().ufo.dir
    '                         indicates if UFO currently flying (t/f)             player().ufo.active
    '                         score of UFO if hit                                 player().ufo.score
    '                         time to wait in between UFO showings (23 to 27 sec) player().ufo.pause
    Keydown AS INTEGER '      player is holding a key down (t/f)                  player().keydown
END TYPE
'-----------------------------------------------------------------------------------------------------------------------------------------------------------------------+---------+
TYPE SHIELDS '                shield properties                                                                                                                         | SHIELDS |
    rect AS RECT '            shield coordinates                                  shield(player,x).rect                                                                 +---------+
    Image AS LONG '           damaged shield image                                shield(player,x).image
END TYPE
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
TYPE INVADERS '               invader properties                                                                                                                       | INVADERS |
    rect AS RECT '            invader coordinates                                 invader(player,column,row).rect                                                      +----------+
    Active AS INTEGER '       invader active (t/f)                                invader(player,column,row).active
    cell AS INTEGER '         invader image animation cell                        invader(player,column,row).cell
    Image AS INTEGER '        invader image '                                     invader(player,column,row).image
    Width AS INTEGER '        invader width                                       invader(player,column,row).width
    Score AS INTEGER '        invader score                                       invader(player,column,row).score
END TYPE
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
TYPE DROPCOLUMN '             bomb column properties                                                                                                                 | DROPCOLUMN |
    Pause AS INTEGER '        time to wait before dropping bomb in column         dropcolumn().pause           (count down timer)                                    +------------+
    Row AS INTEGER '          row that contains bottom invader (0 for none)       dropcolumn().row
END TYPE
'------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------------+
DIM Player(2) AS PLAYER '           player settings                                                                                                          | Declared Variables |
DIM Invader(2, 11, 5) AS INVADERS ' 55 invaders (player,column,row)                                                                                          +--------------------+
DIM DropColumn(11) AS DROPCOLUMN '  11 columns for bomb drops
DIM Bomb(3) AS BOMB '               invader bombs dropping
DIM UFO(2) AS UFO '                 UFO settings
DIM IMG AS IMAGES '                 images
DIM SND AS SOUNDS '                 sounds
DIM Display AS DISPLAY '            display screens
DIM Options AS OPTIONS '            player selectable options
DIM Game AS GAME '                  current game settings
DIM Font(255) AS LONG '             game font characters
DIM IMG_Invader(3, 1) AS LONG '     invader images and animation cells (image, cell)
DIM IMG_Bomb(3, 3) AS LONG '        bomb images and animation cells (image, cell)
DIM IMG_Ship(-1 TO 1) AS LONG '     images of player ship (-1,1 explosion, 0 intact)
DIM Shield(2, 6) AS SHIELDS '       player shield images (player, shield)
'                                                                                                                                                             +-------------------+
'                                                                                                                                                             | MAIN CODE SECTION |
'-------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------------+
LoadAssets '                                     load the games graphic and sound files                                                                       |   MAIN GAME LOOP  |
LoadOptions '                                    load the saved game options                                                                                  +-------------------+
Initialize '                                     initialize all game variables
DO '                                             begin game play loop
    IF Options.FreePlay THEN Game.Credits = 99 ' fill the game with coins if set to free play
    InsertCoin '                                 allow the player to insert coins (skipped if free play)
    SelectPlayers '                              select the number of players and play a game
LOOP '                                           loop back forever
'                                                                                                                                                     +---------------------------+
'                                                                                                                                                     |    END MAIN GAME LOOP     |
'-----------------------------------------------------------------------------------------------------------------------------------------------------+---------------------------+
'                                                                                                                                                     | SUBROUTINES AND FUNCTIONS |
'                                                                                                                                                     +---------------------------+
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
SUB PlayGame (Players AS INTEGER) '                                                                                                                                    | PlayGame |
    '+-----------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
    '| Plays a 1 or 2 player game of Space Invaders                                                                                                                               |
    '| Players - number of players (1 or 2)                                                                                                                                       |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Player() AS PLAYER ' need access to shared variables
    SHARED Game AS GAME
    SHARED UFO() AS UFO
    SHARED SND AS SOUNDS
    DIM KeyPress AS INTEGER '   keypress in slow text
    DIM p AS INTEGER '          current player

    StartNewGame '                                                                                         reset variables to start a new game
    Game.Players = Players '                                                                               record the number of players passed in
    p = Game.Player '                                                                                      get the current player
    IF Players = 1 THEN Player(2).GameOver = TRUE '                                                        kill off player 2 if a one player game
    GetReady '                                                                                             inform player 1 to get ready
    DO '                                                                                                   begin main game loop
        DO '                                                                                               begin game level loop
            _LIMIT 60 '                                                                                    game runs at 60 frames per second
            ClearDisplay INGAME '                                                                          clear the display
            IF Game.Pause.Level THEN '                                                                     is game paused for a level change?
                Game.Pause.Level = Game.Pause.Level - 1 '                                                  yes, decrement pause count down timer
                IF Game.Pause.Level = 0 THEN '                                                             has the count down finished?
                    ResetBombs '                                                                           yes, reset the bombs
                    DrawShields NEWLEVEL '                                                                 restore player shields for next level
                END IF
            END IF
            IF Players = 2 THEN '                                                                          is this a 2 player game?
                DrawScore BOTHPLAYERS, INGAME '                                                            yes, display both scores on screen
            ELSE '                                                                                         no, this is a 1 player game
                DrawScore PLAYER1, INGAME '                                                                display just player 1's score
            END IF
            MoveInvaders p, INGAME '                                                                       move the invaders
            DrawInvaders '                                                                                 draw the invaders to the screen
            DrawShields INGAME '                                                                           draw the player's shields
            DrawUFO '                                                                                      draw the UFO when active
            DrawShip '                                                                                     draw the player's ship
            DrawLaser '                                                                                    draw the player's laser when active
            DrawBombs '                                                                                    draw the invader bombs when active
            DrawShipsRemaining '                                                                           draw the number of ships the player has remaining
            UpdateDisplay INGAME '                                                                         update the display with all the changes
            IF _KEYDOWN(27) OR _EXIT THEN ExitGame '                                                       exit the game if the player presses ESC
        LOOP UNTIL Player(p).Laser.Hit.Count = 55 OR Player(p).Ship.Dead OR Game.Landed '                  leave when level finished or player dead
        IF Game.Landed THEN '                                                                              did the invaders land?
            Player(p).Ship.Dead = TRUE '                                                                   yes, the player is dead
            Player(p).Ship.DeadX = Player(p).Ship.rect.x1 '                                                the player died at this location
            _SNDPLAY SND.PlayerHit '                                                                       play the player death sound
            Player(p).Ship.Remain = 1 '                                                                    take all ships away from player
            Game.Landed = FALSE '                                                                          reset for 2nd player if needed
        END IF
        IF Player(p).Laser.Hit.Count = 55 THEN '                                                           did the player shoot all of the invaders?
            Game.Pause.Level = 120 '                                                                       yes, pause for 2 seconds between levels
            StartNewLevel '                                                                                reset variables for a new level
        ELSEIF Player(p).Ship.Dead THEN '                                                                  no, is the player dead?
            ResetBombs '                                                                                   yes, reset the bombs
            IF Game.Pause.Level THEN '                                                                     did the player die between level changes?
                Game.Pause.Level = 0 '                                                                     yes, stop the level pause count down
                DrawShields NEWLEVEL '                                                                     restore the player's shields for the new level
            END IF
            Player(p).Ship.Remain = Player(p).Ship.Remain - 1 '                                            take a ship away from player
            Game.Pause.Die = 180 '                                                                         3 second pause after player dies
            Player(p).Laser.Active = FALSE '                                                               deactivate the player's laser
            Player(p).UFO = UFO(p) '                                                                       save the player's UFO state
            IF UFO(p).Active AND Players = 2 THEN '                                                        is the UFO still active in a 2 player game?
                UFO(p).Active = FALSE '                                                                    yes, deactivate the UFO
                _SNDSTOP SND.UFOFlying '                                                                   stop the UFO sound
            END IF
            DO '                                                                                           begin player death pause loop
                _LIMIT 60 '                                                                                sequence will run at 60 frames per second
                ClearDisplay INGAME '                                                                      clear the display
                IF Players = 1 THEN '                                                                      is this a 1 player game?
                    DrawScore PLAYER1, INGAME '                                                            yes, just draw player 1's score
                    IF Player(1).Ship.Remain > 0 THEN '                                                    does the player have any ships remaining?
                        DrawUFO '                                                                          yes, keep the UFO flying during pause
                    ELSE '                                                                                 no, player 1's game is about to end
                        _SNDSTOP SND.UFOFlying '                                                           stop the UFO sound if it happens to be playing
                    END IF
                ELSE '                                                                                     no, this is a 2 player game
                    DrawScore BOTHPLAYERS, INGAME '                                                        draw both player's scores to the screen
                END IF
                DrawInvaders '                                                                             draw the invaders without moving
                DrawShields INGAME '                                                                       draw the player's current shields
                DrawShipsRemaining '                                                                       draw the number of ships the player has remaining
                IF Game.Pause.Die = 1 THEN '                                                               is this the last frame of the death pause?
                    IF Player(p).Ship.Remain = 0 THEN '                                                    yes, is the player out of ships?
                        IF Players = 1 THEN '                                                              yes, is this a 1 player game?
                            DrawScore PLAYER1, INGAME '                                                    yes, draw player 1's score
                            SlowText 6, 9, "GAME OVER", INGAME, KeyPress '                                 slowly tell player 1 that the game is over
                        ELSE '                                                                             no, this is a 2 player game
                            DrawScore BOTHPLAYERS, INGAME '                                                draw both player scores to the screen
                            SlowText 6, 5, "GAME OVER PLAYER<" + _TRIM$(STR$(p)) + ">", INGAME, KeyPress ' slowly tell the current player that the game is over
                        END IF
                        Player(p).GameOver = TRUE '                                                        the player's game is over
                        SLEEP 2 '                                                                          pause for 2 seconds to let it sink in
                    ELSE '                                                                                 no, the player has at least 1 ship remaining
                        Player(p).Ship.Dead = FALSE '                                                      bring the player back to life
                    END IF
                END IF
                DrawShip '                                                                                 draw the player's ship blowing up
                UpdateDisplay INGAME '                                                                     update the display with all the changes made
                Game.Pause.Die = Game.Pause.Die - 1 '                                                      decrement the death pause count down timer
                IF _KEYDOWN(27) OR _EXIT THEN ExitGame '                                                   leave the game if the player presses ESC
            LOOP UNTIL Game.Pause.Die = 0 '                                                                leave the death loop when the count down has completed
            IF Players = 2 AND (Player(1).GameOver = FALSE OR Player(2).GameOver = FALSE) THEN '           is this a 2 player game with at least 1 player still active?
                DO '                                                                                       yes, begin next player search loop
                    p = p + 1 '                                                                            increment the player number
                    IF p > 2 THEN p = 1 '                                                                  return back to player 1 if needed
                LOOP UNTIL Player(p).Ship.Dead = FALSE '                                                   leave the loop when a live player is found
                Game.Player = p '                                                                          record the new current player
                GetReady '                                                                                 tell the player to get ready
                UFO(p) = Player(p).UFO '                                                                   restore the UFO settings
                IF UFO(p).Active THEN _SNDLOOP SND.UFOFlying '                                             restart the UFO sound if the UFO happens to be flying
            END IF
        END IF
    LOOP UNTIL Player(1).GameOver AND Player(2).GameOver '                                                 leave when both players have exhaused their ships
    _KEYCLEAR '                                                                                            clear all keyboard buffers
    KeyPress = GetKey(NEWGAME) '                                                                           clear getkey buffer
    SaveOptions '                                                                                          save the options in case a new high score was achieved

END SUB
'--------------------------------------------------------------------------------------------------------------------------------+---------------------------------+--+-----------+
SUB DrawBombs () '                                                                                                               | COLLISION: Bomb and Player Ship |  | DrawBombs |
    '+---------------------------------------------------------------------------------------------------------------------------+---------------------------------+--+-----------+
    '| Manages invader bombs and collisions between bombs and the player's ship                                                                                                   |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Invader() AS INVADERS '          need access to shared variables
    SHARED Bomb() AS BOMB
    SHARED IMG_Bomb() AS LONG
    SHARED DropColumn() AS DROPCOLUMN
    SHARED IMG AS IMAGES
    SHARED Game AS GAME
    SHARED Player() AS PLAYER
    SHARED Shield() AS SHIELDS
    SHARED SND AS SOUNDS
    DIM b AS INTEGER '                      bomb counter
    DIM c AS INTEGER '                      column counter
    DIM p AS INTEGER '                      current player

    p = Game.Player '                                                                                   get current player
    DO '                                                                                                begin active bomb loop
        b = b + 1 '                                                                                     increment bomb counter
        IF Bomb(b).Active THEN '                                                                        is this bomb falling?
            IF Bomb(b).Hit.Shield THEN '                                                                yes, has the bomb hit a shield?
                Bomb(b).Hit.Shield = Bomb(b).Hit.Shield - 1 '                                           yes, decrement count down timer
                _PUTIMAGE (Bomb(b).Hit.ShieldX, Bomb(b).Hit.ShieldY), IMG.BombHit '                     show bomb explosion
                IF Bomb(b).Hit.Shield = 0 THEN Bomb(b).Active = FALSE
            ELSEIF Bomb(b).Miss THEN '                                                                  did bomb miss and hit top of screen?
                Bomb(b).Miss = Bomb(b).Miss - 1 '                                                       yes, decrement count down timer
                _PUTIMAGE (Bomb(b).rect.x1 - 1, Bomb(b).rect.y1), IMG.BombHit '                         show bomb explosion image
                IF Bomb(b).Miss = 0 THEN Bomb(b).Active = FALSE '                                       deactivate bomb when count down complete
            ELSE '                                                                                      no, bomb is stil falling
                Bomb(b).rect.y1 = Bomb(b).rect.y1 + 1 + Game.Frame MOD 2 '                              drop bomb at 90 FPS
                Bomb(b).rect.y2 = Bomb(b).rect.y2 + 1 + Game.Frame MOD 2
                Bomb(b).Cell = Bomb(b).Cell + 1 '                                                       increment animation cell
                IF Bomb(b).Cell = 4 THEN Bomb(b).Cell = 1 '                                             reset animation cell when needed
                _PUTIMAGE (Bomb(b).rect.x1, Bomb(b).rect.y1), IMG_Bomb(Bomb(b).Image, Bomb(b).Cell) '   show bomb on screen
                IF Bomb(b).rect.y1 >= 228 THEN '                                                        has bomb hit bottom of screen?
                    Bomb(b).Miss = 5 '                                                                  yes, set count down timer
                ELSE '                                                                                  no, bomb is still on the sceen
                    '******************************************************
                    '** Check for collision between bomb and player ship **
                    '******************************************************
                    IF NOT GODMODE THEN '                                                               is developer in god mode?
                        IF Player(p).Ship.Dead = FALSE THEN '                                           no, is the player ship active?
                            IF RectCollide(Player(p).Ship.rect, Bomb(b).rect) THEN '                    yes, has the bomb hit the player's ship?
                                Player(p).Ship.Dead = TRUE '                                            yes, player is dead
                                Player(p).Ship.DeadX = Player(p).Ship.rect.x1 '                         record where on screen ship was hit
                                _SNDPLAY SND.PlayerHit '                                                play ship explosion sound
                            END IF
                        END IF
                    END IF
                END IF
            END IF
        END IF
    LOOP UNTIL b = Player(p).MaxBombs '                                                                 leave when all bombs checked
    IF Game.Pause.Level THEN EXIT SUB '                                                                 leave subroutine if game is paused between levels
    DO '                                                                                                begin bomb drop column loop
        c = c + 1 '                                                                                     increment column counter
        IF DropColumn(c).Pause THEN '                                                                   is this column ready to have a bomb dropped?
            DropColumn(c).Pause = DropColumn(c).Pause - 1 '                                             no, decrement pause timer
        ELSE '                                                                                          yes, the pause period has ended
            IF DropColumn(c).Row THEN '                                                                 is there an invader in this column?
                IF INT(RND * 11) = 1 THEN '                                                             yes, should a bomb be randomly dropped?
                    b = 0 '                                                                             yes, reset bomb counter
                    DO '                                                                                begin inactive bomb search
                        b = b + 1 '                                                                     increment bomb counter
                    LOOP UNTIL Bomb(b).Active = FALSE OR b = Player(p).MaxBombs '                       leave when inactive bomb found or no inactive bombs
                    IF Bomb(b).Active = FALSE THEN '                                                    is this bomb inactive?
                        Bomb(b).Active = TRUE '                                                         yes, activate the bomb
                        Bomb(b).Image = INT(RND * 3) + 1 '                                              set a random bomb image
                        Bomb(b).Cell = 1 '                                                              reset the animation cell
                        Bomb(b).rect.x1 = Invader(p, c, DropColumn(c).Row).rect.x1 + Invader(p, c, DropColumn(c).Row).Width / 2 - 1 ' calculate bomb location on screen
                        Bomb(b).rect.y1 = Invader(p, c, DropColumn(c).Row).rect.y2 - 2
                        Bomb(b).rect.x2 = Bomb(b).rect.x1 + 2
                        Bomb(b).rect.y2 = Bomb(b).rect.y1 + 7
                        DropColumn(c).Pause = INT(RND * 15) + 55 - Player(p).Laser.Hit.Count '          reset the pause timer for this column
                    END IF
                END IF
            END IF
        END IF
    LOOP UNTIL c = 11 '                                                                                 leave when all columns checked

END SUB
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
SUB ResetBombs () '                                                                                                                                                  | ResetBombs |
    '+---------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
    '| Resets the bombs and columns status                                                                                                                                        |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED DropColumn() AS DROPCOLUMN ' need access to shared variables
    SHARED Bomb() AS BOMB
    DIM c AS INTEGER '                  column counter

    DO '                                            begin column loop
        c = c + 1 '                                 increment column counter
        DropColumn(c).Pause = 60 + INT(RND * 240) ' reset column pause timer
        IF c < 4 THEN Bomb(c) = Bomb(0) '           reset status of bomb
    LOOP UNTIL c = 11 '                             leave when all columns processed

END SUB
'-------------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------+
FUNCTION RectCollide (Rect1 AS RECT, Rect2 AS RECT) '                                                                                                               | RectCollide |
    '+--------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------+
    '| Detects a collision between two rectangular objects                                                                                                                        |
    '| Rect1 - the first set of rectangular coordinates                                                                                                                           |
    '| Rect2 - the second set of rectangular coordinates                                                                                                                          |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    RectCollide = FALSE '                        assume no collision
    IF Rect1.x2 >= Rect2.x1 THEN
        IF Rect1.x1 <= Rect2.x2 THEN
            IF Rect1.y2 >= Rect2.y1 THEN
                IF Rect1.y1 <= Rect2.y2 THEN
                    RectCollide = TRUE '         a collision has occurred
                END IF
            END IF
        END IF
    END IF

END FUNCTION
'--------------------------------------------------------------------------------------------------------------------------------------+---------------------------+--+-----------+
SUB DrawLaser () '                                                                                                                     | COLLISION: Laser and Bomb |  | DrawLaser |
    '+---------------------------------------------------------------------------------------------------------------------------------+---------------------------+--+-----------+
    '| Manages the player's laser and collisions between the laser and bombs                                                                                                      |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Player() AS PLAYER ' need access to shared variables
    SHARED SND AS SOUNDS
    SHARED IMG AS IMAGES
    SHARED Bomb() AS BOMB
    SHARED Game AS GAME
    DIM b AS INTEGER '          bomb counter
    DIM p AS INTEGER '          current player

    p = Game.Player '                                                                                   get current player
    IF NOT _KEYDOWN(32) THEN Player(p).Keydown = FALSE '                                                remember when player releases space bar
    IF Player(p).Laser.Active = FALSE THEN '                                                            is the player's laser flying?
        IF _KEYDOWN(32) AND Player(p).Keydown = FALSE THEN '                                            no, did the player press the space bar after releasing it?
            Player(p).Laser.Active = TRUE '                                                             yes, activate the laser
            Player(p).Laser.ShotsFired = Player(p).Laser.ShotsFired + 1 '                               increment the shots fired counter
            Player(p).Keydown = TRUE '                                                                  remember that player is holding the space bar down
            Player(p).Laser.rect.x1 = Player(p).Ship.rect.x1 + 6 '                                      calculate the position of the laser
            Player(p).Laser.rect.x2 = Player(p).Laser.rect.x1
            Player(p).Laser.rect.y1 = 216
            Player(p).Laser.rect.y2 = 219
            _SNDPLAY SND.Laser '                                                                        play the laser fired sound
        END IF
    ELSE '                                                                                              yes, the player's laser is active
        IF Player(p).Laser.Miss THEN '                                                                  did the player's laser hit the top of screen?
            Player(p).Laser.Miss = Player(p).Laser.Miss - 1 '                                           yes, decrement count down timer
            _PUTIMAGE (Player(p).Laser.rect.x1 - 4, 24), IMG.LaserHit '                                 show the laser explosion
            IF Player(p).Laser.Miss = 0 THEN Player(p).Laser.Active = FALSE '                           deactivate the laser when count down complete
        ELSEIF Player(p).Laser.Hit.Invader THEN '                                                       no, did the laser hit an invader?
            Player(p).Laser.Hit.Invader = Player(p).Laser.Hit.Invader - 1 '                             yes, decrement the count down timer
            _PUTIMAGE (Player(p).Laser.Hit.InvaderX, Player(p).Laser.Hit.InvaderY), IMG.InvaderHit '    show invader explosion
            IF Player(p).Laser.Hit.Invader = 0 THEN Player(p).Laser.Active = FALSE '                    deactivate the laser whn count down complete
        ELSEIF Player(p).Laser.Hit.Bomb THEN '                                                          no, did the laser hit a bomb?
            Player(p).Laser.Hit.Bomb = Player(p).Laser.Hit.Bomb - 1 '                                   yes, decrement the count down timer
            _PUTIMAGE (Player(p).Laser.Hit.BombX, Player(p).Laser.Hit.BombY), IMG.BombHit '             show bomb explosion
            IF Player(p).Laser.Hit.Bomb = 0 THEN '                                                      has the count down timer ended?
                Player(p).Laser.Active = FALSE '                                                        yes, deactivate the player's laser
            END IF
        ELSEIF Player(p).Laser.Hit.Shield THEN '                                                        no, did the laser hit a shield?
            Player(p).Laser.Hit.Shield = Player(p).Laser.Hit.Shield - 1 '                               yes, decrement the count down timer
            _PUTIMAGE (Player(p).Laser.Hit.ShieldX, Player(p).Laser.Hit.ShieldY), IMG.LaserHit '        show laser explosion
            IF Player(p).Laser.Hit.Shield = 0 THEN Player(p).Laser.Active = FALSE '                     deactivate the laser when count down complete
        ELSE '                                                                                          no, laser is still flying
            Player(p).Laser.rect.y1 = Player(p).Laser.rect.y1 - 4 '                                     move the laser upward
            Player(p).Laser.rect.y2 = Player(p).Laser.rect.y2 - 4
            LINE (Player(p).Laser.rect.x1, Player(p).Laser.rect.y1)-(Player(p).Laser.rect.x2, Player(p).Laser.rect.y2 + 3), WHITE ' draw the laser
            IF Player(p).Laser.rect.y1 = 24 THEN '                                                      did the laser hit the top of screen?
                Player(p).Laser.Miss = 10 '                                                             yes, set the count down timer
            ELSE '                                                                                      no, laser still on sreen
                '************************************************
                '** Check for collision between laser and bomb **
                '************************************************
                DO '                                                                                    begin bomb check loop
                    b = b + 1 '                                                                         increment bomb counter
                    IF Bomb(b).Active AND Player(p).Laser.Active THEN '                                 is this bomb falling and player laser active?
                        IF RectCollide(Player(p).Laser.rect, Bomb(b).rect) THEN '                       yes, did the bomb hit the laser?
                            Player(p).Laser.Hit.Bomb = 5 '                                              yes, set the count down timer
                            Player(p).Laser.Hit.BombX = Bomb(b).rect.x1 - 2 '                           record the location of the collision
                            Player(p).Laser.Hit.BombY = Bomb(b).rect.y1 + 5
                            IF INT(RND * 2) = 1 THEN Bomb(b).Active = FALSE '                           randomly deactivate the bomb
                        END IF
                    END IF
                LOOP UNTIL b = Player(p).MaxBombs OR Player(p).Laser.Hit.Bomb '                         leave when all bombs checked
            END IF
        END IF
    END IF

END SUB
'-----------------------------------------------------------------------------------------------------------------------+---------------------------------------+--+--------------+
SUB DrawInvaders () '                                                                                                   | COLLISION:Invader and Laser or Shield |  | DrawInvaders |
    '+------------------------------------------------------------------------------------------------------------------+---------------------------------------+--+--------------+
    '| Draws the active invaders to the screen and handles collisions between invaders and the player's laser and invaders and shields                                            |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Invader() AS INVADERS '     need access to shared variables
    SHARED DropColumn() AS DROPCOLUMN
    SHARED IMG_Invader() AS LONG
    SHARED Shield() AS SHIELDS
    SHARED Display AS DISPLAY
    SHARED Options AS OPTIONS
    SHARED Player() AS PLAYER
    SHARED SND AS SOUNDS
    SHARED Game AS GAME
    DIM x AS INTEGER '                 invader column counter
    DIM y AS INTEGER '                 invader row counter
    DIM s AS INTEGER '                 shield counter
    DIM shx AS INTEGER '               location of invader and shield collision
    DIM shy AS INTEGER
    DIM p AS INTEGER '                 current player

    IF Game.Pause.Level THEN EXIT SUB '                                                                leave subroutine if game paused between levels
    p = Game.Player '                                                                                  get current player
    DO '                                                                                               begin invader column loop
        x = x + 1 '                                                                                    increment column counter
        y = 0 '                                                                                        reset row counter
        DropColumn(x).Row = 0 '                                                                        assume no invaders in this column
        DO '                                                                                           begin invader row loop
            y = y + 1 '                                                                                increment row counter
            IF Invader(p, x, y).Active THEN '                                                          is this invader active?
                _PUTIMAGE (Invader(p, x, y).rect.x1, Invader(p, x, y).rect.y1), IMG_Invader(Invader(p, x, y).Image, Invader(p, x, y).cell) ' yes, draw the invader
                IF y > DropColumn(x).Row THEN DropColumn(x).Row = y '                                  record the lowest invader in the column
                '********************************************
                '** Check for invader and shield collision **
                '********************************************
                IF Invader(p, x, y).rect.y1 > 191 AND Invader(p, x, y).rect.y1 < 209 THEN '            is the invader in the shield area?
                    s = 0 '                                                                            yes, reset shield counter
                    DO '                                                                               begin shield loop
                        s = s + 1 '                                                                    increment shield counter
                        IF RectCollide(Invader(p, x, y).rect, Shield(p, s).rect) THEN '                is the invader colliding with a shield?
                            shx = Invader(p, x, y).rect.x1 - Shield(p, s).rect.x1 '                    yes, record the location of the collision
                            shy = Invader(p, x, y).rect.y1 - 192
                            _DEST Shield(p, s).Image '                                                 draw on the shield image
                            LINE (shx, shy)-(shx + Invader(p, x, y).Width - 1, shy + 7), BLACK, BF '   remove portion of shield where collision occurring
                            _DEST Display.WorkScreen '                                                 return to drawing on work display
                        END IF
                    LOOP UNTIL s = Options.Shields '                                                   leave when all shields have been checked
                ELSEIF Invader(p, x, y).rect.y1 = 216 THEN '                                           has the invader landed at ship location?
                    '**************************************
                    '** Invader reached bottom of screen **
                    '**************************************
                    Game.Landed = TRUE '                                                               yes, remember that invader has landed
                END IF
                '*******************************************
                '** Check for invader and laser collision **
                '*******************************************
                IF Player(p).Laser.Active THEN '                                                       is the player's laser flying?
                    IF RectCollide(Player(p).Laser.rect, Invader(p, x, y).rect) THEN '                 yes, has the laser collided with an invader?
                        Player(p).Score = Player(p).Score + Invader(p, x, y).Score '                   yes, add the invader's score to the player's score
                        Player(p).Laser.Hit.Count = Player(p).Laser.Hit.Count + 1 '                    increment the player's invader hit counter
                        IF Player(p).Laser.Hit.Count = 27 THEN '                                       have half the invaders been destroyed?
                            IF Player(p).MaxBombs < 3 THEN '                                           yes, are the maximum number of bombs dropping?
                                Player(p).MaxBombs = Player(p).MaxBombs + 1 '                          no, increase the amount of bombs invaders allowed to drop
                            END IF
                        END IF
                        Player(p).Laser.Hit.Invader = 5 '                                              set count down timer
                        Player(p).Laser.Hit.InvaderX = Invader(p, x, y).rect.x1 - ((14 - Invader(p, x, y).Width) \ 2) ' record where the collision occurred
                        Player(p).Laser.Hit.InvaderY = Invader(p, x, y).rect.y1
                        Invader(p, x, y).Active = FALSE '                                              deactivate this invader
                        _SNDPLAY SND.InvaderHit '                                                      play the invader explosion sound
                    END IF
                END IF
            END IF
        LOOP UNTIL y = 5 '                                                                             leave when all rows checked
    LOOP UNTIL x = 11 '                                                                                leave whan all columns checked

END SUB
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
SUB DrawShip () '                                                                                                                                                      | DrawShip |
    '+-----------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
    '| Manages the player's ship                                                                                                                                                  |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED IMG_Ship() AS LONG ' need access to shared variables
    SHARED Player() AS PLAYER
    SHARED Game AS GAME
    DIM p AS INTEGER '          current player

    p = Game.Player '                                                                                  get current player
    IF Game.Pause.Die THEN '                                                                           is a death pause happening?
        IF Game.Frame MOD 5 = 0 THEN Player(p).Ship.DeadImage = -Player(p).Ship.DeadImage '            yes, toggle the death image every 5 frames
        _PUTIMAGE (Player(p).Ship.DeadX, Player(p).Ship.rect.y1), IMG_Ship(Player(p).Ship.DeadImage) ' draw the ship death image
        IF Game.Pause.Die = 1 THEN '                                                                   is this the last frame of the death pause?
            Player(p).Ship.rect.x1 = 15 '                                                              yes, reset player's ship location
            Player(p).Ship.rect.x2 = 27
        END IF
    ELSE '                                                                                             no, player is still alive
        IF _KEYDOWN(19712) THEN '                                                                      is player pressing the right arrow key?
            Player(p).Ship.rect.x1 = Player(p).Ship.rect.x1 + 1 '                                      yes, move ship to the right
            Player(p).Ship.rect.x2 = Player(p).Ship.rect.x2 + 1
        END IF
        IF _KEYDOWN(19200) THEN '                                                                      is player pressing the left arrow key?
            Player(p).Ship.rect.x1 = Player(p).Ship.rect.x1 - 1 '                                      yes, move ship to the left
            Player(p).Ship.rect.x2 = Player(p).Ship.rect.x2 - 1
        END IF
        IF Player(p).Ship.rect.x1 < 15 THEN '                                                          is the ship moving too far left?
            Player(p).Ship.rect.x1 = 15 '                                                              yes, hold ship at left side of screen
            Player(p).Ship.rect.x2 = 27
        END IF
        IF Player(p).Ship.rect.x2 > 207 THEN '                                                         is the ship moving too far right?
            Player(p).Ship.rect.x1 = 194 '                                                             yes, hold ship at right side of screen
            Player(p).Ship.rect.x2 = 207
        END IF
        _PUTIMAGE (Player(p).Ship.rect.x1, Player(p).Ship.rect.y1), IMG_Ship(0) '                      draw player ship
    END IF

END SUB
'-----------------------------------------------------------------------------------------------------------------------------------------+--------------------------+--+---------+
SUB DrawUFO () '                                                                                                                          | COLLISION: Laser and UFO |  | DrawUFO |
    '+------------------------------------------------------------------------------------------------------------------------------------+--------------------------+--+---------+
    '| Manages the UFO and collisions between the UFO and player's laser                                                                                                          |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Player() AS PLAYER ' need access to shared variables
    SHARED UFO() AS UFO
    SHARED Game AS GAME
    SHARED IMG AS IMAGES
    SHARED SND AS SOUNDS
    DIM p AS INTEGER '          current player

    p = Game.Player '                                                             get current player
    IF Player(p).Laser.Hit.UFO THEN '                                             did the player's laser hit the UFO?
        Player(p).Laser.Hit.UFO = Player(p).Laser.Hit.UFO - 1 '                   yes, decrement count down timer
        Text Player(p).Laser.Hit.UFOX, 32, _TRIM$(STR$(UFO(p).Score)) '           display the UFO score where UFO was hit
    END IF
    IF UFO(p).Active THEN '                                                       is the UFO flying across the screen?
        IF Game.Frame MOD 4 THEN '                                                yes, skip every 4th frame
            UFO(p).rect.x1 = UFO(p).rect.x1 + UFO(p).Dir '                        move UFO at 45 FPS
            UFO(p).rect.x2 = UFO(p).rect.x1 + 15
        END IF
        IF UFO(p).rect.x1 > 199 OR UFO(p).rect.x1 < 8 THEN '                      has the UFO reached the edge of screen?
            _SNDSTOP SND.UFOFlying '                                              yes, stop the UFO sound
            UFO(p).Active = FALSE '                                               deactivate the UFO
        ELSE '                                                                    no, UFO is still flying
            _PUTIMAGE (UFO(p).rect.x1, UFO(p).rect.y1), IMG.UFO '                 draw the UFO
            '***********************************************
            '** Check for collision between laser and UFO **
            '***********************************************
            IF Player(p).Laser.Active THEN '                                      is the player's laser active?
                IF RectCollide(Player(p).Laser.rect, UFO(p).rect) THEN '          has the player's laser hit the UFO?
                    SELECT CASE INT(RND * 3) + 1 '                                yes, set a random score value
                        CASE 1
                            UFO(p).Score = 50 '                                   50 points
                        CASE 2
                            UFO(p).Score = 100 '                                  100 points
                        CASE 3
                            UFO(p).Score = 200 '                                  200 points
                    END SELECT
                    IF Player(p).Laser.ShotsFired = 23 THEN UFO(p).Score = 300 '  300 points if the player has fired 23 lasers
                    IF Player(p).Laser.ShotsFired > 23 THEN '                     has the player fired more than 23 lasers?
                        IF (Player(p).Laser.ShotsFired - 23) MOD 15 = 0 THEN '    every 15 laser firings afterwards?
                            UFO(p).Score = 300 '                                  yes, UFO score is again 300 points
                        END IF
                    END IF
                    Player(p).Score = Player(p).Score + UFO(p).Score '            add the UFO score to the player's score
                    Player(p).Laser.Hit.UFO = 60 '                                set the count down timer
                    Player(p).Laser.Hit.UFOX = UFO(p).rect.x1 + 32 '              record where the UFO was hit
                    UFO(p).Active = FALSE '                                       deactivate the UFO
                    Player(p).Laser.Active = FALSE '                              deactivate the player's laser
                    _SNDSTOP SND.UFOFlying '                                      stop the UFO sound
                    _SNDPLAY SND.UFOHit '                                         play the UFO explosion sound
                END IF
            END IF
        END IF
    ELSE '                                                                        no, UFO is currently inactive
        UFO(p).Pause = UFO(p).Pause - 1 '                                         decrement the UFO pause timer
        IF UFO(p).Pause = 0 THEN '                                                has the timer ended?
            _SNDLOOP SND.UFOFlying '                                              yes, play the UFO flying sound
            UFO(p).Active = TRUE '                                                activate the UFO
            UFO(p).Pause = 1500 + INT(RND * 120) - INT(RND * 120) '               reset the UFO pause for 23 to 27 seconds
            IF INT(RND * 2) = 1 THEN '                                            which direction should UFO come from?
                UFO(p).Dir = 1 '                                                  UFO will travel left to right
                UFO(p).rect.x1 = 8 '                                              position UFO at left of screen
            ELSE
                UFO(p).Dir = -1 '                                                 UFO will travel right to left
                UFO(p).rect.x1 = 199 '                                            position UFO at right of screen
            END IF
        END IF
    END IF

END SUB
'------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
SUB MoveInvaders (p AS INTEGER, Mode AS INTEGER) STATIC '                                                                                                          | MoveInvaders |
    '+-------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
    '| Manages the wave motion of invaders across the screen                                                                                                                      |
    '| p    - current player                                                                                                                                                      |
    '| Mode - signal to clear variables for a movement reset (CLEARVARIABLES or -1)                                                                                               |
    '| NOTE: This subroutine retains values between calls (STATIC)                                                                                                                |
    '|                                                                                                                                                                            |
    '| Moves the invaders across the screen by simulating the "strobing" effect of an Intel 8080 barely able to keep up with the graphics. Only one invader is updated per frame  |
    '| of the game to achieve the slow CPU effect. As fewer invaders are alive this subroutine will simulate the speeding up effect of the invaders as if an 8080 CPU has less    |
    '| work to do.                                                                                                                                                                |
    '|                                                                                                                                                                            |
    '| From: https://en.wikipedia.org/wiki/Space_Invaders                                                                                                                         |
    '|                                                                                                                                                                            |
    '| "While programming the game, Nishikado discovered that the processor was able to render the alien graphics faster the fewer were on screen. Rather than design the game to |
    '|  compensate for the speed increase, he decided to keep it as a challenging gameplay mechanic."                                                                             |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Invader() AS INVADERS ' need access to shared variables
    SHARED Player() AS PLAYER
    SHARED SND AS SOUNDS
    SHARED Game AS GAME
    DIM c(2) AS INTEGER '          column counter
    DIM r(2) AS INTEGER '          row counter
    DIM Down(2) AS INTEGER '       invader to move down flag
    DIM Edge(2) AS INTEGER '       invader hit edge of screen flag
    DIM Beat(2) AS INTEGER '       current heart throb sound to play

    IF Mode = CLEARVARIABLES OR Game.Pause.Level THEN '                                                       time to reset the movement variables?
        c(p) = 0 '                                                                                            yes, reset all movement variables
        r(p) = 0
        Down(p) = FALSE
        Edge(p) = FALSE
        Beat(p) = 0
        EXIT SUB '                                                                                            leave the subroutine
    END IF
    DO '                                                                                                      begin active invader search loop
        c(p) = c(p) + 1 '                                                                                     increment column counter
        IF c(p) = 12 THEN '                                                                                   has the last column been reached?
            c(p) = 1 '                                                                                        yes, reset the column counter
            r(p) = r(p) - 1 '                                                                                 decrement the row counter
        END IF
        IF r(p) = 0 THEN '                                                                                    has the top row been reached?
            r(p) = 5 '                                                                                        yes, reset the row counter
            IF Edge(p) THEN '                                                                                 has the invader reached the edge of the screen?
                Down(p) = TRUE '                                                                              yes, flag the invader for a downward movement
                Edge(p) = FALSE '                                                                             reset the edge detection flag
            ELSEIF Down(p) THEN '                                                                             no, has the invader been flagged for a downward movement?
                Down(p) = FALSE '                                                                             yes, reset the downward movement flag
                Player(p).idir = -Player(p).idir '                                                            reverse the direction of the invaders
            END IF
            Beat(p) = Beat(p) + 1 '                                                                           increment the heart throb sound counter
            IF Beat(p) = 5 THEN Beat(p) = 1 '                                                                 reset the counter when needed
            SELECT CASE Beat(p) '                                                                             which sound to play?
                CASE 1
                    _SNDPLAY SND.Beat1
                CASE 2
                    _SNDPLAY SND.Beat2
                CASE 3
                    _SNDPLAY SND.Beat3
                CASE 4
                    _SNDPLAY SND.Beat4
            END SELECT
        END IF
    LOOP UNTIL Invader(p, c(p), r(p)).Active '                                                                leave when an active invader found
    Invader(p, c(p), r(p)).cell = 1 - Invader(p, c(p), r(p)).cell '                                           toggle the invader animation cell
    IF Down(p) THEN '                                                                                         is this invader flagged for downward movement?
        Invader(p, c(p), r(p)).rect.y1 = Invader(p, c(p), r(p)).rect.y1 + 8 '                                 yes, move the invader down one row
        Invader(p, c(p), r(p)).rect.y2 = Invader(p, c(p), r(p)).rect.y2 + 8
    ELSE '                                                                                                    no, move the invader right or left
        Invader(p, c(p), r(p)).rect.x1 = Invader(p, c(p), r(p)).rect.x1 + Player(p).idir '                    move the invader horizontally across screen
        Invader(p, c(p), r(p)).rect.x2 = Invader(p, c(p), r(p)).rect.x2 + Player(p).idir
        IF Invader(p, c(p), r(p)).rect.x1 <= 8 OR Invader(p, c(p), r(p)).rect.x2 >= 215 THEN Edge(p) = TRUE ' set the edge flag when an invader reaches side of screen
    END IF

END SUB
'-----------------------------------------------------------------------------------------------------------------------------------------------------------------+---------------+
SUB ResetInvaders (p AS INTEGER) '                                                                                                                                | ResetInvaders |
    '+------------------------------------------------------------------------------------------------------------------------------------------------------------+---------------+
    '| Resets the invader start positions for the current level of the player                                                                                                     |
    '| p - player to reset invaders for                                                                                                                                           |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Invader() AS INVADERS ' need access to shared variables
    SHARED Player() AS PLAYER
    DIM r AS INTEGER '             row counter
    DIM c AS INTEGER '             column counter
    DIM Level AS INTEGER '         current player level

    Level = Player(p).Level '                                                                   get current player level
    IF Level > 5 THEN Level = 5 '                                                               keep the level at 5
    DO '                                                                                        begin invader row loop
        r = r + 1 '                                                                             increment row counter
        c = 0 '                                                                                 reset column counter
        DO '                                                                                    begin invader column loop
            c = c + 1 '                                                                         increment column counter
            Invader(p, c, r).Active = TRUE '                                                    activate this invader
            Invader(p, c, r).rect.x1 = (16 * c) + 8 - Invader(p, c, r).Width \ 2 '              calculate position of invader on screen
            Invader(p, c, r).rect.x2 = Invader(p, c, r).rect.x1 + Invader(p, c, r).Width - 1
            Invader(p, c, r).rect.y1 = (32 + (r * 16)) + Level * 16
            Invader(p, c, r).rect.y2 = Invader(p, c, r).rect.y1 + 7
            Invader(p, c, r).cell = 0 '                                                         reset invader animation cell
        LOOP UNTIL c = 11 '                                                                     leave when all columns checked
    LOOP UNTIL r = 5 '                                                                          leave when all rows checked

END SUB
'--------------------------------------------------------------------------------------------------------------------------+-------------------------------------+--+-------------+
SUB DrawShields (Mode AS INTEGER) '                                                                                        | COLLISION: Shield and Laser or Bomb |  | DrawShields |
    '+---------------------------------------------------------------------------------------------------------------------+-------------------------------------+--+-------------+
    '| Draws the player's shields to the screen and handles collisions between the shields and lasers or bombs                                                                    |
    '| Mode - How to handle drawing of the shields                                                                                                                                |
    '|        -1 (NEWGAME)  - reset the shields for both players                                                                                                                  |
    '|         0 (NEWLEVEL) - reset the shields for the current player                                                                                                            |
    '|         5 (INGAME)   - manage the shields during game play                                                                                                                 |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+

    SHARED Options AS OPTIONS '  need access to shared variables
    SHARED IMG AS IMAGES
    SHARED Shield() AS SHIELDS
    SHARED Player() AS PLAYER
    SHARED Display AS DISPLAY
    SHARED Bomb() AS BOMB
    SHARED Game AS GAME
    DIM x AS INTEGER '            shield counter
    DIM p AS INTEGER '            current player
    DIM lx AS INTEGER '           location of laser hitting shield
    DIM ly AS INTEGER
    DIM scan AS INTEGER '         pixel perfect scanner
    DIM rndx AS INTEGER '         random variance added to hit locations
    DIM rndy AS INTEGER
    DIM b AS INTEGER '            bomb counter
    DIM shx AS INTEGER '          location of hit on shield
    DIM shy AS INTEGER

    p = Game.Player '                                                                                     get current player
    x = 1 '                                                                                               set shield counter
    DO '                                                                                                  begin shield loop
        SELECT CASE Mode '                                                                                which mode was requested?
            CASE NEWGAME '                                                                                a new game is beginning
                _PUTIMAGE , IMG.Shield, Shield(PLAYER1, x).Image '                                        reset the shield images with undamaged images
                _PUTIMAGE , IMG.Shield, Shield(PLAYER2, x).Image
            CASE NEWLEVEL '                                                                               a new level is beginning for the current player
                _PUTIMAGE , IMG.Shield, Shield(p, x).Image '                                              reset the shield images with undamaged images
            CASE INGAME '                                                                                 a game level is currently being played
                _SETALPHA 0, BLACK, Shield(p, x).Image '                                                  set the transparency color of shields
                _PUTIMAGE (Shield(p, x).rect.x1, 192), Shield(p, x).Image '                               draw the player's shield to the screen
                '******************************************
                '** Check for laser and shield collision **
                '******************************************
                IF Player(p).Laser.Active AND Player(p).Laser.Hit.Shield = 0 THEN '                       if the player's laser flying and didn't hit a shield?
                    IF RectCollide(Player(p).Laser.rect, Shield(p, x).rect) THEN '                        yes, did the laser hit the shield?
                        lx = Player(p).Laser.rect.x1 - Shield(p, x).rect.x1 '                             yes, remember location of collision
                        ly = Player(p).Laser.rect.y1 - 192
                        scan = 0 '                                                                        reset pixel perfect scanner
                        _SOURCE Shield(p, x).Image '                                                      work with shield image
                        _DEST Shield(p, x).Image
                        DO '                                                                              begin pixel perfect collision loop
                            IF POINT(lx, ly + scan) = WHITE THEN '                                        did this part of laser hit the shiled?
                                rndx = INT(RND * 2) - INT(RND * 2) '                                      yes, set some random variance in collision location
                                rndy = INT(RND * 2)
                                Player(p).Laser.Hit.ShieldX = Player(p).Laser.rect.x1 - 4 + rndx '        record location of collision on screen
                                Player(p).Laser.Hit.ShieldY = Player(p).Laser.rect.y1 - 3 + scan + rndy '
                                _PUTIMAGE (lx - 4 + rndx, ly - 3 + scan + rndy), IMG.LaserHitMask '       damage the shields
                                Player(p).Laser.Hit.Shield = 5 '                                          set count down timer
                            END IF
                            scan = scan + 1 '                                                             move to next pixel location on laser
                        LOOP UNTIL scan = 4 OR Player(p).Laser.Hit.Shield '                               leave when laser length scanned or a hit on shield occurred
                        _SOURCE Display.WorkScreen '                                                      return back to the work display
                        _DEST Display.WorkScreen
                    END IF
                END IF
                '*****************************************
                '** Check for bomb and shield collision **
                '*****************************************
                b = 0 '                                                                                   reset bomb counter
                DO '                                                                                      begin bomb loop
                    b = b + 1 '                                                                           increment bomb counter
                    IF Bomb(b).Active AND Bomb(b).Hit.Shield = 0 THEN '                                   is this bomb dropping and not hit a shield?
                        IF RectCollide(Bomb(b).rect, Shield(p, x).rect) THEN '                            yes, has the bomb hit a shield?
                            shx = Bomb(b).rect.x1 - Shield(p, x).rect.x1 '                                yes, record location of collision
                            shy = Bomb(b).rect.y1 - 192
                            _SOURCE Shield(p, x).Image '                                                  work with shield image
                            _DEST Shield(p, x).Image
                            scan = -1 '                                                                   reset pixel perfect scanner
                            DO '                                                                          begin pixel perfect collision loop
                                IF POINT(shx + scan, shy) = WHITE THEN '                                  did this part of bomb hit the shield?
                                    rndy = INT(RND * 3) '                                                 yes, set som random variance in collision location
                                    Bomb(b).Hit.ShieldX = Bomb(b).rect.x1 - 2 '                           record screen location of collision
                                    Bomb(b).Hit.ShieldY = Bomb(b).rect.y1 - 4 + scan + rndy
                                    _PUTIMAGE (shx - 2, shy - 4 + scan + rndy), IMG.BombHitMask '         damage the shields
                                    Bomb(b).Hit.Shield = 5 '                                              set countdown timer
                                END IF
                                scan = scan + 1 '                                                         move to next pixel location on bomb
                            LOOP UNTIL scan = 2 OR Bomb(b).Hit.Shield '                                   leave when bomb scanned or a hit on shield occurred
                            _SOURCE Display.WorkScreen '                                                  return back to work display
                            _DEST Display.WorkScreen
                        END IF
                    END IF
                LOOP UNTIL b = Player(p).MaxBombs '                                                       leave when all bombs checked
        END SELECT
        x = x + 1 '                                                                                       increment shield counter
    LOOP UNTIL x > Options.Shields '                                                                      leave when all shields checked

END SUB
'-----------------------------------------------------------------------------------------------------------------------------------------------------------------+---------------+
SUB StartNewLevel () '                                                                                                                                            | StartNewLevel |
    '+------------------------------------------------------------------------------------------------------------------------------------------------------------+---------------+
    '| Resets the variables in preparation for a new player level                                                                                                                 |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Player() AS PLAYER '         need access to shared variables
    SHARED DropColumn() AS DROPCOLUMN
    SHARED Game AS GAME
    DIM p AS INTEGER '                  current player

    p = Game.Player '                                        get current player
    Player(p).Level = Player(p).Level + 1 '                  increment player level
    IF Player(p).Level = 11 THEN Player(p).Level = 1 '       reset player level when level 11 reached
    Player(p).idir = 2 '                                     reset invader movement
    Player(p).Laser.Hit.Count = 0 '                          reset invader hit count
    Player(p).MaxBombs = Player(p).Level '                   calculate maximum number of invader bombs allowed
    IF Player(p).MaxBombs > 3 THEN Player(p).MaxBombs = 3 '  no more than 3 bombs allowed
    MoveInvaders p, CLEARVARIABLES '                         clear the invader movement variables
    ResetInvaders p '                                        reset invader locations

END SUB
'------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
SUB StartNewGame () '                                                                                                                                              | StartNewGame |
    '+-------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
    '| Resets the variables in preparation for a new game                                                                                                                         |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED DropColumn() AS DROPCOLUMN ' need access to shared variables
    SHARED Player() AS PLAYER
    SHARED UFO() AS UFO
    SHARED Game AS GAME

    RANDOMIZE TIMER '                      seed the RND generator
    Game.Player = 1 '                      set the current player to player 1
    Game.Landed = FALSE '                  reset the invader landed flag
    Player(1) = Player(0) '                reset player variables
    Player(2) = Player(0)
    UFO(1) = UFO(0) '                      reset UFO variables
    UFO(2) = UFO(0)
    DrawShields NEWGAME '                  restore the shield images
    ResetInvaders PLAYER1 '                reset invader locations
    ResetInvaders PLAYER2
    MoveInvaders PLAYER1, CLEARVARIABLES ' clear the invader movement variables
    MoveInvaders PLAYER2, CLEARVARIABLES
    ResetBombs
    '                                      reset the bombs and drop columns
END SUB
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
SUB Initialize () '                                                                                                                                                  | Initialize |
    '+---------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
    '+ Initializes all variables upon power up                                                                                                                                    |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Player() AS PLAYER '    need access to shared variables
    SHARED Invader() AS INVADERS
    SHARED Shield() AS SHIELDS
    SHARED UFO() AS UFO
    SHARED Bomb() AS BOMB
    SHARED Options AS OPTIONS
    SHARED Game AS GAME
    DIM p AS INTEGER '             player counter
    DIM c AS INTEGER '             column counter
    DIM r AS INTEGER '             row counter

    Game.Credits = 0 '                                  set in game settings
    Game.Player = 1
    Game.Players = 1
    Game.Pause.Level = 0
    Game.Pause.Die = 0
    Game.Landed = FALSE
    Bomb(0).rect.x1 = 0 '                               set default bomb settings
    Bomb(0).rect.y1 = 0
    Bomb(0).rect.x2 = 0
    Bomb(0).rect.y2 = 0
    Bomb(0).Hit.Shield = 0
    Bomb(0).Hit.ShieldX = 0
    Bomb(0).Hit.ShieldY = 0
    Bomb(0).Image = 1
    Bomb(0).Cell = 1
    Bomb(0).Active = FALSE
    Bomb(0).Miss = 0
    Bomb(1) = Bomb(0) '                                 set bomb settings to default
    Bomb(2) = Bomb(0)
    Bomb(3) = Bomb(0)
    UFO(0).Score = 0 '                                  set default UFO settings
    UFO(0).Pause = 1500
    UFO(0).Active = FALSE
    UFO(0).Dir = 1
    UFO(0).rect.x1 = 8
    UFO(0).rect.y1 = 32
    UFO(0).rect.y2 = 39
    UFO(1) = UFO(0) '                                   set UFO settings to default
    UFO(2) = UFO(0)
    Player(0).Ship.rect.y1 = 216 '                      set default player settings
    Player(0).Ship.rect.y2 = 223
    Player(0).Ship.Dead = FALSE
    Player(0).Ship.DeadX = 0
    Player(0).Ship.DeadImage = 1
    Player(0).Ship.Extra = FALSE
    Player(0).Score = 0
    Player(0).Ship.Remain = 3
    Player(0).idir = 2
    Player(0).Level = 1
    Player(0).GameOver = FALSE
    Player(0).MaxBombs = 1
    Player(0).Laser.Active = FALSE
    Player(0).Laser.ShotsFired = 0
    Player(0).Laser.Hit.Count = 0
    Player(0).Laser.Miss = 0
    Player(0).Laser.Hit.Invader = 0
    Player(0).Laser.Hit.InvaderX = 0
    Player(0).Laser.Hit.InvaderY = 0
    Player(0).Laser.Hit.Shield = 0
    Player(0).Laser.Hit.ShieldX = 0
    Player(0).Laser.Hit.ShieldY = 0
    Player(0).Laser.Hit.Bomb = 0
    Player(0).Laser.Hit.BombX = 0
    Player(0).Laser.Hit.BombY = 0
    Player(0).UFO = UFO(0)
    Player(1) = Player(0) '                             set player settings to default
    Player(2) = Player(0)
    RANDOMIZE TIMER '                                   seed the RND generator
    DO '                                                begin player loop
        p = p + 1 '                                     increment player counter
        r = 0 '                                         reset row counter
        DO '                                            begin invader row loop
            r = r + 1 '                                 increment row counter
            c = 0 '                                     reset column counter
            DO '                                        begin invader column loop
                c = c + 1 '                             increment column counter
                SELECT CASE r '                         which row?
                    CASE 1 '                            row 1 (top row)
                        Invader(p, c, r).Image = 1 '    set invader image
                        Invader(p, c, r).Width = 8 '    set invader width
                        Invader(p, c, r).Score = 30 '   set invader score
                    CASE 2 TO 3 '                       rows 2 and 3
                        Invader(p, c, r).Image = 2
                        Invader(p, c, r).Width = 11
                        Invader(p, c, r).Score = 20
                    CASE 4 TO 5 '                       rows 4 and 5 (bottom row)
                        Invader(p, c, r).Image = 3
                        Invader(p, c, r).Width = 12
                        Invader(p, c, r).Score = 10
                END SELECT
            LOOP UNTIL c = 11 '                         leave when all columns processed
        LOOP UNTIL r = 5 '                              leave when all rows processed
        Shield(p, 1).rect.x1 = 32 '                     set first shield screen location
        Shield(p, 1).rect.x2 = 57
        SELECT CASE Options.Shields '                   how many shields set in options?
            CASE 3 '                                    3 shields
                Shield(p, 2).rect.x1 = 99 '             2nd shield location
                Shield(p, 2).rect.x2 = 124
                Shield(p, 3).rect.x1 = 166 '            3rd shield location
                Shield(p, 3).rect.x2 = 191
            CASE 4 '                                    4 shields (default)
                Shield(p, 2).rect.x1 = 77 '             2nd shield location
                Shield(p, 2).rect.x2 = 102
                Shield(p, 3).rect.x1 = 121 '            3rd shield location
                Shield(p, 3).rect.x2 = 146
                Shield(p, 4).rect.x1 = 166 '            4th shield location
                Shield(p, 4).rect.x2 = 191
            CASE 5 '                                    5 shields
                Shield(p, 2).rect.x1 = 66 '             2nd shield location
                Shield(p, 2).rect.x2 = 91
                Shield(p, 3).rect.x1 = 99 '             3rd shield location
                Shield(p, 3).rect.x2 = 124
                Shield(p, 4).rect.x1 = 132 '            4th shield location
                Shield(p, 4).rect.x2 = 157
                Shield(p, 5).rect.x1 = 166 '            5th shield location
                Shield(p, 5).rect.x2 = 191
            CASE 6 '                                    6 shields
                Shield(p, 2).rect.x1 = 59 '             2nd shield location
                Shield(p, 2).rect.x2 = 84
                Shield(p, 3).rect.x1 = 86 '             3rd shield location
                Shield(p, 3).rect.x2 = 111
                Shield(p, 4).rect.x1 = 113 '            4th shield location
                Shield(p, 4).rect.x2 = 138
                Shield(p, 5).rect.x1 = 140 '            5th shield location
                Shield(p, 5).rect.x2 = 165
                Shield(p, 6).rect.x1 = 167 '            6th shield location
                Shield(p, 6).rect.x2 = 192
        END SELECT
        Shield(p, 1).rect.y1 = 192 '                    set all shield Y locations
        Shield(p, 1).rect.y2 = 207
        Shield(p, 2).rect.y1 = 192
        Shield(p, 2).rect.y2 = 207
        Shield(p, 3).rect.y1 = 192
        Shield(p, 3).rect.y2 = 207
        Shield(p, 4).rect.y1 = 192
        Shield(p, 4).rect.y2 = 207
        Shield(p, 5).rect.y1 = 192
        Shield(p, 5).rect.y2 = 207
        Shield(p, 6).rect.y1 = 192
        Shield(p, 6).rect.y2 = 207
    LOOP UNTIL p = 2 '                                  leave when both players processed

END SUB
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
SUB GetReady () '                                                                                                                                                      | GetReady |
    '+-----------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
    '| Informs the player to get ready to play a game or take turns in a 2 player game                                                                                            |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Game AS GAME '  need access to shared variables
    DIM Blink AS INTEGER ' blink toggle
    DIM Pause AS INTEGER ' pause counter

    Pause = 180 '                                                                              set 3 second pause
    DO '                                                                                       begin pause loop
        _LIMIT 60 '                                                                            60 frames per second
        ClearDisplay INCOIN '                                                                  clear the display
        IF Game.Frame MOD 5 = 0 THEN Blink = NOT Blink '                                       toggle blink flag every 5 frames
        Text 13, 7, "PLAY PLAYER<" + _TRIM$(STR$(Game.Player)) + ">" '                         print player notice
        IF Blink THEN DrawScore Game.Player, INCOIN '                                          draw the score when flag set
        IF Game.Players = 2 THEN '                                                             is this a 2 player game?
            IF Game.Player = 1 THEN DrawScore PLAYER2, INCOIN ELSE DrawScore PLAYER1, INCOIN ' yes, draw the other player's score non-blinking
        END IF
        UpdateDisplay INCOIN '                                                                 update the display with changes
        Pause = Pause - 1 '                                                                    decrement count down timer
    LOOP UNTIL Pause = 0 '                                                                     leave when timer ended

END SUB
'-----------------------------------------------------------------------------------------------------------------------------------------------------------------+---------------+
SUB SelectPlayers () '                                                                                                                                            | SelectPlayers |
    '+------------------------------------------------------------------------------------------------------------------------------------------------------------+---------------+
    '| Allows the player to choose the number of players when a coin has been inserted into the game                                                                              |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Game AS GAME '     need access to shared variables
    DIM KeyPress AS INTEGER ' any key pressed

    DO '                                               begin select loop
        _LIMIT 30 '                                    30 frames per second
        ClearDisplay INSELECT '                        clear the display
        DrawScore BOTHPLAYERS, INSELECT '              draw the score of both players
        DrawCredits '                                  draw number of credits in machine
        DrawShipsRemaining '                           draw the number of ships remaining for player 1
        Text 12, 12, "PUSH" '                          display text
        IF Game.Credits = 1 THEN '                     only 1 coin inserted?
            Text 14, 4, "ONLY <1>PLAYER BUTTON" '      yes, display appropriate text
        ELSE '                                         no, more than 1 coin
            Text 14, 2, "<1> OR <2>PLAYERS BUTTON" '   display appropriate text
        END IF
        UpdateDisplay INSELECT '                       update the display with changes
        KeyPress = GetKey(INSELECT) '                  get any key that may have been pressed
        IF KeyPress = 49 THEN '                        did player press the 1 key?
            Game.Players = 1 '                         yes, set the number of players
            Game.Credits = Game.Credits - 1 '          subtract a credit from the game
            PlayGame PLAYER1 '                         player a 1 player game
        ELSEIF KeyPress = 50 THEN '                    no, did player press the 2 key?
            IF Game.Credits > 1 THEN '                 yes, is there more than 1 credit in game?
                Game.Players = 2 '                     yes, set number of players
                Game.Credits = Game.Credits - 2 '      subtract 2 credits from game
                PlayGame PLAYER2 '                     play a two player game
            END IF
        ELSEIF KeyPress = 79 THEN '                    did player press the O key?
            SetOptions '                               yes, go to set options screen
        END IF
    LOOP UNTIL Game.Credits = 0 '                      leave when all credits have been used

END SUB
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
SUB SlowText (Row AS INTEGER, Column AS INTEGER, Txt AS STRING, Mode AS INTEGER, KeyPress AS INTEGER) '                                                                | SlowText |
    '+-----------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
    '| Displays text on screen slowly at 1/10th second per letter                                                                                                                 |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Font() AS LONG ' need access to shared variables
    SHARED Game AS GAME
    DIM c AS INTEGER '      column counter
    DIM r AS INTEGER '      row counter
    DIM p AS INTEGER '      text character position counter

    r = Row * 8 '                                          yes, calculate text row
    c = Column * 8 '                                       calculate text column
    p = 0 '                                                reset character position counter
    DO '                                                   begin text loop
        _LIMIT 60 '                                        60 frames per second
        KeyPress = GetKey(Mode) '                          get any key pressed
        IF Game.Frame MOD 6 = 0 THEN '                     have 6 frames gone by? (1/10th second)
            p = p + 1 '                                    yes, increment character position counter
            _PUTIMAGE (c, r), Font(ASC(MID$(Txt, p, 1))) ' draw font character onto screen
            c = c + 8 '                                    move to next text column
        END IF
        UpdateDisplay Mode '                               update the display with changes
    LOOP UNTIL p = LEN(Txt) OR KeyPress '                  leave when text finished or a valid key pressed

END SUB
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
SUB InsertCoin () '                                                                                                                                                  | InsertCoin |
    '+---------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
    '| Plays the various intro screens while waiting for a coin to be inserted                                                                                                    |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Font() AS LONG '        need access to shared variables
    SHARED IMG AS IMAGES
    SHARED IMG_Invader() AS LONG
    SHARED IMG_Bomb() AS LONG
    SHARED Display AS DISPLAY
    SHARED Game AS GAME
    DIM FlippedY AS INTEGER '      toggle used to flip between normal Y and flipped Y screens
    DIM AddedC AS INTEGER '        toggle used to flip between normal C and added C screens
    DIM KeyPress AS INTEGER '      contains any key pressed by player
    DIM Bcycle AS INTEGER '        bomb animation cycler
    DIM ShowQB64 AS INTEGER '      toogle used to flip showing QB64 screen or not
    DIM x AS INTEGER '             generic counter
    DIM Flip AS INTEGER '          invader animation cycler

    DO '                                                                                       begin key check loop
        FlippedY = TRUE '                                                                      set initial state of toggles
        AddedC = TRUE
        ShowQB64 = FALSE
        DO '                                                                                   begin insert coin animation loop
            FlippedY = NOT FlippedY '                                                          toggle flipped Y screen
            ClearDisplay INCOIN '                                                              clear the display
            DrawScore BOTHPLAYERS, INCOIN '                                                    draw the score of both players
            DrawCredits '                                                                      draw number of credits in machine
            DrawShipsRemaining '                                                               draw the number of ships remaining
            IF NOT FlippedY THEN '                                                             show the flipped Y screen?
                SlowText 8, 12, "PLAY", INCOIN, KeyPress: IF KeyPress THEN EXIT DO '           no, display text and exit if key pressed
                _PUTIMAGE , Display.WorkScreen, Display.CorrectY, (0, 64)-(223, 71) '          get an image of non-flipped Y
            ELSE '                                                                             yes, show the slipped Y screen
                SlowText 8, 12, "PLA ", INCOIN, KeyPress: IF KeyPress THEN EXIT DO '           display text and exit if key pressed
                _PUTIMAGE , Display.WorkScreen, Display.WithoutY, (0, 64)-(223, 71) '          get an image of the Y missing
                _PUTIMAGE (127, 70)-(120, 63), Font(89) '                                      draw an upside down Y
                _PUTIMAGE , Display.WorkScreen, Display.WithY, (0, 64)-(223, 71) '             get an image of the upside down Y
            END IF
            SlowText 11, 7, "SPACE  INVADERS", INCOIN, KeyPress: IF KeyPress THEN EXIT DO '    display text and exit if key pressed
            SLEEP 1 '                                                                          pause for 1 second
            Text 15, 4, "*SCORE ADVANCE TABLE*" '                                              display text
            _PUTIMAGE (64, 136), IMG.UFO: Text 17, 10, "=" '                                   display UFO=
            _PUTIMAGE (68, 152), IMG_Invader(1, 1): Text 19, 10, "=" '                         display invaders and =
            _PUTIMAGE (67, 168), IMG_Invader(2, 0): Text 21, 10, "="
            _PUTIMAGE (66, 184), IMG_Invader(3, 1): Text 23, 10, "="
            SlowText 17, 11, "? MYSTERY", INCOIN, KeyPress: IF KeyPress THEN EXIT DO '         display point values and exit if key pressed
            SlowText 19, 11, "30 POINTS", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
            SlowText 21, 11, "20 POINTS", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
            SlowText 23, 11, "10 POINTS", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
            IF FlippedY THEN '                                                                 has the Y been flipped?
                FOR x = 240 TO 124 STEP -2 '                                                   yes, cycle from right of screen to left
                    _LIMIT 30 '                                                                loop will run at 30 frames per second
                    _PUTIMAGE (0, 64), Display.WithY '                                         show image with flipped Y
                    _PUTIMAGE (x + 4, 64), IMG_Invader(1, Flip) '                              show invader on screen at x coordinate
                    Flip = 1 - Flip '                                                          flip between invader images (0 to 1)
                    DrawScore BOTHPLAYERS, INCOIN '                                            draw the score of both players
                    DrawCredits '                                                              draw number of credits in machine
                    DrawShipsRemaining '                                                       draw the number of ships remaining
                    UpdateDisplay INCOIN '                                                     show results on display screen
                    KeyPress = GetKey(INCOIN): IF KeyPress THEN EXIT DO '                      leave if a key has been pressed
                NEXT x
                FOR x = 124 TO 240 STEP 2 '                                                    cycle back to the right of screen
                    _LIMIT 30 '                                                                loop will run at 30 frames per second
                    _PUTIMAGE (0, 64), Display.WithoutY '                                      show image without a Y
                    _PUTIMAGE (x + 4, 64), IMG_Invader(1, Flip) '                              place invader at x location
                    _PUTIMAGE (x + 3, 70)-(x - 4, 63), Font(89) '                              place flipped Y behind invader
                    Flip = 1 - Flip '                                                          flip between invader images (0 to 1)
                    DrawScore BOTHPLAYERS, INCOIN '                                            draw the score of both players
                    DrawCredits '                                                              draw number of credits in machine
                    DrawShipsRemaining '                                                       draw the number of ships remaining
                    UpdateDisplay INCOIN '                                                     show results on display screen
                    KeyPress = GetKey(INCOIN): IF KeyPress THEN EXIT DO '                      leave if a key has been pressed
                NEXT x
                FOR x = 240 TO 124 STEP -2 '                                                   cycle from right of screen to left
                    _LIMIT 30 '                                                                loop will run at 30 frames per second
                    _PUTIMAGE (0, 64), Display.WithoutY '                                      show image without a Y
                    _PUTIMAGE (x + 4, 64), IMG_Invader(1, Flip) '                              place invader at x location
                    _PUTIMAGE (x - 4, 64), Font(89) '                                          place regular Y in front of invader
                    Flip = 1 - Flip '                                                          flip between invader images (0 to 1)
                    DrawScore BOTHPLAYERS, INCOIN '                                            draw the score of both players
                    DrawCredits '                                                              draw number of credits in machine
                    DrawShipsRemaining '                                                       draw the number of ships remaining
                    UpdateDisplay INCOIN '                                                     show results on display screen
                    KeyPress = GetKey(INCOIN): IF KeyPress THEN EXIT DO '                      leave if a key has been pressed
                NEXT x
                _PUTIMAGE (0, 64), Display.CorrectY '                                          show image with correct Y
                SLEEP 1 '                                                                      pause for 1 second
                UpdateDisplay INCOIN '                                                         show result on display screen
            END IF
            SLEEP 2 '                                                                          pause for 2 seconds

            '** PLAY DEMO **

            AddedC = NOT AddedC '                                                              toggle added C screen
            ClearDisplay INCOIN '                                                              clear the display
            DrawScore BOTHPLAYERS, INCOIN '                                                    draw the score of both players
            DrawCredits '                                                                      draw number of credits in machine
            DrawShipsRemaining '                                                               draw the number of ships remaining
            Text 14, 8, "INSERT  COIN" '                                                       display text
            _PUTIMAGE , Display.WorkScreen, Display.NormalC, (0, 32)-(223, 119) '              get an image without an extra C
            IF AddedC THEN '                                                                   time to add an extra C?
                Text 14, 15, "C" '                                                             yes, display another C
                _PUTIMAGE , Display.WorkScreen, Display.AddedC, (0, 32)-(223, 119) '           get an image with the extra C
            END IF
            SlowText 18, 6, "<1 OR 2 PLAYERS>", INCOIN, KeyPress: IF KeyPress THEN EXIT DO '   display text and exit if key pressed
            SlowText 21, 6, "*1 PLAYER  1 COIN", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
            SlowText 24, 6, "*2 PLAYERS 2 COINS", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
            IF AddedC THEN '                                                                   has an extra C been added?
                Flip = 1 '                                                                     yes, reset invader image flip
                FOR x = -16 TO 120 STEP 2 '                                                    start from left side of screen to right
                    _LIMIT 30 '                                                                limit loop to 30 frames per second
                    _PUTIMAGE (0, 32), Display.AddedC '                                        show image with the extra C
                    _PUTIMAGE (x, 32), IMG_Invader(1, Flip) '                                  place moving invader on screen
                    Flip = 1 - Flip '                                                          flip between invader images (0 to 1)
                    UpdateDisplay INCOIN '                                                     show results on display screen
                    KeyPress = GetKey(INCOIN): IF KeyPress THEN EXIT DO '                      leave if a key has been pressed
                NEXT x
                _DELAY .125 '                                                                  pause for 1/8th second
                Bcycle = 1 '                                                                   reset bomb image cycler
                FOR x = 40 TO 108 STEP 2 '                                                     cycle from under invader to letter C
                    _LIMIT 30 '                                                                limit loop to 30 frames per second
                    _PUTIMAGE (0, 32), Display.AddedC '                                        show image with the extra C
                    _PUTIMAGE (120, 32), IMG_Invader(1, Flip) '                                place stationary invader on screen
                    _PUTIMAGE (123, x), IMG_Bomb(1, Bcycle) '                                  place moving bomb on screen
                    Bcycle = Bcycle + 1 '                                                      cycle to next bomb image
                    IF Bcycle = 4 THEN Bcycle = 0 '                                            cycle back to 1 if needed
                    UpdateDisplay INCOIN '                                                     show results on display screen
                    KeyPress = GetKey(INCOIN): IF KeyPress THEN EXIT DO '                      leave if a key has been pressed
                NEXT x
                _PUTIMAGE (0, 32), Display.NormalC '                                           show image without the extra C
                _PUTIMAGE (122, 112), IMG.BombHit '                                            place bomb hit explosion over C
                _PUTIMAGE (120, 32), IMG_Invader(1, Flip) '                                    place stationary invader on screen
                UpdateDisplay INCOIN '                                                         show results on display screen
                _DELAY .0625 '                                                                 pause for 1/16th second
                _PUTIMAGE (0, 32), Display.NormalC '                                           show image without the extra C
                _PUTIMAGE (120, 32), IMG_Invader(1, Flip) '                                    place stationary invader on screen
                UpdateDisplay INCOIN '                                                         show results on display screen
            END IF
            SLEEP 2 '                                                                          pause for 2 seconds
            ShowQB64 = NOT ShowQB64 '                                                          toggle QB64 screen
            IF ShowQB64 THEN '                                                                 time to show QB64 screen?
                ClearDisplay INCOIN '                                                          yes, clear the display
                DrawScore BOTHPLAYERS, INCOIN '                                                draw the score of both players
                DrawCredits '                                                                  draw number of credits in machine
                DrawShipsRemaining '                                                           draw the number of ships remaining
                _PUTIMAGE (75, 71), IMG.QB64PE '                                               show QB64PE image
                SlowText 5, 8, "Created With", INCOIN, KeyPress: IF KeyPress THEN EXIT DO '    draw credits and leave if a key pressed
                SlowText 7, 11, "QB64PE", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
                SlowText 19, 4, "www.qb64phoenix.com", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
                SlowText 21, 7, "QB64 remake by", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
                SlowText 23, 7, "TERRY  RITCHIE", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
                SlowText 25, 3, "quickbasic64@gmail.com", INCOIN, KeyPress: IF KeyPress THEN EXIT DO
                SLEEP 4 '                                                                      pause for 4 seconds
                KeyPress = GetKey(INCOIN): IF KeyPress THEN EXIT DO '                          leave if a key has been pressed
            END IF
        LOOP
        IF KeyPress = 79 THEN SetOptions '                                                     go to options screen if O key pressed
    LOOP UNTIL KeyPress = 67 '                                                                 leave when C key pressed

END SUB
'------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
SUB SetupDisplay () '                                                                                                                                              | SetupDisplay |
    '+-------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
    '+ Sets up the display screen according to options selected                                                                                                                   |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Options AS OPTIONS ' need access to shared variables
    SHARED Display AS DISPLAY
    SHARED IMG AS IMAGES
    DIM TmpScreen AS LONG '     temporary screen image if needed
    DIM IMG_Icon AS LONG '      window icon image
    DIM Xclicked AS INTEGER '   window X close trap

    IF _FULLSCREEN THEN _FULLSCREEN _OFF '                                                   leave full screen if currently enabled
    IF Display.Screen THEN '                                                                 has a window already been created?
        TmpScreen = _NEWIMAGE(1, 1, 32) '                                                    yes, create a temporary new window
        SCREEN TmpScreen '                                                                   change to the new window
        _FREEIMAGE Display.Screen '                                                          remove old window image from RAM
    END IF
    IF Options.Bezel THEN '                                                                  is the bezel selected to display?
        Display.Screen = _NEWIMAGE(640 * Options.ScreenSize, 360 * Options.ScreenSize, 32) ' yes, make the display screen the size of the bezel
        _PUTIMAGE , Display.Bezel, Display.Screen '                                          place the bezel image onto the display screen
        Display.Bez.x1 = 208 * Options.ScreenSize '                                          calculate the location of the work screen coordinates within the bezel image
        Display.Bez.y1 = 74 * Options.ScreenSize
        Display.Bez.x2 = Display.Bez.x1 + 224 * Options.ScreenSize
        Display.Bez.y2 = Display.Bez.y1 + 248 * Options.ScreenSize
    ELSE '                                                                                   no, no bezel image to be used
        Display.Screen = _NEWIMAGE(224 * Options.ScreenSize, 248 * Options.ScreenSize, 32) ' create player display screen based on size of game screen
    END IF
    SCREEN Display.Screen '                                                                  create game window
    _DELAY .5
    _SCREENMOVE _MIDDLE '                                                                    move game screen to center of desktop
    IMG_Icon = _LOADIMAGE("siicon.bmp", 32) '                                                load window icon
    _ICON IMG_Icon '                                                                         set window icon
    _FREEIMAGE IMG_Icon '                                                                    icon image no longer needed
    _TITLE "QB64 Space Invaders" '                                                           set window title
    IF TmpScreen THEN _FREEIMAGE TmpScreen '                                                 remove temporary image if created
    IF Options.FullScreen THEN _FULLSCREEN _SQUAREPIXELS , _SMOOTH '                         go full screen if option set
    _DEST Display.WorkScreen '                                                               game updates take place on this screen
    _SOURCE Display.WorkScreen '                                                             game updates take place on this screen
    CLS , BLACK '                                                                            clear the screen with black background
    Xclicked = _EXIT '                                                                       trap the window X close button

END SUB
'------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
SUB ClearDisplay (Mode AS INTEGER) '                                                                                                                               | ClearDisplay |
    '+-------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------+
    '| Clears the display according to mode requested                                                                                                                             |
    '| Mode - 5 (INGAME), 2 (INCOIN), 3 (INSELECT) - clear with background or colored foil                                                                                        |
    '|        1 (INOPTIONS) - clear completely to black                                                                                                                           |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Options AS OPTIONS ' need access to shared variables
    SHARED Display AS DISPLAY

    SELECT CASE Mode '                                                 which type of screen clear needs to be done?
        CASE INGAME, INCOIN, INSELECT '                                game screen is currently showing
            IF Options.Background THEN '                               is the option to show a background image set?
                _PUTIMAGE , Display.Background '                       yes, clear the working screen with the background image
            ELSE '                                                     no, no background image is to be used
                CLS , BLACK '                                          clear the working screen with solid black
                LINE (0, 24)-(223, 55), _RGB32(255, 0, 0, 20), BF '    simulate the foil strips that were placed over
                LINE (0, 184)-(223, 239), _RGB32(0, 255, 0, 15), BF '  the screen in arcades to give the illusion that
                LINE (16, 240)-(135, 247), _RGB32(0, 255, 0, 15), BF ' the game was in color
            END IF
        CASE INOPTIONS '                                               options screen is currently showing
            CLS , BLACK '                                              clear the screen in black
            _PUTIMAGE , Display.OptionScreen, Display.WorkScreen '     place the option screen onto the work screen
    END SELECT

END SUB
'-----------------------------------------------------------------------------------------------------------------------------------------------------------------+---------------+
SUB UpdateDisplay (Mode AS INTEGER) '                                                                                                                             | UpdateDisplay |
    '+------------------------------------------------------------------------------------------------------------------------------------------------------------+---------------+
    '| Updates the display with changes according to mode requested                                                                                                               |
    '| Mode - 5 (INGAME), 2 (INCOIN), 3 (INSELECT) - display high score, line under player, and apply color strips                                                                |
    '|        1 (INOPTIONS) - just copy work screen to work mask, no changes                                                                                                      |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Game AS GAME '       need access to shared variables
    SHARED Display AS DISPLAY
    SHARED Options AS OPTIONS

    Game.Frame = Game.Frame + 1 '                                                                                       increment master game frame counter
    IF Game.Frame = 32761 THEN Game.Frame = 1 '                                                                         reset master game frame counter when needed
    SELECT CASE Mode '                                                                                                  which type of screen update needs to be done?
        CASE INGAME, INCOIN, INSELECT '                                                                                 game screen is currently showing
            Text 0, 1, "SCORE<1> HI-SCORE SCORE<2>" '                                                                   draw high score text
            Text 2, 11, RIGHT$("00000" + _TRIM$(STR$(Game.HighScore)), 5)
            LINE (0, 237)-(223, 237), WHITE '                                                                           draw line under player
            _PUTIMAGE , Display.ColorMask, Display.WorkMask '                                                           place color strips onto a temporary image
            _SETALPHA 0, WHITE, Display.WorkScreen '                                                                    make white the transparent color of the working screen
            _PUTIMAGE , Display.WorkScreen, Display.WorkMask '                                                          place the working screen onto the color strips
        CASE INOPTIONS '                                                                                                options screen is currently showing
            _PUTIMAGE , Display.WorkScreen, Display.WorkMask '                                                          place the working screen onto a temporary image
    END SELECT
    IF Options.Bezel THEN '                                                                                             is the bezel selected to be displayed?
        _PUTIMAGE (Display.Bez.x1, Display.Bez.y1)-(Display.Bez.x2, Display.Bez.y2), Display.WorkMask, Display.Screen ' yes, place screen inside bezel image
    ELSE '                                                                                                              no, no bezel image is to be shown
        _PUTIMAGE , Display.WorkMask, Display.Screen '                                                                  place the temporary image onto the player's view screen
    END IF
    _DISPLAY '                                                                                                          update the player's display screen with changes

END SUB
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
SUB LoadAssets () '                                                                                                                                                  | LoadAssets |
    '+---------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
    '| Load the game's graphics and sound files                                                                                                                                   |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED IMG AS IMAGES '         need access to shared variables
    SHARED SND AS SOUNDS
    SHARED Font() AS LONG
    SHARED IMG_Invader() AS LONG
    SHARED IMG_Bomb() AS LONG
    SHARED IMG_Ship() AS LONG
    SHARED Shield() AS SHIELDS
    SHARED Display AS DISPLAY
    DIM IMG_SpriteSheet AS LONG '  image sprite sheet
    DIM x AS INTEGER '             generic counter
    DIM y AS INTEGER '             generic counter

    SND.Beat1 = _SNDOPEN("sibeat1.ogg") '                                      beat 1 sound
    SND.Beat2 = _SNDOPEN("sibeat2.ogg") '                                      beat 2 sound
    SND.Beat3 = _SNDOPEN("sibeat3.ogg") '                                      beat 3 sound
    SND.Beat4 = _SNDOPEN("sibeat4.ogg") '                                      beat 4 sound
    SND.InvaderHit = _SNDOPEN("siidead.ogg") '                                 invader explosion sound
    SND.PlayerHit = _SNDOPEN("sipdead.ogg") '                                  player explosion sound
    SND.UFOHit = _SNDOPEN("siudead.ogg") '                                     ufo explosion sound
    SND.UFOFlying = _SNDOPEN("siufofly.ogg") '                                 ufo flying sound
    SND.Laser = _SNDOPEN("silaser.ogg") '                                      player shooting sound
    SND.Coin = _SNDOPEN("sicoin.ogg") '                                        coin dropping sound
    SND.Extra = _SNDOPEN("siextra.ogg") '                                      extra ship sound
    IMG.QB64PE = _NEWIMAGE(73, 73, 32) '                                       create image containers
    IMG.DipSwitch = _NEWIMAGE(16, 36, 32)
    IMG.UFO = _NEWIMAGE(16, 8, 32)
    IMG.InvaderHit = _NEWIMAGE(14, 8, 32)
    IMG.BombHit = _NEWIMAGE(6, 8, 32)
    IMG.BombHitMask = _NEWIMAGE(6, 8, 32)
    IMG.LaserHit = _NEWIMAGE(8, 8, 32)
    IMG.LaserHitMask = _NEWIMAGE(8, 8, 32)
    IMG.Shield = _NEWIMAGE(26, 16, 32)
    IMG_Ship(-1) = _NEWIMAGE(16, 8, 32)
    IMG_Ship(0) = _NEWIMAGE(13, 8, 32)
    IMG_Ship(1) = _NEWIMAGE(16, 8, 32)
    IMG_Invader(1, 0) = _NEWIMAGE(8, 8, 32)
    IMG_Invader(1, 1) = _NEWIMAGE(8, 8, 32)
    IMG_Invader(2, 0) = _NEWIMAGE(11, 8, 32)
    IMG_Invader(2, 1) = _NEWIMAGE(11, 8, 32)
    IMG_Invader(3, 0) = _NEWIMAGE(12, 8, 32)
    IMG_Invader(3, 1) = _NEWIMAGE(12, 8, 32)
    Display.Bezel = _NEWIMAGE(1920, 1080, 32)
    Display.Screen = _NEWIMAGE(224, 248, 32)
    Display.WorkScreen = _NEWIMAGE(224, 248, 32)
    Display.WorkMask = _NEWIMAGE(224, 248, 32)
    Display.ColorMask = _NEWIMAGE(224, 248, 32)
    Display.Background = _NEWIMAGE(896, 992, 32)
    Display.OptionScreen = _NEWIMAGE(224, 248, 32)
    Display.WithY = _NEWIMAGE(224, 8, 32)
    Display.WithoutY = _NEWIMAGE(224, 8, 32)
    Display.CorrectY = _NEWIMAGE(224, 8, 32)
    Display.AddedC = _NEWIMAGE(224, 88, 32)
    Display.NormalC = _NEWIMAGE(224, 88, 32)
    FOR x = PLAYER1 TO PLAYER2
        FOR y = 1 TO 6
            Shield(x, y).Image = _NEWIMAGE(26, 16, 32) '                       6 shield image containers
        NEXT y
    NEXT x
    IMG_Bomb(1, 0) = _NEWIMAGE(3, 8, 32)
    IMG_Bomb(1, 1) = _NEWIMAGE(3, 8, 32)
    IMG_Bomb(1, 2) = _NEWIMAGE(3, 8, 32)
    IMG_Bomb(1, 3) = _NEWIMAGE(3, 8, 32)
    IMG_Bomb(2, 0) = _NEWIMAGE(3, 8, 32)
    IMG_Bomb(2, 1) = _NEWIMAGE(3, 8, 32)
    IMG_Bomb(2, 2) = _NEWIMAGE(3, 8, 32)
    IMG_Bomb(2, 3) = _NEWIMAGE(3, 8, 32)
    IMG_Bomb(3, 0) = _NEWIMAGE(3, 8, 32)
    IMG_SpriteSheet = _LOADIMAGE("sisprites.png", 32) '                        load the sprite sheet
    _SETALPHA 0, BLACK, IMG_SpriteSheet '                                      make black transparent
    _PUTIMAGE , IMG_SpriteSheet, IMG.QB64PE, (848, 200)-(920, 272) '           get qb64pe logo
    _PUTIMAGE , IMG_SpriteSheet, Display.ColorMask, (400, 200)-(623, 447) '    get color mask
    _PUTIMAGE , IMG_SpriteSheet, Display.OptionScreen, (624, 200)-(847, 447) ' get dip switch screen
    _DEST IMG_SpriteSheet '                                                    draw on sprite sheet
    LINE (400, 200)-(920, 447), BLACK, BF '                                    remove images from inside of bezel
    _PUTIMAGE , IMG_SpriteSheet, Display.Bezel, (0, 8)-(1919, 1087) '          get bezel image
    _PUTIMAGE , IMG_SpriteSheet, Display.Background, (1920, 8)-(2815, 999) '   get background image
    _PUTIMAGE , Display.OptionScreen, IMG.DipSwitch, (16, 32)-(31, 67) '       get single dip switch
    _PUTIMAGE , IMG_SpriteSheet, IMG_Invader(1, 0), (0, 0)-(7, 7) '            get invader 1 image cell 1
    _PUTIMAGE , IMG_SpriteSheet, IMG_Invader(1, 1), (8, 0)-(15, 7) '           get invader 1 image cell 2
    _PUTIMAGE , IMG_SpriteSheet, IMG_Invader(2, 0), (16, 0)-(26, 7) '          get invader 2 image cell 1
    _PUTIMAGE , IMG_SpriteSheet, IMG_Invader(2, 1), (27, 0)-(37, 7) '          get invader 2 image cell 2
    _PUTIMAGE , IMG_SpriteSheet, IMG_Invader(3, 0), (38, 0)-(49, 7) '          get invader 3 image cell 1
    _PUTIMAGE , IMG_SpriteSheet, IMG_Invader(3, 1), (50, 0)-(61, 7) '          get invader 3 image cell 2
    _PUTIMAGE , IMG_SpriteSheet, IMG.UFO, (62, 0)-(77, 7) '                    get UFO image
    _PUTIMAGE , IMG_SpriteSheet, IMG_Ship(-1), (91, 0)-(106, 7) '              get exploding player ship image cell 1
    _PUTIMAGE , IMG_SpriteSheet, IMG_Ship(0), (78, 0)-(90, 7) '                get player ship image
    _PUTIMAGE , IMG_SpriteSheet, IMG_Ship(1), (107, 0)-(122, 7) '              get exploding player ship image cell 2
    _PUTIMAGE , IMG_SpriteSheet, IMG.InvaderHit, (123, 0)-(136, 7) '           get exploding invader image
    _PUTIMAGE , IMG_SpriteSheet, IMG.BombHit, (137, 0)-(142, 7) '              get laser/bomb exploding hit image
    _SOURCE IMG.BombHit '                                                      image used on display screen
    _DEST IMG.BombHitMask '                                                    image used to destroy shields
    CLS , BLACK '                                                              clear image in black
    FOR x = 0 TO 5 '                                                           cycle through columns
        FOR y = 0 TO 7 '                                                       cycle through rows
            IF POINT(x, y) <> WHITE THEN '                                     is pixel at x,y white?
                PSET (x, y), WHITE '                                           no, draw white pixel on mask
            END IF '                                                           (this creates a "negative" or mask
        NEXT y '                                                                of the IMG.bombhit image)
    NEXT x
    _SETALPHA 0, WHITE, IMG.BombHitMask '                                      make white transparent
    _PUTIMAGE , IMG_SpriteSheet, IMG.LaserHit, (143, 0)-(150, 7) '             get laser exploding miss image
    _SOURCE IMG.LaserHit '                                                     image used on display screen
    _DEST IMG.LaserHitMask '                                                   image used to destroy shields
    CLS , BLACK '                                                              clear image in black
    FOR x = 0 TO 7 '                                                           cycle through columns
        FOR y = 0 TO 7 '                                                       cycle through rows
            IF POINT(x, y) <> WHITE THEN '                                     is pixel at x,y white?
                PSET (x, y), WHITE '                                           no, draw white pixel on mask
            END IF '                                                           (this creates a "negative" or mask
        NEXT y '                                                                of the IMG.laserhit image)
    NEXT x
    _SETALPHA 0, WHITE, IMG.LaserHitMask '                                     set white as transparent
    _PUTIMAGE (0, 0), IMG_SpriteSheet, IMG.Shield, (154, 0)-(166, 7) '         get upper left shield image
    _PUTIMAGE (13, 0), IMG_SpriteSheet, IMG.Shield, (167, 0)-(179, 7) '        get upper right shield image
    _PUTIMAGE (0, 8), IMG_SpriteSheet, IMG.Shield, (186, 0)-(198, 7) '         get lower left shield image
    _PUTIMAGE (13, 8), IMG_SpriteSheet, IMG.Shield, (199, 0)-(211, 7) '        get lower right shield image
    _SETALPHA 0, BLACK, IMG.Shield '                                           add transparency to shield image
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(1, 0), (215, 0)-(217, 7) '           get bomb 1 image cell 1
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(1, 1), (218, 0)-(220, 7) '           get bomb 1 image cell 2
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(1, 2), (221, 0)-(223, 7) '           get bomb 1 image cell 3
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(1, 3), (224, 0)-(226, 7) '           get bomb 1 image cell 4
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(2, 0), (227, 0)-(229, 7) '           get bomb 2 image cell 1
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(2, 1), (230, 0)-(232, 7) '           get bomb 2 image cell 2
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(2, 2), (233, 0)-(235, 7) '           get bomb 2 image cell 3
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(2, 3), (236, 0)-(238, 7) '           get bomb 2 image cell 4
    _PUTIMAGE , IMG_SpriteSheet, IMG_Bomb(3, 0), (239, 0)-(241, 7) '           get bomb 3 image cell 1
    IMG_Bomb(3, 1) = _COPYIMAGE(IMG_Bomb(1, 0)) '                              copy bomb 3 image cell 2
    IMG_Bomb(3, 2) = _COPYIMAGE(IMG_Bomb(3, 0)) '                              copy bomb 3 image cell 3
    IMG_Bomb(3, 3) = _COPYIMAGE(IMG_Bomb(1, 2)) '                              copy bomb 3 image cell 4
    FOR x = 1 TO 255 '                                                         cycle through 255 font images
        Font(x) = _NEWIMAGE(8, 8, 32) '                                        create font image container
        _PUTIMAGE (0, 0), IMG_SpriteSheet, Font(x), ((x - 1) * 8 + 242, 0)-(((x - 1) * 8 + 242) + 7, 7)
    NEXT x '                                                                   get font character fron sprite sheet
    _FREEIMAGE IMG_SpriteSheet '                                               remove spritesheet from RAM

END SUB
'-------------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------+
SUB LoadOptions () '                                                                                                                                                | LoadOptions |
    '+--------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------+
    '| Load the game options from the options file. If the file does not exist create one with default settings.                                                                  |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Options AS OPTIONS ' need access to shared variables
    SHARED Game AS GAME

    IF _FILEEXISTS("si.sav") THEN '                                                            does the options file exist?
        OPEN "si.sav" FOR INPUT AS #1 '                                                        yes, open the file for reading
        INPUT #1, Options.Switches '                                                           get the saved DIP switch settings
        INPUT #1, Game.HighScore '                                                             get the save high score
        CLOSE #1 '                                                                             close the options file
    ELSE '                                                                                     no, the options file does not exist
        Options.Switches = 215 '                                                               set DIP switches to default settings
        SaveOptions '                                                                          save the options
    END IF
    IF Options.Switches AND 128 THEN Options.FreePlay = FALSE ELSE Options.FreePlay = TRUE '   set free play according to DIP switch setting
    IF Options.Switches AND 64 THEN Options.ExtraShip = 1500 ELSE Options.ExtraShip = 1000 '   set extra ship value according to DIP switch setting
    IF (Options.Switches AND 48) = 48 THEN '                                                   are DIP switches 3 and 4 on?
        Options.Shields = 6 '                                                                  yes, 6 shields will be used
    ELSEIF Options.Switches AND 32 THEN '                                                      no, is DIP switch 3 on?
        Options.Shields = 5 '                                                                  yes, 5 shields will be used
    ELSEIF Options.Switches AND 16 THEN '                                                      no, is DIP switch 4 on?
        Options.Shields = 4 '                                                                  yes, 4 shields will be used
    ELSE '                                                                                     no, neither DIP switch 3 or 4 is on
        Options.Shields = 3 '                                                                  3 shields will be used
    END IF
    Options.FullScreen = FALSE '                                                               assume full screen mode is disabled
    IF (Options.Switches AND 12) = 12 THEN '                                                   are DIP switches 5 and 6 on?
        Options.FullScreen = TRUE '                                                            yes, full screen mode activated
        Options.ScreenSize = 3 '                                                               game screen will be 3X size full screen
    ELSEIF Options.Switches AND 8 THEN '                                                       no, is DIP switch 5 on?
        Options.ScreenSize = 3 '                                                               yes, game screen will be 3X size windowed
    ELSEIF Options.Switches AND 4 THEN '                                                       no, is DIP switch 6 on?
        Options.ScreenSize = 2 '                                                               yes, game screen will be 2X size windowed
    ELSE '                                                                                     no, neither DIP switch 5 or 6 is on
        Options.ScreenSize = 1 '                                                               game screen will be 1X size windowed
    END IF
    IF Options.Switches AND 2 THEN Options.Background = TRUE ELSE Options.Background = FALSE ' set background image according to DIP switch setting
    IF Options.Switches AND 1 THEN Options.Bezel = TRUE ELSE Options.Bezel = FALSE '           set bezel image according to DIP switch setting
    IF (Options.Bezel = FALSE) AND Options.FullScreen THEN Options.ScreenSize = 4 '            use a 4x screen size for full screen without the bezel
    SetupDisplay '                                                                             set the dsiplay according to chosen options

END SUB
'-------------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------+
SUB SaveOptions () '                                                                                                                                                | SaveOptions |
    '+--------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------+
    '| Save the game's options and high score to the options file                                                                                                                 |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Options AS OPTIONS ' need access to shared variables
    SHARED Game AS GAME

    OPEN "si.sav" FOR OUTPUT AS #1 ' create a file to write to
    PRINT #1, Options.Switches '     write the current options to the file
    PRINT #1, Game.HighScore '       write the high score to the file
    CLOSE #1 '                       close the file

END SUB
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
SUB SetOptions () '                                                                                                                                                  | SetOptions |
    '+---------------------------------------------------------------------------------------------------------------------------------------------------------------+------------+
    '| Displays the options screen allowing the player to select options                                                                                                          |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Options AS OPTIONS ' need access to shared variables
    SHARED IMG AS IMAGES
    SHARED Game AS GAME
    SHARED SND AS SOUNDS
    DIM KeyPress AS INTEGER '   a key that was pressed

    DO '                                                                              begin display loop
        _LIMIT 30 '                                                                   limit to 30 frames per second
        ClearDisplay INOPTIONS '                                                      clear the screen
        IF Options.Switches AND 128 THEN '                                            is switch 1 on?
            Text 14, 1, "1: COIN REQUIRED TO PLAY" '                                  yes, display text
            _PUTIMAGE (31, 67)-(16, 32), IMG.DipSwitch '                              flip switch 1 to on position
        ELSE '                                                                        no, switch 1 is off
            Text 14, 1, "1: GAME SET TO FREE PLAY" '                                  display text
        END IF
        IF Options.Switches AND 64 THEN '                                             is switch 2 on?
            Text 16, 1, "2: EXTRA SHIP AT 1500" '                                     yes, display text
            _PUTIMAGE (57, 67)-(42, 32), IMG.DipSwitch '                              flip switch 2 to on position
        ELSE '                                                                        no, switch 2 is off
            Text 16, 1, "2: EXTRA SHIP AT 1000" '                                     display text
        END IF
        IF (Options.Switches AND 48) = 48 THEN '                                      are switches 3 and 4 on?
            Text 18, 1, "3: SIX BASES DURING PLAY" '                                  yes, display text
            Text 19, 1, "4: HELP MOMMY - 6 BASES"
        ELSEIF Options.Switches AND 32 THEN '                                         no, is switch 3 on?
            Text 18, 1, "3: FIVE BASES DURING PLAY" '                                 yes, display text
            Text 19, 1, "4: EASY - 5 BASES"
        ELSEIF Options.Switches AND 16 THEN '                                         no, is switch 4 on?
            Text 18, 1, "3: FOUR BASES DURING PLAY" '                                 yes, display text
            Text 19, 1, "4: DEFAULT - 4 BASES"
        ELSE '                                                                        no, both switches 3 and 4 are off
            Text 18, 1, "3: THREE BASES DURING PLAY" '                                display text
            Text 19, 1, "4: HARD - 3 BASES"
        END IF
        IF Options.Switches AND 32 THEN _PUTIMAGE (83, 67)-(68, 32), IMG.DipSwitch '  if switch 3 is on flip it to the on position
        IF Options.Switches AND 16 THEN _PUTIMAGE (109, 67)-(94, 32), IMG.DipSwitch ' if switch 4 is on flip it to the on position
        IF (Options.Switches AND 12) = 12 THEN '                                      are switches 5 and 6 on?
            Text 21, 1, "5: FULL SCREEN" '                                            yes, display text
            Text 22, 1, "6: (3X ORIGINAL SIZE)"
        ELSEIF Options.Switches AND 8 THEN '                                          no, is switch 5 on?
            Text 21, 1, "5: LARGE 672x744 WINDOW" '                                   yes, display text
            Text 22, 1, "6: (3X ORIGINAL SIZE)"
        ELSEIF Options.Switches AND 4 THEN '                                          no, is switch 6 on?
            Text 21, 1, "5: MEDIUM 448x496 WINDOW" '                                  yes, display text
            Text 22, 1, "6: (2X ORIGINAL SIZE)"
        ELSE '                                                                        no, both switches 5 and 6 are off
            Text 21, 1, "5: SMALL 224x248 WINDOW" '                                   display text
            Text 22, 1, "6: (ORIGINAL SIZE)"
        END IF
        IF Options.Switches AND 8 THEN _PUTIMAGE (135, 67)-(120, 32), IMG.DipSwitch ' if switch 5 is on flip it to the on position
        IF Options.Switches AND 4 THEN _PUTIMAGE (161, 67)-(146, 32), IMG.DipSwitch ' if switch 6 is on flip it to the on position
        IF Options.Switches AND 2 THEN '                                              is switch 7 on?
            Text 24, 1, "7: SHOW BACKGROUND IMAGE" '                                  yes, display text
            _PUTIMAGE (187, 67)-(172, 32), IMG.DipSwitch '                            flip switch 7 to on position
        ELSE '                                                                        no, switch 7 is off
            Text 24, 1, "7: NO BACKGROUND IMAGE" '                                    display text
        END IF
        IF Options.Switches AND 1 THEN '                                              is switch 8 on?
            Text 26, 1, "8: SHOW BEZEL IMAGE" '                                       yes, display text
            _PUTIMAGE (213, 67)-(198, 32), IMG.DipSwitch '                            flip switch 8 to the on position
        ELSE '                                                                        no, switch 8 is off
            Text 26, 1, "8: NO BEZEL IMAGE" '                                         display text
        END IF
        Text 28, 2, "  <R> RESET HIGH SCORE" '                                        display input options
        Text 29, 2, "  <S> SAVE SETTINGS"
        Text 30, 2, "<1-8> TOGGLE DIP SWITCH"
        UpdateDisplay INOPTIONS '                                                     update display with changes
        DO '                                                                          begin keyboard input loop
            _LIMIT 30 '                                                               limit to 30 frames per second
            KeyPress = GetKey(INOPTIONS) '                                            get a valid keyboard input
        LOOP UNTIL KeyPress '                                                         leave when valid keyboard input received
        IF KeyPress = 82 THEN '                                                       was the R key pressed?
            Game.HighScore = 0 '                                                      yes, reset the high score
            _SNDPLAY SND.Extra '                                                      play sound to acknowledge
        END IF
        IF KeyPress <> 83 THEN '                                                      was the S key pressed?
            KeyPress = ABS(KeyPress - 56) '                                           no, convert keypress vale to 7 through 0
            Options.Switches = Options.Switches XOR 2 ^ KeyPress '                    flip appropriate switch setting
        END IF
    LOOP UNTIL KeyPress = 83 '                                                        leave when S keyboard input received
    SaveOptions '                                                                     save game options
    LoadOptions '                                                                     load game options to make any changes take effect

END SUB
'--------------------------------------------------------------------------------------------------------------------------------------------------------------------------+------+
SUB Text (Row AS INTEGER, Column AS INTEGER, Txt AS STRING) '                                                                                                              | Text |
    '+---------------------------------------------------------------------------------------------------------------------------------------------------------------------+------+
    '| Displays text instantly at the requested location                                                                                                                          |
    '| Row    - row where text is to printed (if row > 31 then actual screen coordinates are used. This is for the UFO text)                                                      |
    '| Column - column where text is to be printed                                                                                                                                |
    '| Txt    - text string to be printed                                                                                                                                         |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Font() AS LONG ' need access to shared variables
    DIM c AS INTEGER '      column counter
    DIM r AS INTEGER '      row counter
    DIM p AS INTEGER '      text character position counter

    IF Row < 31 THEN '                                 text on an 8x8 grid?
        r = Row * 8 '                                  yes, calculate text row
        c = Column * 8 '                               calculate text column
    ELSE '                                             no, use actual coordinates on screen for UFO text
        r = Column '                                   convert row to X screen coordinate
        c = Row - 32 '                                 convert column to Y screen coordinate
    END IF
    DO '                                               begin text print loop
        p = p + 1 '                                    increment character position counter
        _PUTIMAGE (c, r), Font(ASC(MID$(Txt, p, 1))) ' draw font character onto screen
        c = c + 8 '                                    move to next text column
    LOOP UNTIL p = LEN(Txt) '                          leave when all characters printed

END SUB
'------------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------+
FUNCTION GetKey (Mode AS INTEGER) '                                                                                                                                      | GetKey |
    '+-------------------------------------------------------------------------------------------------------------------------------------------------------------------+--------+
    '| Get key presses from the player                                                                                                                                            |
    '| Mode - 1 (INOPTIONS) S, 1-8, and R keys only are returned                                                                                                                  |
    '|        2 (INCOIN)    O key (letter O) is the only one returned                                                                                                             |
    '|        3 (INSELECT)  1, 2, and O (letter O) keys are only returned                                                                                                         |
    '|       -1 (NEWGMAE)   clears the buffer and exits                                                                                                                           |
    '| The C key is always monitored and returned. Additionaly when C is pressed a credit is added to the game.                                                                   |
    '| The ESC key is always monitored and the game exited if pressed.                                                                                                            |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Game AS GAME '      need access to shared variables
    SHARED SND AS SOUNDS
    STATIC Buffer AS INTEGER ' buffer to hold last key pressed (retains value between calls)
    DIM KeyPress AS INTEGER '  last key that was pressed

    IF _EXIT THEN ExitGame '                          leave game if player closes window with X button
    IF Mode = NEWGAME THEN Buffer = 0: EXIT FUNCTION 'clear the buffer and exit
    KeyPress = _KEYHIT '                              get a key if pressed
    IF KeyPress = 0 THEN EXIT FUNCTION '              leave function if no key pressed
    IF KeyPress < 0 THEN Buffer = 0: EXIT FUNCTION '  if a key was released then clear the buffer and leave function
    IF Buffer THEN GetKey = 0: EXIT FUNCTION '        if a key is being held down return nothing and leave function
    IF KeyPress = 27 THEN ExitGame '                  exit the game if the ESC key pressed
    Buffer = KeyPress '                               put the key into the buffer
    IF KeyPress = 67 OR KeyPress = 99 THEN '          was the C key pressed?
        Game.Credits = Game.Credits + 1 '             yes, insert a coin
        IF Game.Credits > 99 THEN Game.Credits = 99 ' limit amount of coins in game
        _SNDPLAY SND.Coin '                           play the coin dropping sound
        GetKey = 67 '                                 return the key to the insert coin screen
        EXIT FUNCTION '                               leave the function
    END IF
    SELECT CASE Mode '                                which mode is the game in?
        CASE INOPTIONS '                              the options screen is showing
            IF KeyPress = 83 OR KeyPress = 115 THEN ' was the S key pressed?
                GetKey = 83 '                         yes, return the key to the options screen
            END IF
            IF KeyPress > 48 AND KeyPress < 57 THEN ' was the 1 through 8 key pressed?
                GetKey = KeyPress '                   yes, return the key to the options screen
            END IF
            IF KeyPress = 82 OR KeyPress = 114 THEN ' was the R key pressed?
                GetKey = 82 '                         yes, return the key to the options screen
            END IF
        CASE INCOIN '                                 the insert coin screen is showing
            IF KeyPress = 79 OR KeyPress = 111 THEN ' was the O key pressed?
                GetKey = 79 '                         yes, return the key to the insert coin screen
            END IF
        CASE INSELECT '                               the select players screen is showing
            IF KeyPress = 49 OR KeyPress = 50 THEN '  was the 1 or 2 key pressed?
                GetKey = KeyPress '                   yes, return the key to the select players screen
            END IF
            IF KeyPress = 79 OR KeyPress = 111 THEN ' was the O key pressed?
                GetKey = 79 '                         yes, return the key to the insert coin screen
            END IF
    END SELECT

END FUNCTION
'---------------------------------------------------------------------------------------------------------------------------------------------------------------------+-----------+
SUB DrawScore (p AS INTEGER, Mode AS INTEGER) '                                                                                                                       | DrawScore |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------+-----------+
    '| Draws the given player's score in the mode requested                                                                                                                       |
    '| p    - player 1 or 2 - 3 (BOTHPLAYERS) to have both players high scores drawn at same time                                                                                 |
    '| Mode - 5 (INGAME) - update the high score if it has been surpassed                                                                                                         |
    '|        any other value is ignored                                                                                                                                          |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Player() AS PLAYER ' need access to shared variables
    SHARED Game AS GAME
    SHARED Options AS OPTIONS
    SHARED SND AS SOUNDS

    IF p = PLAYER1 OR p = BOTHPLAYERS THEN '                                                            player 1 selected?
        Text 2, 2, RIGHT$("00000" + _TRIM$(STR$(Player(1).Score)), 5) '                                 yes, print palyer 1's score
    END IF
    IF p = PLAYER2 OR p = BOTHPLAYERS THEN '                                                            player 2 selected?
        Text 2, 20, RIGHT$("00000" + _TRIM$(STR$(Player(2).Score)), 5) '                                yes, print player 2's score
    END IF
    IF Player(Game.Player).Ship.Extra = FALSE THEN '                                                    has the current player been awarded an extra ship?
        IF Player(Game.Player).Score >= Options.ExtraShip THEN '                                        no, is the current player's score high enough for an extra ship?
            Player(Game.Player).Ship.Remain = Player(Game.Player).Ship.Remain + 1 '                     yes, award the player another ship
            Player(Game.Player).Ship.Extra = TRUE '                                                     remember that an extra ship was awarded
            _SNDPLAY SND.Extra
        END IF
    END IF
    IF Mode = INGAME THEN '                                                                             is a game currently in progress?
        IF Player(Game.Player).Score > Game.HighScore THEN Game.HighScore = Player(Game.Player).Score ' yes, update the high score is a player exceeds it
    END IF

END SUB
'------------------------------------------------------------------------------------------------------------------------------------------------------------+--------------------+
SUB DrawShipsRemaining () '                                                                                                                                  | DrawShipsRemaining |
    '+-------------------------------------------------------------------------------------------------------------------------------------------------------+--------------------+
    '| Draws the player's remaining ships                                                                                                                                         |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Player() AS PLAYER ' need access to shared variables
    SHARED IMG_Ship() AS LONG
    SHARED Game AS GAME
    DIM s AS INTEGER '          ship counter
    DIM p AS INTEGER '          curent player

    p = Game.Player '                                 get current player
    Text 30, 1, _TRIM$(STR$(Player(p).Ship.Remain)) ' print number of total ships
    s = 1 '                                           reset ship counter
    WHILE s <= Player(p).Ship.Remain - 1 '            display a ship?
        _PUTIMAGE (8 + (s * 16), 239), IMG_Ship(0) '  yes, draw ship
        s = s + 1 '                                   increment ship counter
    WEND '                                            leave when no more ships to draw

END SUB
'-------------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------+
SUB DrawCredits () '                                                                                                                                                | DrawCredits |
    '+--------------------------------------------------------------------------------------------------------------------------------------------------------------+-------------+
    '| Draw the number of credits in the game                                                                                                                                     |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED Game AS GAME ' ned access to shared variables

    Text 28, 1, "<C>OIN <O>PTIONS <ESC>EXIT" '                              print the options text
    Text 30, 17, "CREDIT-" + RIGHT$("00" + _TRIM$(STR$(Game.Credits)), 2) ' print the number of credits inserted

END SUB
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
SUB ExitGame () '                                                                                                                                                      | ExitGame |
    '+-----------------------------------------------------------------------------------------------------------------------------------------------------------------+----------+
    '| Frees all game assets from RAM and exits the game                                                                                                                          |
    '+----------------------------------------------------------------------------------------------------------------------------------------------------------------------------+
    SHARED SND AS SOUNDS '                   need access to shared variables
    SHARED IMG AS IMAGES
    SHARED IMG_Ship() AS LONG
    SHARED IMG_Invader() AS LONG
    SHARED IMG_Bomb() AS LONG
    SHARED Display AS DISPLAY
    SHARED Shield() AS SHIELDS
    SHARED Font() AS LONG
    DIM x AS INTEGER '                       generic counter
    DIM y AS INTEGER '                       generic counter

    IF _FULLSCREEN THEN _FULLSCREEN _OFF '   leave full screen if enabled
    SCREEN 0, 0, 0, 0 '                      switch to a pure text screen
    _SNDCLOSE SND.Beat1 '                    remove sounds from RAM
    _SNDCLOSE SND.Beat2
    _SNDCLOSE SND.Beat3
    _SNDCLOSE SND.Beat4
    _SNDCLOSE SND.InvaderHit
    _SNDCLOSE SND.PlayerHit
    _SNDCLOSE SND.UFOHit
    _SNDCLOSE SND.UFOFlying
    _SNDCLOSE SND.Laser
    _SNDCLOSE SND.Coin
    _SNDCLOSE SND.Extra
    _FREEIMAGE IMG.QB64PE '                  remove images from RAM
    _FREEIMAGE IMG.DipSwitch
    _FREEIMAGE IMG.UFO
    _FREEIMAGE IMG.InvaderHit
    _FREEIMAGE IMG.BombHit
    _FREEIMAGE IMG.BombHitMask
    _FREEIMAGE IMG.LaserHit
    _FREEIMAGE IMG.LaserHitMask
    _FREEIMAGE IMG.Shield
    _FREEIMAGE IMG_Ship(-1)
    _FREEIMAGE IMG_Ship(0)
    _FREEIMAGE IMG_Ship(1)
    _FREEIMAGE IMG_Invader(1, 0)
    _FREEIMAGE IMG_Invader(1, 1)
    _FREEIMAGE IMG_Invader(2, 0)
    _FREEIMAGE IMG_Invader(2, 1)
    _FREEIMAGE IMG_Invader(3, 0)
    _FREEIMAGE IMG_Invader(3, 1)
    _FREEIMAGE Display.Bezel
    _FREEIMAGE Display.Screen
    _FREEIMAGE Display.WorkScreen
    _FREEIMAGE Display.WorkMask
    _FREEIMAGE Display.ColorMask
    _FREEIMAGE Display.Background
    _FREEIMAGE Display.OptionScreen
    _FREEIMAGE Display.WithY
    _FREEIMAGE Display.WithoutY
    _FREEIMAGE Display.CorrectY
    _FREEIMAGE Display.AddedC
    _FREEIMAGE Display.NormalC
    _FREEIMAGE IMG_Bomb(1, 0)
    _FREEIMAGE IMG_Bomb(1, 1)
    _FREEIMAGE IMG_Bomb(1, 2)
    _FREEIMAGE IMG_Bomb(1, 3)
    _FREEIMAGE IMG_Bomb(2, 0)
    _FREEIMAGE IMG_Bomb(2, 1)
    _FREEIMAGE IMG_Bomb(2, 2)
    _FREEIMAGE IMG_Bomb(2, 3)
    _FREEIMAGE IMG_Bomb(3, 0)
    _FREEIMAGE IMG_Bomb(3, 1)
    _FREEIMAGE IMG_Bomb(3, 2)
    _FREEIMAGE IMG_Bomb(3, 3)
    FOR x = 1 TO 2
        FOR y = 1 TO 6
            _FREEIMAGE Shield(x, y).Image
        NEXT y
    NEXT x
    FOR x = 1 TO 255
        _FREEIMAGE Font(x)
    NEXT x
    SYSTEM '                                return to the operating system

END SUB
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------



Attached Files Thumbnail(s)
   

.zip   SpaceInvaders.zip (Size: 2.43 MB / Downloads: 84)
Print this item

  CHAIN
Posted by: Udix - 11-01-2022, 11:59 AM - Forum: Help Me! - Replies (7)

Hi,
I am having a problem using QB64-CHAIN on an Windows XP-x86-system. 
From QB 4.5 , Original, Sources. I am copying and using COM1_EX.bas and COM2_EX.bas.
The CHAIN command in COM1_EX is supposed to copy via COMMON a 1-dim-array()  into COM2_EX - where however it's array-contents never arrive.
I did create COM2_EX.exe to be chained.
I also tried it with COMMON SHARED , and by using the identical variable names in both programs - no effect.
While using QB 4.5 the two programs chainig work ok.
Any ideas what may create that behavior? 
Udix

Print this item

  Is _MOUSEMOVEMENTY and _MOUSEMOVEMENTX supposed to act this way?
Posted by: Pete - 11-01-2022, 02:08 AM - Forum: Help Me! - Replies (10)

So what I thought should be doable with thee two commands cannot be accomplished.

Here is what the code demonstrates.

1) Sets the program screen to the top left corner of the desktop.
2) Make a screen click, which places the mouse at 0, 0 of your desktop.
3) Now carefully move the mouse around in the program screen, and after a bit, carefully move it back to the upper right corner.

Well it should be back to 0, 0, right? Well, far from it. I mean provided the mouse isn't moved past the borders of the desktop, I would think the relative coordinates should be the same at the same points the mouse originated from: 0, 0 when back at the top left side of the desktop.

Code: (Select All)
_SCREENMOVE 0, 0
_DELAY .1
_SCREENCLICK 0, 0
DO
    _LIMIT 30
    WHILE _MOUSEINPUT
        x = x + _MOUSEMOVEMENTX
        y = y + _MOUSEMOVEMENTY
    WEND
    LOCATE 1, 1: PRINT y; x; "    ";
    IF LEN(INKEY$) THEN
        EXIT DO
    END IF
LOOP
END

Pete

Print this item

  Sound playback from embedded data
Posted by: King Mocker - 10-31-2022, 11:24 PM - Forum: Help Me! - Replies (5)

Hi,
Long time listener, first time caller.

I'm wondering if someone can advise on solution for a problem I'm having.
I am writing a program and would like to make it as a standalone/self-contained .exe, without any extra files being required.

I have been looking at Dav's embedded data routine to embed WAV sound files and play them back.
I have had playback success using the WINMM API method PlaySoundA& but this only works with Windows.
The _SNDPLAY command needs a filehandle to work meaning the file must be loaded creating a filehandle.

I've seen RhoSigmas embed method but this extracts the data to files which are then read back it. I'd like to avoid this if possible.

When I saw Steve's post about MEMFILEs I thought I'd hit the jackpot but alas I didn't, unless I misunderstood it, always a possibility.

So, is what I am asking even possible?
Can embedded data be used as a source for _SNPLAY or _SNDOPEN without actual files being used?
Is there another method for me to consider?

Thanks
KM

Print this item

  QB64 Delenda Est!
Posted by: bearheathen - 10-31-2022, 04:32 PM - Forum: General Discussion - Replies (44)

NOTE: This is literally a copy/paste. I just posted the same thing in the "Official" Discord.

You know, a year ago I found out my wife had cancer.

The last person I spoke to here and in the (now old) forums at the time - as a new/old programmer returning after years - was Fellipe Heitor. I had a half-baked UI running in InForm and he was helping me so much (everyone had).

My wife is better now, and I come back to find that horrible things have happened. I will not be speaking with Fellipe today, it would seem. What a shame. I have no dog in this current fight, so let me share what this looks like from the outside looking in:

I see division.

I see good developers not only not talking to each other - but talking poorly about one another. Yet you are all titans to we newcomers.

Legends of the community such as you all should not be divided. I was so *excited* to come back to programming after a year-long, intense battle only to find something I cared about has been decimated with disagreement.

Keybone here and Spriggsy there. Cory on one side, Steve on another? ***WHAT?*** Do you have *any idea* how much I have learned from you folks?!?!?! And now you're hardly on **speaking** terms? Nonsense!

It has always wanted to make a game in QB64, and was happily following Terry Ritchie's tutorial when we got the worst news. Now I can't. Now I have to **choose** which QB64 to use?!?! Not in the "FOSS" sense, but in the "*pick a side*" sense? The two QB64s have already diverged in different directions! Soon libraries won't even be compatible, and then we **really **lose.

The reason anyone is upset to begin with is you're all *passionate*. The dark side of passion can sometimes create divisions (as is clearly the case).

The worst part is that **none** of you did this. **All** of you have done nothing but fight to **SAVE **QB64. And yes, I've been on the PE forums too scratching my head trying to figure out what the hell is going on.  I will make a similar appeal there as soon as I post this here.

Don't make me choose. Life is short. Unite QB64 again!

Print this item

  Particle Life
Posted by: a740g - 10-31-2022, 02:47 PM - Forum: a740g - Replies (14)

Enjoy this little digital fidget toy. Big Grin

Hit the Random button for fun!

A special thank you to @TerryRitchie for ideas that I got from his Graphic Line Input Library & Button Library.

The project is hosted on GitHub.

[Image: Screenshot-2024-10-15-014127.png]

[Image: screenshot1.png]

[Image: screenshot2.png]

[Image: screenshot3.png]

[Image: screenshot4.png]



Attached Files
.zip   Particle-Life.zip (Size: 456.5 KB / Downloads: 16)
Print this item

  Officially goin out of my mind. Trivial INT problem
Posted by: bert22306 - 10-31-2022, 03:34 AM - Forum: General Discussion - Replies (11)

Doggone it, I couldn't figure out why this program wasn't working, and it boils down to this:

Code: (Select All)
x = 35.51
y = x * 100
z = Int(y)
Print x, y, z

WTH, over? My results are: 35.51    3551    3550

How does that make any sense? Why is Int(3551) not simply 3551??

Print this item