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
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
' 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
' 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
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"
' 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)
'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
' /////////////////////////////////////////////////////////////////////////////
' 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
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)
' 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)
' 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
' 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
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
' 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 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
' /////////////////////////////////////////////////////////////////////////////
' 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
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%
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
' /////////////////////////////////////////////////////////////////////////////
' 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
'' ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
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.
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.
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
'----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
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
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
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?
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!