oldschool-toontown/toontown/ai/ToontownAIRepository.py

475 lines
19 KiB
Python
Raw Normal View History

2019-11-11 00:46:41 +00:00
from direct.directnotify import DirectNotifyGlobal
from panda3d.core import *
from panda3d.toontown import *
2019-11-17 21:29:23 +00:00
from otp.ai.AIZoneData import AIZoneDataStore
from otp.ai.TimeManagerAI import TimeManagerAI
2019-11-17 21:29:23 +00:00
from otp.distributed.OtpDoGlobals import *
from toontown.ai.HolidayManagerAI import HolidayManagerAI
2019-11-17 21:57:18 +00:00
from toontown.ai.NewsManagerAI import NewsManagerAI
2019-12-06 02:38:58 +00:00
from toontown.ai.WelcomeValleyManagerAI import WelcomeValleyManagerAI
2019-11-23 01:58:35 +00:00
from toontown.building.DistributedTrophyMgrAI import DistributedTrophyMgrAI
2019-11-17 21:29:23 +00:00
from toontown.catalog.CatalogManagerAI import CatalogManagerAI
2020-01-09 01:10:31 +00:00
from toontown.coghq.CogSuitManagerAI import CogSuitManagerAI
2020-01-11 14:50:33 +00:00
from toontown.coghq.CountryClubManagerAI import CountryClubManagerAI
2020-01-08 20:51:15 +00:00
from toontown.coghq.FactoryManagerAI import FactoryManagerAI
2020-01-10 21:23:33 +00:00
from toontown.coghq.LawOfficeManagerAI import LawOfficeManagerAI
2020-01-10 02:42:01 +00:00
from toontown.coghq.MintManagerAI import MintManagerAI
from toontown.coghq.PromotionManagerAI import PromotionManagerAI
from toontown.distributed.ToontownDistrictAI import ToontownDistrictAI
from toontown.distributed.ToontownDistrictStatsAI import ToontownDistrictStatsAI
from toontown.distributed.ToontownInternalRepository import ToontownInternalRepository
2019-11-23 01:58:35 +00:00
from toontown.hood import ZoneUtil
from toontown.hood.BRHoodDataAI import BRHoodDataAI
2020-01-11 14:50:33 +00:00
from toontown.hood.BossbotHQDataAI import BossbotHQDataAI
2020-01-08 20:51:15 +00:00
from toontown.hood.CSHoodDataAI import CSHoodDataAI
2020-01-10 02:46:11 +00:00
from toontown.hood.CashbotHQDataAI import CashbotHQDataAI
from toontown.hood.DDHoodDataAI import DDHoodDataAI
from toontown.hood.DGHoodDataAI import DGHoodDataAI
from toontown.hood.DLHoodDataAI import DLHoodDataAI
from toontown.hood.GSHoodDataAI import GSHoodDataAI
2020-01-08 02:12:08 +00:00
from toontown.hood.GZHoodDataAI import GZHoodDataAI
2020-01-10 21:23:33 +00:00
from toontown.hood.LawbotHQDataAI import LawbotHQDataAI
from toontown.hood.MMHoodDataAI import MMHoodDataAI
2020-01-08 06:40:54 +00:00
from toontown.hood.OZHoodDataAI import OZHoodDataAI
from toontown.hood.TTHoodDataAI import TTHoodDataAI
2019-11-23 01:58:35 +00:00
from toontown.pets.PetManagerAI import PetManagerAI
from toontown.quest.QuestManagerAI import QuestManagerAI
2020-01-02 00:55:54 +00:00
from toontown.racing import RaceGlobals
2019-12-06 03:03:12 +00:00
from toontown.racing.DistributedLeaderBoardAI import DistributedLeaderBoardAI
2019-12-06 03:42:29 +00:00
from toontown.racing.DistributedRacePadAI import DistributedRacePadAI
2020-01-02 17:46:49 +00:00
from toontown.racing.DistributedStartingBlockAI import DistributedStartingBlockAI
from toontown.racing.DistributedStartingBlockAI import DistributedViewingBlockAI
2019-12-06 03:42:29 +00:00
from toontown.racing.DistributedViewPadAI import DistributedViewPadAI
2019-12-06 03:17:10 +00:00
from toontown.racing.RaceManagerAI import RaceManagerAI
from toontown.safezone.SafeZoneManagerAI import SafeZoneManagerAI
from toontown.shtiker.CogPageManagerAI import CogPageManagerAI
2021-07-07 06:51:22 +00:00
from toontown.spellbook.ToontownMagicWordManagerAI import ToontownMagicWordManagerAI
2019-11-23 01:58:35 +00:00
from toontown.suit.SuitInvasionManagerAI import SuitInvasionManagerAI
from toontown.toon import NPCToons
2019-11-23 01:58:35 +00:00
from toontown.toonbase import ToontownGlobals
from toontown.uberdog.DistributedInGameNewsMgrAI import DistributedInGameNewsMgrAI
import os
2019-11-11 00:46:41 +00:00
class ToontownAIRepository(ToontownInternalRepository):
notify = DirectNotifyGlobal.directNotify.newCategory('ToontownAIRepository')
def __init__(self, baseChannel, serverId, districtName):
ToontownInternalRepository.__init__(self, baseChannel, serverId, dcSuffix='AI')
self.districtName = districtName
2019-11-20 22:05:59 +00:00
self.doLiveUpdates = config.GetBool('want-live-updates', True)
2019-11-23 01:58:35 +00:00
self.wantCogdominiums = config.GetBool('want-cogdominiums', True)
self.useAllMinigames = config.GetBool('want-all-minigames', True)
self.dataFolder = config.GetString('server-data-folder', '')
if self.dataFolder:
self.dataFolder = self.dataFolder + '/'
2019-11-11 00:46:41 +00:00
self.districtId = None
self.district = None
2019-11-23 01:18:26 +00:00
self.districtStats = None
2019-11-17 21:29:23 +00:00
self.holidayManager = None
self.zoneDataStore = None
2019-11-23 01:58:35 +00:00
self.petMgr = None
self.suitInvasionManager = None
self.zoneAllocator = None
self.zoneId2owner = {}
self.questManager = None
self.promotionMgr = None
self.cogPageManager = None
2019-12-06 03:17:10 +00:00
self.raceMgr = None
2020-01-11 14:50:33 +00:00
self.countryClubMgr = None
2020-01-08 20:51:15 +00:00
self.factoryMgr = None
2020-01-10 02:42:01 +00:00
self.mintMgr = None
2020-01-10 21:23:33 +00:00
self.lawMgr = None
2020-01-09 01:10:31 +00:00
self.cogSuitMgr = None
self.timeManager = None
2019-11-20 22:05:59 +00:00
self.newsManager = None
2019-12-06 02:38:58 +00:00
self.welcomeValleyManager = None
2019-11-17 21:29:23 +00:00
self.inGameNewsMgr = None
self.catalogManager = None
2019-11-23 01:58:35 +00:00
self.trophyMgr = None
self.safeZoneManager = None
2021-07-07 06:51:22 +00:00
self.magicWordManager = None
2019-11-23 01:58:35 +00:00
self.zoneTable = {}
self.dnaStoreMap = {}
self.dnaDataMap = {}
self.hoods = []
self.buildingManagers = {}
self.suitPlanners = {}
2019-11-11 00:46:41 +00:00
def handleConnected(self):
ToontownInternalRepository.handleConnected(self)
# Generate our district...
self.notify.info('Generating district...')
2019-11-11 00:46:41 +00:00
self.districtId = self.allocateChannel()
self.district = ToontownDistrictAI(self)
self.district.setName(self.districtName)
self.district.generateWithRequiredAndId(self.districtId, self.getGameDoId(), OTP_ZONE_ID_DISTRICTS)
2019-11-11 01:50:13 +00:00
# Claim ownership of that district...
self.notify.info('Declaring ownership...')
2019-11-11 01:50:13 +00:00
self.district.setAI(self.ourChannel)
# Setup necessary files and things.
self.setupFiles()
2019-11-17 21:29:23 +00:00
# Create our local objects.
self.notify.info('Creating local objects...')
2019-11-17 21:29:23 +00:00
self.createLocals()
# Create our global objects.
self.notify.info('Creating global objects...')
2019-11-17 21:29:23 +00:00
self.createGlobals()
2019-11-23 01:58:35 +00:00
# Create our zones.
self.notify.info('Creating zones...')
2019-11-23 01:58:35 +00:00
self.createZones()
2019-11-11 01:50:13 +00:00
# Make our district available, and we're done.
self.district.b_setAvailable(True)
self.notify.info('Done.')
2019-11-17 21:29:23 +00:00
def createLocals(self):
"""
Creates "local" (non-distributed) objects.
"""
# Create our holiday manager...
self.holidayManager = HolidayManagerAI(self)
# Create our zone data store...
self.zoneDataStore = AIZoneDataStore()
2019-11-23 01:58:35 +00:00
# Create our pet manager...
self.petMgr = PetManagerAI(self)
# Create our suit invasion manager...
self.suitInvasionManager = SuitInvasionManagerAI(self)
# Create our zone allocator...
self.zoneAllocator = UniqueIdAllocator(ToontownGlobals.DynamicZonesBegin, ToontownGlobals.DynamicZonesEnd)
# Create our quest manager...
self.questManager = QuestManagerAI(self)
# Create our promotion manager...
self.promotionMgr = PromotionManagerAI(self)
# Create our Cog page manager...
self.cogPageManager = CogPageManagerAI(self)
2019-12-06 03:17:10 +00:00
# Create our race manager...
self.raceMgr = RaceManagerAI(self)
2020-01-11 14:50:33 +00:00
# Create our country club manager...
self.countryClubMgr = CountryClubManagerAI(self)
2020-01-08 20:51:15 +00:00
# Create our factory manager...
self.factoryMgr = FactoryManagerAI(self)
2020-01-10 02:42:01 +00:00
# Create our mint manager...
self.mintMgr = MintManagerAI(self)
2020-01-10 21:23:33 +00:00
# Create our law office manager...
self.lawMgr = LawOfficeManagerAI(self)
2020-01-09 01:10:31 +00:00
# Create our Cog suit manager...
self.cogSuitMgr = CogSuitManagerAI(self)
2019-11-17 21:29:23 +00:00
def createGlobals(self):
"""
Creates "global" (distributed) objects.
"""
2019-11-23 01:18:26 +00:00
# Generate our district stats...
self.districtStats = ToontownDistrictStatsAI(self)
self.districtStats.settoontownDistrictId(self.districtId)
self.districtStats.generateWithRequiredAndId(self.allocateChannel(), self.district.getDoId(),
OTP_ZONE_ID_DISTRICTS_STATS)
# Generate our time manager...
self.timeManager = TimeManagerAI(self)
self.timeManager.generateWithRequired(OTP_ZONE_ID_MANAGEMENT)
2019-11-17 21:57:18 +00:00
# Generate our news manager...
self.newsManager = NewsManagerAI(self)
self.newsManager.generateWithRequired(OTP_ZONE_ID_MANAGEMENT)
2019-12-06 02:38:58 +00:00
# Generate our Welcome Valley manager...
self.welcomeValleyManager = WelcomeValleyManagerAI(self)
self.welcomeValleyManager.generateWithRequired(OTP_ZONE_ID_MANAGEMENT)
2019-11-17 21:29:23 +00:00
# Generate our in-game news manager...
self.inGameNewsMgr = DistributedInGameNewsMgrAI(self)
self.inGameNewsMgr.generateWithRequired(OTP_ZONE_ID_MANAGEMENT)
# Generate our catalog manager...
self.catalogManager = CatalogManagerAI(self)
2019-11-20 20:12:05 +00:00
self.catalogManager.generateWithRequired(OTP_ZONE_ID_MANAGEMENT)
2019-11-17 21:29:23 +00:00
2019-11-23 01:58:35 +00:00
# Generate our trophy manager...
self.trophyMgr = DistributedTrophyMgrAI(self)
self.trophyMgr.generateWithRequired(OTP_ZONE_ID_MANAGEMENT)
# Generate our safezone manager...
self.safeZoneManager = SafeZoneManagerAI(self)
self.safeZoneManager.generateWithRequired(OTP_ZONE_ID_MANAGEMENT)
2021-07-07 06:51:22 +00:00
# Generate our magic word manager...
self.magicWordManager = ToontownMagicWordManagerAI(self)
self.magicWordManager.generateWithRequired(OTP_ZONE_ID_MANAGEMENT)
2019-11-23 01:58:35 +00:00
def generateHood(self, hoodConstructor, zoneId):
# Bossbot HQ doesn't use DNA, so we skip over that.
if zoneId != ToontownGlobals.BossbotHQ:
self.dnaStoreMap[zoneId] = DNAStorage()
self.dnaDataMap[zoneId] = loadDNAFileAI(self.dnaStoreMap[zoneId], self.genDNAFileName(zoneId))
if zoneId in ToontownGlobals.HoodHierarchy:
for streetId in ToontownGlobals.HoodHierarchy[zoneId]:
self.dnaStoreMap[streetId] = DNAStorage()
self.dnaDataMap[streetId] = loadDNAFileAI(self.dnaStoreMap[streetId], self.genDNAFileName(streetId))
hood = hoodConstructor(self, zoneId)
hood.startup()
self.hoods.append(hood)
def createZones(self):
# First, generate our zone2NpcDict...
NPCToons.generateZone2NpcDict()
2019-11-23 04:51:24 +00:00
# Donald's Dock
self.zoneTable[ToontownGlobals.DonaldsDock] = (
(ToontownGlobals.DonaldsDock, 1, 0), (ToontownGlobals.BarnacleBoulevard, 1, 1),
(ToontownGlobals.SeaweedStreet, 1, 1), (ToontownGlobals.LighthouseLane, 1, 1)
)
self.generateHood(DDHoodDataAI, ToontownGlobals.DonaldsDock)
2019-11-23 01:58:35 +00:00
# Toontown Central
self.zoneTable[ToontownGlobals.ToontownCentral] = (
(ToontownGlobals.ToontownCentral, 1, 0), (ToontownGlobals.SillyStreet, 1, 1),
(ToontownGlobals.LoopyLane, 1, 1), (ToontownGlobals.PunchlinePlace, 1, 1)
)
self.generateHood(TTHoodDataAI, ToontownGlobals.ToontownCentral)
# The Brrrgh
self.zoneTable[ToontownGlobals.TheBrrrgh] = (
(ToontownGlobals.TheBrrrgh, 1, 0), (ToontownGlobals.WalrusWay, 1, 1),
(ToontownGlobals.SleetStreet, 1, 1), (ToontownGlobals.PolarPlace, 1, 1)
)
self.generateHood(BRHoodDataAI, ToontownGlobals.TheBrrrgh)
# Minnie's Melodyland
self.zoneTable[ToontownGlobals.MinniesMelodyland] = (
(ToontownGlobals.MinniesMelodyland, 1, 0), (ToontownGlobals.AltoAvenue, 1, 1),
(ToontownGlobals.BaritoneBoulevard, 1, 1), (ToontownGlobals.TenorTerrace, 1, 1)
)
self.generateHood(MMHoodDataAI, ToontownGlobals.MinniesMelodyland)
# Daisy Gardens
self.zoneTable[ToontownGlobals.DaisyGardens] = (
(ToontownGlobals.DaisyGardens, 1, 0), (ToontownGlobals.ElmStreet, 1, 1),
(ToontownGlobals.MapleStreet, 1, 1), (ToontownGlobals.OakStreet, 1, 1)
)
self.generateHood(DGHoodDataAI, ToontownGlobals.DaisyGardens)
2020-01-08 06:40:54 +00:00
# Chip 'n Dale's Acorn Acres
self.zoneTable[ToontownGlobals.OutdoorZone] = (
(ToontownGlobals.OutdoorZone, 1, 0),
)
self.generateHood(OZHoodDataAI, ToontownGlobals.OutdoorZone)
# Goofy Speedway
self.zoneTable[ToontownGlobals.GoofySpeedway] = (
(ToontownGlobals.GoofySpeedway, 1, 0),
)
self.generateHood(GSHoodDataAI, ToontownGlobals.GoofySpeedway)
# Donald's Dreamland
self.zoneTable[ToontownGlobals.DonaldsDreamland] = (
(ToontownGlobals.DonaldsDreamland, 1, 0), (ToontownGlobals.LullabyLane, 1, 1),
(ToontownGlobals.PajamaPlace, 1, 1)
)
self.generateHood(DLHoodDataAI, ToontownGlobals.DonaldsDreamland)
2020-01-11 14:50:33 +00:00
# Bossbot HQ
self.zoneTable[ToontownGlobals.BossbotHQ] = (
(ToontownGlobals.BossbotHQ, 0, 0),
)
self.generateHood(BossbotHQDataAI, ToontownGlobals.BossbotHQ)
2020-01-08 20:51:15 +00:00
# Sellbot HQ
self.zoneTable[ToontownGlobals.SellbotHQ] = (
(ToontownGlobals.SellbotHQ, 0, 1), (ToontownGlobals.SellbotFactoryExt, 0, 1)
)
self.generateHood(CSHoodDataAI, ToontownGlobals.SellbotHQ)
2020-01-10 02:42:01 +00:00
# Cashbot HQ
self.zoneTable[ToontownGlobals.CashbotHQ] = (
(ToontownGlobals.CashbotHQ, 0, 1),
)
self.generateHood(CashbotHQDataAI, ToontownGlobals.CashbotHQ)
2020-01-10 21:23:33 +00:00
# Lawbot HQ
self.zoneTable[ToontownGlobals.LawbotHQ] = (
(ToontownGlobals.LawbotHQ, 0, 1),
)
self.generateHood(LawbotHQDataAI, ToontownGlobals.LawbotHQ)
2020-01-08 02:12:08 +00:00
# Chip 'n Dale's MiniGolf
self.zoneTable[ToontownGlobals.GolfZone] = (
(ToontownGlobals.GolfZone, 1, 0),
)
self.generateHood(GZHoodDataAI, ToontownGlobals.GolfZone)
2019-12-06 02:38:58 +00:00
# Welcome Valley zones
self.welcomeValleyManager.createWelcomeValleyZones()
# Assign the initial suit buildings.
for suitPlanner in list(self.suitPlanners.values()):
suitPlanner.assignInitialSuitBuildings()
2019-11-23 01:58:35 +00:00
def genDNAFileName(self, zoneId):
canonicalZoneId = ZoneUtil.getCanonicalZoneId(zoneId)
canonicalHoodId = ZoneUtil.getCanonicalHoodId(canonicalZoneId)
hood = ToontownGlobals.dnaMap[canonicalHoodId]
if canonicalHoodId == canonicalZoneId:
canonicalZoneId = 'sz'
phase = ToontownGlobals.phaseMap[canonicalHoodId]
else:
phase = ToontownGlobals.streetPhaseMap[canonicalHoodId]
2020-01-08 02:16:42 +00:00
if 'outdoor_zone' in hood or 'golf_zone' in hood:
phase = '6'
2019-11-23 01:58:35 +00:00
return 'phase_%s/dna/%s_%s.dna' % (phase, hood, canonicalZoneId)
def lookupDNAFileName(self, dnaFileName):
searchPath = DSearchPath()
searchPath.appendDirectory(Filename('resources/phase_3.5/dna'))
searchPath.appendDirectory(Filename('resources/phase_4/dna'))
searchPath.appendDirectory(Filename('resources/phase_5/dna'))
searchPath.appendDirectory(Filename('resources/phase_5.5/dna'))
searchPath.appendDirectory(Filename('resources/phase_6/dna'))
searchPath.appendDirectory(Filename('resources/phase_8/dna'))
searchPath.appendDirectory(Filename('resources/phase_9/dna'))
searchPath.appendDirectory(Filename('resources/phase_10/dna'))
searchPath.appendDirectory(Filename('resources/phase_11/dna'))
searchPath.appendDirectory(Filename('resources/phase_12/dna'))
searchPath.appendDirectory(Filename('resources/phase_13/dna'))
filename = Filename(dnaFileName)
found = vfs.resolveFilename(filename, searchPath)
if not found:
self.notify.warning('lookupDNAFileName - %s not found on:' % dnaFileName)
print(searchPath)
else:
return filename.getFullpath()
2019-11-23 01:58:35 +00:00
def loadDNAFileAI(self, dnaStore, dnaFileName):
return loadDNAFileAI(dnaStore, dnaFileName)
def findFishingPonds(self, dnaData, zoneId, area):
return [], [] # TODO
def findPartyHats(self, dnaData, zoneId):
return [] # TODO
2019-12-06 03:42:29 +00:00
def findRacingPads(self, dnaData, zoneId, area, type='racing_pad', overrideDNAZone=False):
kartPads, kartPadGroups = [], []
if type in dnaData.getName():
if type == 'racing_pad':
2020-01-02 00:55:54 +00:00
nameSplit = dnaData.getName().split('_')
2019-12-06 03:42:29 +00:00
racePad = DistributedRacePadAI(self)
racePad.setArea(area)
2020-01-02 00:55:54 +00:00
racePad.index = int(nameSplit[2])
racePad.genre = nameSplit[3]
trackInfo = RaceGlobals.getNextRaceInfo(-1, racePad.genre, racePad.index)
racePad.setTrackInfo([trackInfo[0], trackInfo[1]])
racePad.laps = trackInfo[2]
2019-12-06 03:42:29 +00:00
racePad.generateWithRequired(zoneId)
kartPads.append(racePad)
kartPadGroups.append(dnaData)
elif type == 'viewing_pad':
viewPad = DistributedViewPadAI(self)
viewPad.setArea(area)
viewPad.generateWithRequired(zoneId)
kartPads.append(viewPad)
kartPadGroups.append(dnaData)
for i in range(dnaData.getNumChildren()):
2019-12-06 03:42:29 +00:00
foundKartPads, foundKartPadGroups = self.findRacingPads(dnaData.at(i), zoneId, area, type, overrideDNAZone)
kartPads.extend(foundKartPads)
kartPadGroups.extend(foundKartPadGroups)
return kartPads, kartPadGroups
2020-01-02 19:19:22 +00:00
def findStartingBlocks(self, dnaData, kartPad):
2020-01-02 17:46:49 +00:00
startingBlocks = []
for i in range(dnaData.getNumChildren()):
groupName = dnaData.getName()
block = dnaData.at(i)
blockName = block.getName()
if 'starting_block' in blockName:
cls = DistributedStartingBlockAI if 'racing_pad' in groupName else DistributedViewingBlockAI
x, y, z = block.getPos()
h, p, r = block.getHpr()
padLocationId = int(blockName[-1])
2020-01-02 19:19:22 +00:00
startingBlock = cls(self, kartPad, x, y, z, h, p, r, padLocationId)
startingBlock.generateWithRequired(kartPad.zoneId)
2020-01-02 17:46:49 +00:00
startingBlocks.append(startingBlock)
return startingBlocks
def findLeaderBoards(self, dnaData, zoneId):
2019-12-06 03:03:12 +00:00
leaderBoards = []
if 'leaderBoard' in dnaData.getName():
x, y, z = dnaData.getPos()
h, p, r = dnaData.getHpr()
leaderBoard = DistributedLeaderBoardAI(self, dnaData.getName(), x, y, z, h, p, r)
leaderBoard.generateWithRequired(zoneId)
leaderBoards.append(leaderBoard)
for i in range(dnaData.getNumChildren()):
2019-12-06 03:03:12 +00:00
foundLeaderBoards = self.findLeaderBoards(dnaData.at(i), zoneId)
leaderBoards.extend(foundLeaderBoards)
return leaderBoards
2019-11-17 21:29:23 +00:00
def getTrackClsends(self):
return False
def getAvatarExitEvent(self, avId):
return 'distObjDelete-%d' % avId
def getAvatarDisconnectReason(self, avId):
return self.timeManager.avId2disconnectcode.get(avId, ToontownGlobals.DisconnectUnknown)
2019-11-17 21:29:23 +00:00
def getZoneDataStore(self):
return self.zoneDataStore
def incrementPopulation(self):
2019-11-23 01:18:26 +00:00
self.districtStats.b_setAvatarCount(self.districtStats.getAvatarCount() + 1)
2019-11-17 21:29:23 +00:00
def decrementPopulation(self):
2019-11-23 01:18:26 +00:00
self.districtStats.b_setAvatarCount(self.districtStats.getAvatarCount() - 1)
2019-11-17 21:29:23 +00:00
def allocateZone(self, owner=None):
zoneId = self.zoneAllocator.allocate()
if owner:
self.zoneId2owner[zoneId] = owner
return zoneId
def deallocateZone(self, zone):
if self.zoneId2owner.get(zone):
del self.zoneId2owner[zone]
self.zoneAllocator.free(zone)
2020-01-10 22:20:26 +00:00
def trueUniqueName(self, idString):
return self.uniqueName(idString)
def setupFiles(self):
if not os.path.exists(self.dataFolder):
os.mkdir(self.dataFolder)