P3D-Legacy/P3D/World/Level.vb

1287 lines
41 KiB
VB.net
Raw Normal View History

2017-02-26 15:52:11 +01:00
Imports System.Threading
2016-09-19 03:26:44 +02:00
''' <summary>
2016-09-07 18:50:38 +02:00
''' A class that manages the collection of entities to represent a map.
''' </summary>
Public Class Level
#Region "Fields"
Private _routeSign As RouteSign = Nothing
Private _world As World = Nothing
Private _pokemonEncounter As PokemonEncounter = Nothing
''' <summary>
''' Stores warp data for warping to a new map.
''' </summary>
Public WarpData As WarpDataStruct
''' <summary>
''' Stores temporary Pokémon encounter data.
''' </summary>
Public PokemonEncounterData As PokemonEcounterDataStruct
' Level states:
2016-09-07 18:50:38 +02:00
Private _isSurfing As Boolean = False
Private _isRiding As Boolean = False
Private _usedStrength As Boolean = False
Private _isDark As Boolean = False
Private _walkedSteps As Integer = 0
Private _offsetMapUpdateDelay As Integer = 50 ' Ticks until the next Offset Map update occurs.
2016-09-07 18:50:38 +02:00
' Map properties:
2017-02-26 15:52:11 +01:00
Private _terrain As Terrain = New Terrain(Terrain.TerrainTypes.Plain)
2016-09-07 18:50:38 +02:00
Private _mapName As String = ""
Private _musicLoop As String = ""
Private _levelFile As String = ""
Private _canTeleport As Boolean = True
Private _canDig As Boolean = False
Private _canFly As Boolean = False
Private _rideType As Integer = 0
Private _weatherType As Integer = 0
Public _DayTime As World.DayTimes = World.GetTime
2016-09-07 18:50:38 +02:00
Private _environmentType As Integer = 0
Private _saveonly As Boolean = False
Private _disableMenus As Boolean = False
2016-09-07 18:50:38 +02:00
Private _wildPokemonGrass As Boolean = True
Private _wildPokemonFloor As Boolean = False
Private _wildPokemonWater As Boolean = True
Private _showOverworldPokemon As Boolean = True
Private _currentRegion As String = "Johto"
2022-07-06 06:44:43 +02:00
Private _regionalForm As String = ""
2016-09-07 18:50:38 +02:00
Private _hiddenabilitychance As Integer = 0
Private _lightingType As Integer = 0
Private _isSafariZone As Boolean = False
Private _isOutside As Boolean = False
2016-09-07 18:50:38 +02:00
Private _isBugCatchingContest As Boolean = False
Private _bugCatchingContestData As String = ""
Private _battleMapData As String = ""
' Entity enumerations:
2016-09-07 18:50:38 +02:00
Private _ownPlayer As OwnPlayer
Private _ownOverworldPokemon As OverworldPokemon
Private _entities As New List(Of Entity)
Private _floors As New List(Of Entity)
Private _shaders As New List(Of Shader)
Private _backdropRenderer As BackdropRenderer
Private _networkPlayers As New List(Of NetworkPlayer)
Private _networkPokemon As New List(Of NetworkPokemon)
Private _offsetMapEntities As New List(Of Entity)
Private _offsetMapFloors As New List(Of Entity)
' Radio:
2016-09-07 18:50:38 +02:00
Private _isRadioOn As Boolean = False
Private _selectedRadioStation As GameJolt.PokegearScreen.RadioStation = Nothing
Private _radioChannels As New List(Of Decimal)
Private _offsetTimer As New Timers.Timer()
Private _isUpdatingOffsetMaps As Boolean = False
#End Region
#Region "Properties"
''' <summary>
''' The Terrain of this level.
''' </summary>
Public ReadOnly Property Terrain() As Terrain
Get
Return Me._terrain
End Get
End Property
''' <summary>
''' A RouteSign on the top left corner of the screen to display the map's name.
''' </summary>
Public ReadOnly Property RouteSign() As RouteSign
Get
Return Me._routeSign
End Get
End Property
''' <summary>
''' Indicates whether the player is Surfing.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property Surfing() As Boolean
Get
Return _isSurfing
End Get
Set(value As Boolean)
Me._isSurfing = value
End Set
End Property
''' <summary>
''' Indicates whether the player is Riding.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property Riding() As Boolean
Get
Return Me._isRiding
End Get
Set(value As Boolean)
Me._isRiding = value
End Set
End Property
''' <summary>
''' Indicates whether the player used Strength already.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property UsedStrength() As Boolean
Get
Return Me._usedStrength
End Get
Set(value As Boolean)
Me._usedStrength = value
End Set
End Property
''' <summary>
''' The reference to the active OwnPlayer instance.
''' </summary>
Public Property OwnPlayer() As OwnPlayer
Get
Return Me._ownPlayer
End Get
Set(value As OwnPlayer)
Me._ownPlayer = value
End Set
End Property
''' <summary>
''' The reference to the active OverworldPokemon instance.
''' </summary>
Public Property OverworldPokemon() As OverworldPokemon
Get
Return Me._ownOverworldPokemon
End Get
Set(value As OverworldPokemon)
Me._ownOverworldPokemon = value
End Set
End Property
''' <summary>
''' The array of entities composing the map.
''' </summary>
Public Property Entities() As List(Of Entity)
Get
Return Me._entities
End Get
Set(value As List(Of Entity))
Me._entities = value
End Set
End Property
''' <summary>
''' The array of floors the player can move on.
''' </summary>
Public Property Floors() As List(Of Entity)
Get
Return Me._floors
End Get
Set(value As List(Of Entity))
Me._floors = value
End Set
End Property
''' <summary>
''' The array of shaders that add specific lighting to the map.
''' </summary>
Public Property Shaders() As List(Of Shader)
Get
Return Me._shaders
End Get
Set(value As List(Of Shader))
Me._shaders = value
End Set
End Property
''' <summary>
''' The array of players on the server to render.
''' </summary>
Public Property NetworkPlayers() As List(Of NetworkPlayer)
Get
Return Me._networkPlayers
End Get
Set(value As List(Of NetworkPlayer))
Me._networkPlayers = value
End Set
End Property
''' <summary>
''' The array of Pokémon on the server to render.
''' </summary>
Public Property NetworkPokemon() As List(Of NetworkPokemon)
Get
Return Me._networkPokemon
End Get
Set(value As List(Of NetworkPokemon))
Me._networkPokemon = value
End Set
End Property
''' <summary>
''' The array of entities the offset maps are composed of.
''' </summary>
Public Property OffsetmapEntities() As List(Of Entity)
Get
Return Me._offsetMapEntities
End Get
Set(value As List(Of Entity))
Me._offsetMapEntities = value
End Set
End Property
''' <summary>
''' The array of floors the offset maps are composed of.
''' </summary>
Public Property OffsetmapFloors() As List(Of Entity)
Get
Return Me._offsetMapFloors
End Get
Set(value As List(Of Entity))
Me._offsetMapFloors = value
End Set
End Property
''' <summary>
''' The name of the current map.
''' </summary>
''' <remarks>This name gets displayed on the RouteSign.</remarks>
Public Property MapName() As String
Get
Return Me._mapName
End Get
Set(value As String)
Me._mapName = value
End Set
End Property
''' <summary>
''' The default background music for this level.
''' </summary>
''' <remarks>Doesn't play for Surfing, Riding and Radio.</remarks>
2016-09-07 18:50:38 +02:00
Public Property MusicLoop() As String
Get
Return Me._musicLoop
End Get
Set(value As String)
Me._musicLoop = value
End Set
End Property
''' <summary>
''' The file this level got loaded from.
''' </summary>
''' <remarks>The path is relative to the \maps\ or \GameMode\[gamemode]\maps\ path.</remarks>
Public Property LevelFile() As String
Get
Return Me._levelFile
End Get
Set(value As String)
Me._levelFile = value
End Set
End Property
''' <summary>
''' Whether the player can use the move Teleport.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property CanTeleport As Boolean
Get
Return Me._canTeleport
End Get
Set(value As Boolean)
Me._canTeleport = value
End Set
End Property
''' <summary>
''' Whether the player can use the move Dig or an Escape Rope.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property CanDig As Boolean
Get
Return Me._canDig
End Get
Set(value As Boolean)
Me._canDig = value
End Set
End Property
''' <summary>
''' Whether the player can use the move Fly.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property CanFly As Boolean
Get
Return Me._canFly
End Get
Set(value As Boolean)
Me._canFly = value
End Set
End Property
''' <summary>
''' The type of Ride the player can use on this map.
''' </summary>
''' <remarks>0 = Depends on CanDig and CanFly, 1 = True, 2 = False, 3 = Can't stop riding</remarks>
2016-09-07 18:50:38 +02:00
Public Property RideType As Integer
Get
Return Me._rideType
End Get
Set(value As Integer)
Me._rideType = value
End Set
End Property
''' <summary>
''' The Weather on this map.
''' </summary>
''' <remarks>For the weather, look at the WeatherTypes enumeration in World.vb</remarks>
Public Property WeatherType As Integer
Get
Return Me._weatherType
End Get
Set(value As Integer)
Me._weatherType = value
End Set
End Property
''' <summary>
''' The DayTime on this map.
''' </summary>
''' <remarks>For the day time, look at the DayTime enumeration in World.vb</remarks>
Public Property DayTime As Integer
Get
Select Case Me._DayTime
Case World.DayTimes.Night
Return 1
Case World.DayTimes.Morning
Return 2
Case World.DayTimes.Day
Return 3
Case World.DayTimes.Evening
Return 4
Case Else
Return World.GetTime
End Select
End Get
Set(value As Integer)
Select Case value
Case 1
Me._DayTime = World.DayTimes.Night
Case 2
Me._DayTime = World.DayTimes.Morning
Case 3
Me._DayTime = World.DayTimes.Day
Case 4
Me._DayTime = World.DayTimes.Evening
Case Else
Me._DayTime = World.GetTime
End Select
End Set
End Property
''' <summary>
''' If only the Save option should be available in the menu for this map.
''' </summary>
Public Property SaveOnly As Boolean
Get
Return Me._saveonly
End Get
Set(value As Boolean)
Me._saveonly = value
End Set
End Property
''' <summary>
''' If the player should be prevented from opening the start menu or accessing the PokéGear..
''' </summary>
Public Property DisableMenus As Boolean
Get
Return Me._disableMenus
End Get
Set(value As Boolean)
Me._disableMenus = value
End Set
End Property
2016-09-07 18:50:38 +02:00
''' <summary>
''' The environment type for this map.
''' </summary>
Public Property EnvironmentType As Integer
Get
Return Me._environmentType
End Get
Set(value As Integer)
Me._environmentType = value
End Set
End Property
''' <summary>
''' Whether the player can encounter wild Pokémon in the Grass entities.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property WildPokemonGrass As Boolean
Get
Return Me._wildPokemonGrass
End Get
Set(value As Boolean)
_wildPokemonGrass = value
End Set
End Property
''' <summary>
''' Whether the player can encounter wild Pokémon on every floor tile.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property WildPokemonFloor As Boolean
Get
Return Me._wildPokemonFloor
End Get
Set(value As Boolean)
Me._wildPokemonFloor = value
End Set
End Property
''' <summary>
''' Whether the player can encounter wild Pokémon while Surfing.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property WildPokemonWater As Boolean
Get
Return Me._wildPokemonWater
End Get
Set(value As Boolean)
Me._wildPokemonWater = value
End Set
End Property
''' <summary>
''' Whether the map is dark, and needs to be lightened up by Flash.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property IsDark As Boolean
Get
Return Me._isDark
End Get
Set(value As Boolean)
Me._isDark = value
End Set
End Property
''' <summary>
''' Whether the Overworld Pokémon is visible.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property ShowOverworldPokemon As Boolean
Get
Return Me._showOverworldPokemon
End Get
Set(value As Boolean)
Me._showOverworldPokemon = value
End Set
End Property
''' <summary>
''' The amount of walked steps on this map.
''' </summary>
Public Property WalkedSteps As Integer
Get
Return Me._walkedSteps
End Get
Set(value As Integer)
Me._walkedSteps = value
End Set
End Property
''' <summary>
''' The region this map is assigned to.
''' </summary>
''' <remarks>The default is "Johto".</remarks>
Public Property CurrentRegion As String
Get
Return Me._currentRegion
End Get
Set(value As String)
Me._currentRegion = value
End Set
End Property
2022-07-06 06:44:43 +02:00
''' <summary>
''' Regional forms available on this level.
''' </summary>
Public Property RegionalForm As String
Get
Return _regionalForm
End Get
Set(value As String)
_regionalForm = value
End Set
End Property
2016-09-07 18:50:38 +02:00
''' <summary>
''' Chance of a Hidden Ability being on a wild Pokémon.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property HiddenAbilityChance As Integer
Get
Return Me._hiddenabilitychance
End Get
Set(value As Integer)
Me._hiddenabilitychance = value
End Set
End Property
''' <summary>
2022-09-15 21:12:12 +02:00
''' The LightingType of this map. More information in the Lighting\UpdateLighting.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property LightingType As Integer
Get
Return Me._lightingType
End Get
Set(value As Integer)
Me._lightingType = value
End Set
End Property
''' <summary>
''' Whether the map is a part of the Safari Zone. This changes the Battle Menu and the Menu Screen.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property IsSafariZone As Boolean
Get
Return Me._isSafariZone
End Get
Set(value As Boolean)
Me._isSafariZone = value
End Set
End Property
''' <summary>
''' Overrides whether the map is outside. Affects item pickup interaction.
''' </summary>
Public Property IsOutside As Boolean
Get
Return Me._isOutside
End Get
Set(value As Boolean)
Me._isOutside = value
End Set
End Property
2016-09-07 18:50:38 +02:00
''' <summary>
''' Whether the map is a part of the Bug Catching Contest. This changes the Battle Menu and the Menu Screen.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property IsBugCatchingContest As Boolean
Get
Return Me._isBugCatchingContest
End Get
Set(value As Boolean)
Me._isBugCatchingContest = value
End Set
End Property
''' <summary>
''' Holds data for the Bug Catching Contest.
''' </summary>
''' <remarks>Composed of 3 values, separated by ",": 0 = script location for ending the contest, 1 = script location for selecting the remaining balls item, 2 = Menu Item name for the remaining balls item.</remarks>
Public Property BugCatchingContestData As String
Get
Return Me._bugCatchingContestData
End Get
Set(value As String)
Me._bugCatchingContestData = value
End Set
End Property
''' <summary>
''' Used to modify the Battle Map camera position.
''' </summary>
''' <remarks>Data: MapName,x,y,z OR Mapname OR x,y,z OR empty</remarks>
Public Property BattleMapData() As String
Get
Return Me._battleMapData
End Get
Set(value As String)
Me._battleMapData = value
End Set
End Property
''' <summary>
''' Used to modify the Battle Map.
''' </summary>
''' <remarks>Data: MapName,x,y,z OR Mapname OR empty</remarks>
Public Property SurfingBattleMapData As String
2016-09-07 18:50:38 +02:00
''' <summary>
''' The instance of the World class, handling time, season and weather based operations.
''' </summary>
Public Property World() As World
Get
Return Me._world
End Get
Set(value As World)
Me._world = value
End Set
End Property
''' <summary>
''' Whether the Radio is currently activated.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property IsRadioOn() As Boolean
Get
Return Me._isRadioOn
End Get
Set(value As Boolean)
Me._isRadioOn = value
End Set
End Property
''' <summary>
''' The currently selected Radio station. If possible, this will replace the Music Loop.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property SelectedRadioStation() As GameJolt.PokegearScreen.RadioStation
Get
Return Me._selectedRadioStation
End Get
Set(value As GameJolt.PokegearScreen.RadioStation)
Me._selectedRadioStation = value
End Set
End Property
''' <summary>
''' Allowed Radio channels on this map.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Property AllowedRadioChannels() As List(Of Decimal)
Get
Return Me._radioChannels
End Get
Set(value As List(Of Decimal))
Me._radioChannels = value
End Set
End Property
''' <summary>
''' Handles wild Pokémon encounters.
''' </summary>
Public ReadOnly Property PokemonEncounter() As PokemonEncounter
Get
Return Me._pokemonEncounter
End Get
End Property
''' <summary>
''' The backdrop renderer of this level.
''' </summary>
Public ReadOnly Property BackdropRenderer() As BackdropRenderer
Get
Return _backdropRenderer
End Get
End Property
2022-07-06 06:44:43 +02:00
2016-09-07 18:50:38 +02:00
#End Region
#Region "Structures"
''' <summary>
''' A structure to store warp data in.
''' </summary>
Public Structure WarpDataStruct
''' <summary>
''' The destination map file.
''' </summary>
Public WarpDestination As String
''' <summary>
''' The position to warp the player to.
''' </summary>
Public WarpPosition As Vector3
''' <summary>
''' The check to see if the player should get warped next tick.
''' </summary>
Public DoWarpInNextTick As Boolean
''' <summary>
''' Amount of 90° rotations counterclockwise.
2016-09-07 18:50:38 +02:00
''' </summary>
Public WarpRotations As Integer
''' <summary>
''' The correct camera yaw to set the camera to after the warping.
''' </summary>
Public CorrectCameraYaw As Single
''' <summary>
''' If the warp action got triggered by a warp block.
''' </summary>
Public IsWarpBlock As Boolean
End Structure
''' <summary>
''' A structure to store wild Pokémon encounter data in.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Structure PokemonEcounterDataStruct
''' <summary>
''' The assumed position the player will be in when encounterning the Pokémon.
''' </summary>
Public Position As Vector3
''' <summary>
''' Whether the player encountered a Pokémon.
2016-09-07 18:50:38 +02:00
''' </summary>
Public EncounteredPokemon As Boolean
''' <summary>
''' The encounter method.
''' </summary>
Public Method As Spawner.EncounterMethods
''' <summary>
''' The link to the .poke file used to spawn the Pokémon in.
''' </summary>
Public PokeFile As String
End Structure
#End Region
''' <summary>
''' Creates a new instance of the Level class.
''' </summary>
Public Sub New()
Me._routeSign = New RouteSign()
Me.WarpData = New WarpDataStruct()
Me.PokemonEncounterData = New PokemonEcounterDataStruct()
Me._pokemonEncounter = New PokemonEncounter(Me)
Me.StartOffsetMapUpdate()
Me._backdropRenderer = New BackdropRenderer()
Me._backdropRenderer.Initialize()
End Sub
''' <summary>
''' Initializes the offset map update cycle.
''' </summary>
Public Sub StartOffsetMapUpdate()
If Not Me._offsetTimer Is Nothing Then
Me._offsetTimer.Stop()
End If
Me._offsetTimer = New Timers.Timer()
Me._offsetTimer.Interval = 16
Me._offsetTimer.AutoReset = True
AddHandler Me._offsetTimer.Elapsed, AddressOf Me.UpdateOffsetMap
Me._offsetTimer.Start()
Logger.Debug("Started Offset map update")
End Sub
Public Sub StopOffsetMapUpdate()
Me._offsetTimer.Stop()
While Me._isUpdatingOffsetMaps
2017-02-26 15:52:11 +01:00
Thread.Sleep(1)
End While
2016-09-07 18:50:38 +02:00
Logger.Debug("Stopped Offset map update")
End Sub
''' <summary>
''' Loads a level from a levelfile.
''' </summary>
''' <param name="Levelpath">The path to load the level from. Start with "|" to prevent loading a levelfile.</param>
Public Sub Load(ByVal Levelpath As String, Optional Reload As Boolean = False)
' copy all changed files
If GameController.IS_DEBUG_ACTIVE Then
DebugFileWatcher.TriggerReload()
End If
' Create a parameter array to pass over to the LevelLoader:
2016-09-07 18:50:38 +02:00
Dim params As New List(Of Object)
params.AddRange({Levelpath, False, New Vector3(0, 0, 0), 0, New List(Of String)})
' Create the world and load the level:
2016-09-07 18:50:38 +02:00
World = New World(0, 0)
If Levelpath.StartsWith("|") = False Then
Me.StopOffsetMapUpdate()
Dim levelLoader As New LevelLoader()
levelLoader.LoadLevel(params.ToArray(), Reload)
2016-09-07 18:50:38 +02:00
Else
Logger.Debug("Don't attempt to load a levelfile.")
End If
' Create own player entity and OverworldPokémon entity and add them to the entity enumeration:
2017-02-26 15:52:11 +01:00
OwnPlayer = New OwnPlayer(0, 0, 0, {TextureManager.DefaultTexture}, Core.Player.Skin, 0, 0, "", "Gold", 0)
2016-09-07 18:50:38 +02:00
OverworldPokemon = New OverworldPokemon(Screen.Camera.Position.X, Screen.Camera.Position.Y, Screen.Camera.Position.Z + 1)
OverworldPokemon.ChangeRotation()
Entities.AddRange({OwnPlayer, OverworldPokemon})
Me.Surfing = Core.Player.startSurfing
If Me.Surfing = True And OwnPlayer.SkinName = Core.Player.Skin Then
With Screen.Level.OwnPlayer
Core.Player.TempSurfSkin = .SkinName
Dim pokemonNumber As Integer = Core.Player.Pokemons(Core.Player.SurfPokemon).Number
Dim SkinName As String = "[POKEMON|N]" & pokemonNumber & PokemonForms.GetOverworldAddition(Core.Player.Pokemons(Core.Player.SurfPokemon))
If Core.Player.Pokemons(Core.Player.SurfPokemon).IsShiny = True Then
SkinName = "[POKEMON|S]" & pokemonNumber & PokemonForms.GetOverworldAddition(Core.Player.Pokemons(Core.Player.SurfPokemon))
End If
.SetTexture(SkinName, False)
.UpdateEntity()
If MusicManager.CurrentSong.Name.ToLower <> "surf" AndAlso (Screen.Level.IsRadioOn = False OrElse GameJolt.PokegearScreen.StationCanPlay(Screen.Level.SelectedRadioStation) = False) Then
MusicManager.Play("surf", True)
End If
End With
End If
2016-09-07 18:50:38 +02:00
Me.StartOffsetMapUpdate()
End Sub
''' <summary>
''' Renders the level.
''' </summary>
Public Sub Draw()
Me._backdropRenderer.Draw()
' Set the effect's View and Projection matrices:
2016-09-07 18:50:38 +02:00
Screen.Effect.View = Screen.Camera.View
Screen.Effect.Projection = Screen.Camera.Projection
' Reset the Debug values:
2016-09-07 18:50:38 +02:00
DebugDisplay.DrawnVertices = 0
DebugDisplay.MaxVisibleVertices = 0
2016-09-07 18:50:38 +02:00
DebugDisplay.MaxVertices = 0
DebugDisplay.MaxDistance = 0
Dim AllEntities As New List(Of Entity)
Dim AllFloors As New List(Of Entity)
AllEntities.AddRange(Entities)
AllFloors.AddRange(Floors)
2016-09-07 18:50:38 +02:00
If Core.GameOptions.LoadOffsetMaps > 0 Then
AllEntities.AddRange(OffsetmapEntities)
AllFloors.AddRange(OffsetmapFloors)
2016-09-07 18:50:38 +02:00
End If
AllEntities = (From f In AllEntities Order By f.CameraDistance Descending).ToList()
AllFloors = (From f In AllFloors Order By f.CameraDistance Descending).ToList()
'Render floors:
For i = 0 To AllFloors.Count - 1
If i <= AllFloors.Count - 1 Then
AllFloors(i).Render()
If AllFloors(i).Visible = True Then
DebugDisplay.MaxVisibleVertices += AllFloors(i).VertexCount
End If
DebugDisplay.MaxVertices += AllFloors(i).VertexCount
2016-09-07 18:50:38 +02:00
End If
Next
'Render all other entities:
For i = 0 To AllEntities.Count - 1
If i <= AllEntities.Count - 1 Then
AllEntities(i).Render()
If AllEntities(i).Visible = True Then
DebugDisplay.MaxVisibleVertices += AllEntities(i).VertexCount
End If
DebugDisplay.MaxVertices += AllEntities(i).VertexCount
2016-09-07 18:50:38 +02:00
End If
Next
If IsDark = True Then
DrawFlashOverlay()
End If
End Sub
''' <summary>
''' Updates the level's logic.
''' </summary>
Public Sub Update()
Me._backdropRenderer.Update()
Me.UpdatePlayerWarp()
Me._pokemonEncounter.TriggerBattle()
2016-09-07 18:50:38 +02:00
' Reload map from file (Debug or Sandbox Mode):
2016-09-07 18:50:38 +02:00
If GameController.IS_DEBUG_ACTIVE = True Or Core.Player.SandBoxMode = True Then
If KeyBoardHandler.KeyPressed(Keys.R) = True And Core.CurrentScreen.Identification = Screen.Identifications.OverworldScreen Then
Core.OffsetMaps.Clear()
Logger.Debug(String.Format("Reload map file: {0}", Me._levelFile))
Me.Load(LevelFile, True)
2016-09-07 18:50:38 +02:00
End If
End If
' Update all network players and Pokémon:
2016-09-07 18:50:38 +02:00
If JoinServerScreen.Online = True Then
Core.ServersManager.PlayerManager.UpdatePlayers()
End If
' Call Update and UpdateEntity methods of all entities:
2016-09-07 18:50:38 +02:00
Me.UpdateEntities()
End Sub
''' <summary>
''' Updates all entities on the map and offset map and sorts the enumarations.
''' </summary>
Public Sub UpdateEntities()
' Update and remove entities:
2016-09-07 18:50:38 +02:00
If LevelLoader.IsBusy = False Then
For i = 0 To Entities.Count - 1
If i <= Entities.Count - 1 Then
2017-02-26 15:52:11 +01:00
If Entities.Count - 1 >= i AndAlso Entities(i).CanBeRemoved Then
2016-09-07 18:50:38 +02:00
Entities.RemoveAt(i)
i -= 1
Else
2017-02-26 15:52:11 +01:00
If Entities(i).NeedsUpdate Then
2016-09-07 18:50:38 +02:00
Entities(i).Update()
End If
2017-02-26 15:52:11 +01:00
' UpdateEntity for all entities:
Me.Entities(i).UpdateEntity()
2016-09-07 18:50:38 +02:00
End If
Else
Exit For
End If
Next
End If
' UpdateEntity for all floors:
2016-09-07 18:50:38 +02:00
For i = 0 To Me.Floors.Count - 1
If i <= Me.Floors.Count - 1 Then
Me.Floors(i).UpdateEntity()
End If
Next
Me.SortEntities()
End Sub
''' <summary>
''' Sorts the entity enumerations.
''' </summary>
Public Sub SortEntities()
If LevelLoader.IsBusy = False Then
Entities = (From f In Entities Order By f.CameraDistance Descending).ToList()
End If
End Sub
''' <summary>
''' Sorts and updates offset map entities.
''' </summary>
Public Sub UpdateOffsetMap()
Me._isUpdatingOffsetMaps = True
If Core.GameOptions.LoadOffsetMaps > 0 Then
' The Update function of entities on offset maps are not getting called.
2016-09-07 18:50:38 +02:00
If Me._offsetMapUpdateDelay <= 0 Then ' Only when the delay is 0, update.
' Sort the list:
2016-09-07 18:50:38 +02:00
If LevelLoader.IsBusy = False Then
OffsetmapEntities = (From e In OffsetmapEntities Order By e.CameraDistance Descending).ToList()
End If
Me._offsetMapUpdateDelay = Core.GameOptions.LoadOffsetMaps - 1 'Set the new delay
' Remove entities that CanBeRemoved (see what I did there?):
' Now it also updates the remaining entities.
2016-09-07 18:50:38 +02:00
For i = 0 To OffsetmapEntities.Count - 1
If i <= OffsetmapEntities.Count - 1 Then
If OffsetmapEntities(i).CanBeRemoved Then
2016-09-07 18:50:38 +02:00
OffsetmapEntities.RemoveAt(i)
i -= 1
Else
2023-10-03 09:43:58 +02:00
If TryCast(OffsetmapEntities(i), NPC) IsNot Nothing Then
CType(OffsetmapEntities(i), NPC).Update()
End If
OffsetmapEntities(i).UpdateEntity()
2016-09-07 18:50:38 +02:00
End If
Else
Exit For
End If
Next
' Call UpdateEntity on all offset map floors:
2016-09-07 18:50:38 +02:00
For i = Me.OffsetmapFloors.Count - 1 To 0 Step -1
If i <= Me.OffsetmapFloors.Count - 1 Then
Me.OffsetmapFloors(i).UpdateEntity()
End If
Next
Else
Me._offsetMapUpdateDelay -= 1
End If
End If
Me._isUpdatingOffsetMaps = False
End Sub
''' <summary>
''' Renders offset map entities.
''' </summary>
Private Sub RenderOffsetMap()
' Render floors:
2016-09-07 18:50:38 +02:00
For i = 0 To Me.OffsetmapFloors.Count - 1
If i <= Me.OffsetmapFloors.Count - 1 Then
If Not Me.OffsetmapFloors(i) Is Nothing Then
Me.OffsetmapFloors(i).Render()
If OffsetmapFloors(i).Visible = True Then
DebugDisplay.MaxVisibleVertices += OffsetmapFloors(i).VertexCount
End If
2016-09-07 18:50:38 +02:00
DebugDisplay.MaxVertices += Me.OffsetmapFloors(i).VertexCount
End If
End If
Next
' Render entities:
2016-09-07 18:50:38 +02:00
For i = 0 To Me.OffsetmapEntities.Count - 1
If i <= Me.OffsetmapEntities.Count - 1 Then
If Not Me.OffsetmapEntities(i) Is Nothing Then
If OffsetmapEntities(i).Visible = True Then
DebugDisplay.MaxVisibleVertices += OffsetmapEntities(i).VertexCount
End If
2016-09-07 18:50:38 +02:00
Me.OffsetmapEntities(i).Render()
DebugDisplay.MaxVertices += Me.OffsetmapEntities(i).VertexCount
End If
End If
Next
End Sub
''' <summary>
''' Draws the flash overlay to the screen.
''' </summary>
Private Sub DrawFlashOverlay()
Core.SpriteBatch.Draw(TextureManager.GetTexture("GUI\Overworld\flash_overlay"), New Rectangle(0, 0, Core.windowSize.Width, Core.windowSize.Height), Color.White)
End Sub
''' <summary>
''' Handles warp events for the player.
''' </summary>
Private Sub UpdatePlayerWarp()
If WarpData.DoWarpInNextTick = True Then ' If a warp event got scheduled.
' Disable wild Pokémon:
2016-09-07 18:50:38 +02:00
Me._wildPokemonFloor = False
Me.PokemonEncounterData.EncounteredPokemon = False
' Set the Surfing flag for the next map:
2016-09-07 18:50:38 +02:00
Core.Player.startSurfing = Surfing
' Change the player position:
2016-09-07 18:50:38 +02:00
Screen.Camera.Position = WarpData.WarpPosition
Dim tempProperties As String = Me.CanDig.ToString() & "," & Me.CanFly.ToString() ' Store properties to determine if the "enter" sound should be played.
2016-09-07 18:50:38 +02:00
' Store skin values:
2016-09-07 18:50:38 +02:00
Dim usingGameJoltTexture As Boolean = OwnPlayer.UsingGameJoltTexture
Core.Player.Skin = OwnPlayer.SkinName
' Load the new level:
2016-09-07 18:50:38 +02:00
Dim params As New List(Of Object)
params.AddRange({WarpData.WarpDestination, False, New Vector3(0, 0, 0), 0, New List(Of String)})
World = New World(0, 0)
Me.StopOffsetMapUpdate()
Dim levelLoader As New LevelLoader()
levelLoader.LoadLevel(params.ToArray())
Core.Player.AddVisitedMap(Me.LevelFile) ' Add new map to visited maps list.
UsedStrength = False ' Disable Strength usuage upon map switch.
Me.Surfing = Core.Player.startSurfing ' Set the Surfing property after map switch.
2016-09-07 18:50:38 +02:00
' Create player and Pokémon entities:
2017-02-26 15:52:11 +01:00
OwnPlayer = New OwnPlayer(0, 0, 0, {TextureManager.DefaultTexture}, Core.Player.Skin, 0, 0, "", "Gold", 0)
2016-09-07 18:50:38 +02:00
OwnPlayer.SetTexture(Core.Player.Skin, usingGameJoltTexture)
OverworldPokemon = New OverworldPokemon(Screen.Camera.Position.X, Screen.Camera.Position.Y, Screen.Camera.Position.Z + 1)
OverworldPokemon.Visible = False
OverworldPokemon.warped = True
Entities.AddRange({OwnPlayer, OverworldPokemon})
' Set Ride skin, if needed:
2016-09-07 18:50:38 +02:00
If Riding = True And CanRide() = False Then
Riding = False
OwnPlayer.SetTexture(Core.Player.TempRideSkin, True)
Core.Player.Skin = Core.Player.TempRideSkin
End If
' If any turns after the warp are defined, apply them:
2016-09-07 18:50:38 +02:00
Screen.Camera.InstantTurn(WarpData.WarpRotations)
' Make the RouteSign appear:
2016-09-07 18:50:38 +02:00
Me._routeSign.Setup(MapName)
' Play the correct music track:
If MusicManager.ForceMusic = "" Then
If IsRadioOn = True AndAlso GameJolt.PokegearScreen.StationCanPlay(Me.SelectedRadioStation) = True Then
MusicManager.Play(SelectedRadioStation.Music, True)
2016-09-07 18:50:38 +02:00
Else
IsRadioOn = False
If Me.Surfing = True Then
MusicManager.Play("surf", True)
2016-09-07 18:50:38 +02:00
Else
If Me.Riding = True Then
MusicManager.Play("ride", True)
Else
If MusicManager.GetSong(MusicLoop) IsNot Nothing Then
MusicManager.Play(MusicLoop, True, 0.01F)
Else
If MusicManager._currentSongName Is "silence" Or MusicManager.CurrentSong Is Nothing Then
MusicManager.Play("silence")
End If
End If
End If
2016-09-07 18:50:38 +02:00
End If
End If
Else
If MusicManager._currentSongName <> MusicManager.ForceMusic Then
MusicManager.Play(MusicManager.ForceMusic, True)
End If
2016-09-07 18:50:38 +02:00
End If
' Initialize the world with newly loaded environment variables:
2016-09-07 18:50:38 +02:00
World.Initialize(Screen.Level.EnvironmentType, Screen.Level.WeatherType)
' If this map is on the restplaces list, set the player's last restplace to this map:
2016-09-07 18:50:38 +02:00
Dim restplaces As List(Of String) = System.IO.File.ReadAllLines(GameModeManager.GetMapPath("restplaces.dat")).ToList()
For Each line As String In restplaces
Dim place As String = line.GetSplit(0, "|")
If place = LevelFile Then
Core.Player.LastRestPlace = place
Core.Player.LastRestPlacePosition = line.GetSplit(1, "|")
End If
Next
' If the warp happened through a warp block, make the player walk one step forward after switching to the new map:
If WarpData.IsWarpBlock = True Then
2016-09-07 18:50:38 +02:00
Screen.Camera.StopMovement()
Screen.Camera.Move(1.0F)
End If
' Because of the map change, Roaming Pokémon are moving to their next location on the world map:
2016-09-07 18:50:38 +02:00
RoamingPokemon.ShiftRoamingPokemon(-1)
' Check if the enter sound should be played by checking if CanDig or CanFly properties are different from the last map:
2016-09-07 18:50:38 +02:00
If tempProperties <> Me.CanDig.ToString() & "," & Me.CanFly.ToString() Then
SoundManager.PlaySound("enter", False)
Fix audio engine & contentpacks (#35) * Replaced existing Gen 2 SFX with better sounding ones (no ugly reverb) Replaced MediaPlayer with NAudio(and NAudio.Vorbis) for playing music in order to fix random stopping issues. The game now directly loads .ogg files instead of the .wma/.xnb combination from before. ContentPacks are now able to replace Music and SFX again (I haven't added a menu yet for choosing ContentPacks). To make older GameModes work with current versions, you can add the GameMode property EnterType and set it to "1", this will make the game use the older 2D NewGameScreen. * Delete GameController.vb * Add gamecontroller.vb back * Fix sfx missing * Battleintro doodle now doesn't loop anymore (for no trainer) Changed the shutter sound (aka Large Door sound) to something more large door-y Made the enter sound slightly louder The enter sound now plays when going through any warp to or from an indoor map (including falling through holes) The flying sound effect is played earlier in the animation after selecting a location Changed played sound effect when using the Escape Rope to "teleport" instead of "destroy" The bump noise now also plays when bumping into something in first person Fixed small gap between the end of the intro song and the start of the main song Replaced some songs with better songs * Fixed some more intro issues * Forgot to change a thing * Fixed an error where the main music would play, ignoring the muted musicmanager. * fix indenting in musicmanager * The music player will now only start playback on a new song if the music player is not muted, fixed the end time calculation of the intro of a song after muting, Music can't be unmuted now as long as a sound effect plays that stops the music. * Fixed league restplace position, fixed sound effects sharing the volume slider of the music, sound effects are now also muted when pressing M, changed music on/off popup to audio on/off, removed bump delay in first person, added more control on whether played songs should be looping or not. * Fixed some more scripts that turn on thirdperson mode but don't check if it was on before or set it back to what it was before afterwards, also fixed a small error in creditsscreen.vb. * Fixed indenting error in musicmanager.vb, fixed an error of mine where the loopsong parameter would be seen as the playintro parameter. * Added more music commands, added quite some menu select noises, will add more later * More select sound effects! * Fix music not resuming after soundeffect * Trainer using item now plays the single_heal soundeffect * Pokémon cries now sound louder * Possibly fixing crash when playing Pokémon cry at volume higher than 0.71 * Added better quality Pokémon cries, made random overworld cries slightly less loud, added cries for 719 and 720. * Sound effects now sound louder * Revert "Added better quality Pokémon cries, made random overworld cries slightly less loud, added cries for 719 and 720." This reverts commit 8c9296ed1a82144d17f303a52c3f2e9e65a5bfea. * Fixed the cause of why the title screen music plays even when the game is muted * Tabs to spaces * Revert Co-authored-by: darkfire006 <blazer257@live.com> Co-authored-by: JappaWakkaP3D <66885565+JappaWakkaP3D@users.noreply.github.com> Co-authored-by: JappaWakkaP3D <jasper.speelman@outlook.com> Co-authored-by: Vitaly Mikhailov <personal@aragas.org>
2020-07-09 19:59:42 +02:00
ElseIf tempProperties = "True,False" And Me.CanDig = True And Me.CanFly = False Then
SoundManager.PlaySound("enter", False)
ElseIf tempProperties = "False,False" And Me.CanDig = False And Me.CanFly = False Then
SoundManager.PlaySound("enter", False)
2016-09-07 18:50:38 +02:00
End If
' Unlock the yaw on the camera:
2016-09-07 18:50:38 +02:00
CType(Screen.Camera, OverworldCamera).YawLocked = False
NetworkPlayer.ScreenRegionChanged()
' If a warp occured, update the camera:
2016-09-07 18:50:38 +02:00
Screen.Camera.Update()
' Disable the warp check:
2016-09-07 18:50:38 +02:00
Me.WarpData.DoWarpInNextTick = False
WarpData.IsWarpBlock = False
If Core.ServersManager.ServerConnection.Connected = True Then
' Update network players:
2016-09-07 18:50:38 +02:00
Core.ServersManager.PlayerManager.NeedsUpdate = True
End If
Fix audio engine & contentpacks (#35) * Replaced existing Gen 2 SFX with better sounding ones (no ugly reverb) Replaced MediaPlayer with NAudio(and NAudio.Vorbis) for playing music in order to fix random stopping issues. The game now directly loads .ogg files instead of the .wma/.xnb combination from before. ContentPacks are now able to replace Music and SFX again (I haven't added a menu yet for choosing ContentPacks). To make older GameModes work with current versions, you can add the GameMode property EnterType and set it to "1", this will make the game use the older 2D NewGameScreen. * Delete GameController.vb * Add gamecontroller.vb back * Fix sfx missing * Battleintro doodle now doesn't loop anymore (for no trainer) Changed the shutter sound (aka Large Door sound) to something more large door-y Made the enter sound slightly louder The enter sound now plays when going through any warp to or from an indoor map (including falling through holes) The flying sound effect is played earlier in the animation after selecting a location Changed played sound effect when using the Escape Rope to "teleport" instead of "destroy" The bump noise now also plays when bumping into something in first person Fixed small gap between the end of the intro song and the start of the main song Replaced some songs with better songs * Fixed some more intro issues * Forgot to change a thing * Fixed an error where the main music would play, ignoring the muted musicmanager. * fix indenting in musicmanager * The music player will now only start playback on a new song if the music player is not muted, fixed the end time calculation of the intro of a song after muting, Music can't be unmuted now as long as a sound effect plays that stops the music. * Fixed league restplace position, fixed sound effects sharing the volume slider of the music, sound effects are now also muted when pressing M, changed music on/off popup to audio on/off, removed bump delay in first person, added more control on whether played songs should be looping or not. * Fixed some more scripts that turn on thirdperson mode but don't check if it was on before or set it back to what it was before afterwards, also fixed a small error in creditsscreen.vb. * Fixed indenting error in musicmanager.vb, fixed an error of mine where the loopsong parameter would be seen as the playintro parameter. * Added more music commands, added quite some menu select noises, will add more later * More select sound effects! * Fix music not resuming after soundeffect * Trainer using item now plays the single_heal soundeffect * Pokémon cries now sound louder * Possibly fixing crash when playing Pokémon cry at volume higher than 0.71 * Added better quality Pokémon cries, made random overworld cries slightly less loud, added cries for 719 and 720. * Sound effects now sound louder * Revert "Added better quality Pokémon cries, made random overworld cries slightly less loud, added cries for 719 and 720." This reverts commit 8c9296ed1a82144d17f303a52c3f2e9e65a5bfea. * Fixed the cause of why the title screen music plays even when the game is muted * Tabs to spaces * Revert Co-authored-by: darkfire006 <blazer257@live.com> Co-authored-by: JappaWakkaP3D <66885565+JappaWakkaP3D@users.noreply.github.com> Co-authored-by: JappaWakkaP3D <jasper.speelman@outlook.com> Co-authored-by: Vitaly Mikhailov <personal@aragas.org>
2020-07-09 19:59:42 +02:00
2016-09-07 18:50:38 +02:00
End If
End Sub
''' <summary>
''' Returns a list of all NPCs on the map.
''' </summary>
Public Function GetNPCs() As List(Of NPC)
Dim reList As New List(Of NPC)
For Each Entity As Entity In Me.Entities
If Entity.EntityID = "NPC" Then
reList.Add(CType(Entity, NPC))
End If
Next
Return reList
End Function
''' <summary>
''' Returns an NPC based on their ID.
''' </summary>
''' <param name="ID">The ID of the NPC to return from the level.</param>
''' <returns>Returns either a matching NPC or Nothing.</returns>
Public Function GetNPC(ByVal ID As Integer) As NPC
For Each NPC As NPC In GetNPCs()
If NPC.NPCID = ID Then
Return NPC
End If
Next
Return Nothing
End Function
''' <summary>
''' Returns an NPC based on the entity ID.
''' </summary>
Public Function GetEntity(ByVal ID As Integer) As Entity
If ID = -1 Then
Throw New Exception("-1 is the default value for NOT having an ID, therefore is not a valid ID.")
Else
For Each ent As Entity In Me.Entities
If ent.ID = ID Then
Return ent
End If
Next
End If
Return Nothing
End Function
''' <summary>
''' Checks all NPCs on the map for if the player is in their line of sight.
''' </summary>
Public Sub CheckTrainerSights()
For Each Entity As Entity In Entities
If Entity.EntityID = "NPC" Then
Dim NPC As NPC = CType(Entity, NPC)
If NPC.IsTrainer = True Then
NPC.CheckInSight()
End If
End If
Next
End Sub
''' <summary>
''' Determines whether the player can use Ride on this map.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Function CanRide() As Boolean
If GameController.IS_DEBUG_ACTIVE = True Or Core.Player.SandBoxMode = True Then 'Always true for Sandboxmode and Debug mode.
Return True
End If
If RideType > 0 Then
Select Case RideType
Case 1
Return True
Case 2
Return False
Case 3
Return True
2016-09-07 18:50:38 +02:00
End Select
End If
If Screen.Level.CanDig = False And Screen.Level.CanFly = False Then
Return False
Else
Return True
End If
End Function
''' <summary>
''' Whether the player can move based on the entity around him.
2016-09-07 18:50:38 +02:00
''' </summary>
Public Function CanMove() As Boolean
For Each e As Entity In Me.Entities
If e.Position.X = Screen.Camera.Position.X And e.Position.Z = Screen.Camera.Position.Z And CInt(e.Position.Y) = CInt(Screen.Camera.Position.Y) Then
Return e.LetPlayerMove()
End If
Next
Return True
End Function
End Class