2023-04-24 04:12:43 +00:00
|
|
|
from panda3d.core import (
|
|
|
|
CompassEffect,
|
|
|
|
ModelPool,
|
|
|
|
NodePath,
|
|
|
|
TexturePool,
|
|
|
|
TransparencyAttrib,
|
|
|
|
Vec4
|
|
|
|
)
|
|
|
|
|
|
|
|
from direct.directnotify.DirectNotifyGlobal import directNotify
|
|
|
|
from direct.fsm.StateData import StateData
|
|
|
|
from direct.gui.OnscreenText import OnscreenText
|
|
|
|
from direct.interval.IntervalGlobal import Sequence, Wait, Func
|
|
|
|
from direct.showbase.MessengerGlobal import messenger
|
|
|
|
from direct.showbase.PythonUtil import uniqueName
|
|
|
|
from direct.task.TaskManagerGlobal import taskMgr
|
|
|
|
|
|
|
|
from toontown.hood import ZoneUtil
|
|
|
|
from toontown.hood.QuietZoneState import QuietZoneState
|
2019-11-02 22:27:54 +00:00
|
|
|
from toontown.toon.Toon import teleportDebug
|
2023-04-24 04:12:43 +00:00
|
|
|
from toontown.toonbase import ToontownGlobals
|
|
|
|
from toontown.toonbase import TTLocalizer
|
|
|
|
from toontown.toonbase.ToonBaseGlobal import base
|
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
|
2023-04-24 04:12:43 +00:00
|
|
|
class Hood(StateData):
|
|
|
|
notify = directNotify.newCategory('Hood')
|
2019-11-02 22:27:54 +00:00
|
|
|
|
|
|
|
def __init__(self, parentFSM, doneEvent, dnaStore, hoodId):
|
2023-04-24 04:12:43 +00:00
|
|
|
StateData.__init__(self, doneEvent)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.loader = 'not initialized'
|
|
|
|
self.parentFSM = parentFSM
|
|
|
|
self.dnaStore = dnaStore
|
|
|
|
self.loaderDoneEvent = 'loaderDone'
|
|
|
|
self.id = None
|
|
|
|
self.hoodId = hoodId
|
|
|
|
self.titleText = None
|
2019-11-17 21:29:23 +00:00
|
|
|
self.titleTextSeq = None
|
2019-11-02 22:27:54 +00:00
|
|
|
self.titleColor = (1, 1, 1, 1)
|
|
|
|
self.holidayStorageDNADict = {}
|
|
|
|
self.spookySkyFile = None
|
|
|
|
self.halloweenLights = []
|
|
|
|
|
|
|
|
def enter(self, requestStatus):
|
|
|
|
zoneId = requestStatus['zoneId']
|
|
|
|
hoodText = self.getHoodText(zoneId)
|
2023-04-24 04:12:43 +00:00
|
|
|
self.titleText = OnscreenText(hoodText, fg=self.titleColor, font=ToontownGlobals.getSignFont(), pos=(0, -0.5), scale=TTLocalizer.HtitleText, drawOrder=0, mayChange=1)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.fsm.request(requestStatus['loader'], [requestStatus])
|
|
|
|
|
|
|
|
def getHoodText(self, zoneId):
|
|
|
|
hoodText = base.cr.hoodMgr.getFullnameFromId(self.id)
|
2023-04-24 04:12:43 +00:00
|
|
|
if self.id != ToontownGlobals.Tutorial:
|
|
|
|
streetName = ToontownGlobals.StreetNames.get(ZoneUtil.getCanonicalBranchZone(zoneId))
|
2019-11-02 22:27:54 +00:00
|
|
|
if streetName:
|
|
|
|
hoodText = hoodText + '\n' + streetName[-1]
|
2023-04-24 04:12:43 +00:00
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
return hoodText
|
|
|
|
|
|
|
|
def spawnTitleText(self, zoneId):
|
|
|
|
hoodText = self.getHoodText(zoneId)
|
|
|
|
self.doSpawnTitleText(hoodText)
|
|
|
|
|
|
|
|
def doSpawnTitleText(self, text):
|
|
|
|
self.titleText.setText(text)
|
|
|
|
self.titleText.show()
|
|
|
|
self.titleText.setColor(Vec4(*self.titleColor))
|
|
|
|
self.titleText.clearColorScale()
|
|
|
|
self.titleText.setFg(self.titleColor)
|
2019-11-17 23:15:48 +00:00
|
|
|
self.titleTextSeq = Sequence(Wait(0.1), Wait(6.0), self.titleText.colorScaleInterval(0.5, Vec4(1.0, 1.0, 1.0, 0.0)), Func(self.hideTitleText))
|
2019-11-17 21:29:23 +00:00
|
|
|
self.titleTextSeq.start()
|
2019-11-02 22:27:54 +00:00
|
|
|
|
|
|
|
def hideTitleText(self):
|
|
|
|
if self.titleText:
|
|
|
|
self.titleText.hide()
|
|
|
|
|
|
|
|
def exit(self):
|
2019-11-17 21:29:23 +00:00
|
|
|
if self.titleTextSeq:
|
|
|
|
self.titleTextSeq.finish()
|
|
|
|
self.titleTextSeq = None
|
2023-04-24 04:12:43 +00:00
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
if self.titleText:
|
|
|
|
self.titleText.cleanup()
|
|
|
|
self.titleText = None
|
2023-04-24 04:12:43 +00:00
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
base.localAvatar.stopChat()
|
|
|
|
|
|
|
|
def load(self):
|
|
|
|
if self.storageDNAFile:
|
2023-04-24 04:12:43 +00:00
|
|
|
base.loader.loadDNAFile(self.dnaStore, self.storageDNAFile)
|
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
newsManager = base.cr.newsManager
|
|
|
|
if newsManager:
|
|
|
|
holidayIds = base.cr.newsManager.getDecorationHolidayId()
|
|
|
|
for holiday in holidayIds:
|
|
|
|
for storageFile in self.holidayStorageDNADict.get(holiday, []):
|
2023-04-24 04:12:43 +00:00
|
|
|
base.loader.loadDNAFile(self.dnaStore, storageFile)
|
2019-11-02 22:27:54 +00:00
|
|
|
|
|
|
|
if ToontownGlobals.HALLOWEEN_COSTUMES not in holidayIds and ToontownGlobals.SPOOKY_COSTUMES not in holidayIds or not self.spookySkyFile:
|
2023-04-24 04:12:43 +00:00
|
|
|
self.sky = base.loader.loadModel(self.skyFile)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.sky.setTag('sky', 'Regular')
|
|
|
|
self.sky.setScale(1.0)
|
|
|
|
self.sky.setFogOff()
|
|
|
|
else:
|
2023-04-24 04:12:43 +00:00
|
|
|
self.sky = base.loader.loadModel(self.spookySkyFile)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.sky.setTag('sky', 'Halloween')
|
2023-04-24 04:12:43 +00:00
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
if not newsManager:
|
2023-04-24 04:12:43 +00:00
|
|
|
self.sky = base.loader.loadModel(self.skyFile)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.sky.setTag('sky', 'Regular')
|
|
|
|
self.sky.setScale(1.0)
|
|
|
|
self.sky.setFogOff()
|
|
|
|
|
|
|
|
def unload(self):
|
|
|
|
if hasattr(self, 'loader'):
|
|
|
|
self.notify.info('Aggressively cleaning up loader: %s' % self.loader)
|
|
|
|
self.loader.exit()
|
|
|
|
self.loader.unload()
|
|
|
|
del self.loader
|
2023-04-24 04:12:43 +00:00
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
del self.fsm
|
|
|
|
del self.parentFSM
|
|
|
|
self.dnaStore.resetHood()
|
|
|
|
del self.dnaStore
|
|
|
|
self.sky.removeNode()
|
|
|
|
del self.sky
|
|
|
|
self.ignoreAll()
|
|
|
|
ModelPool.garbageCollect()
|
|
|
|
TexturePool.garbageCollect()
|
|
|
|
|
|
|
|
def enterStart(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def exitStart(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def isSameHood(self, status):
|
|
|
|
return status['hoodId'] == self.hoodId and status['shardId'] == None
|
|
|
|
|
|
|
|
def enterFinal(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def exitFinal(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def enterQuietZone(self, requestStatus):
|
|
|
|
teleportDebug(requestStatus, 'Hood.enterQuietZone: status=%s' % requestStatus)
|
|
|
|
self._quietZoneDoneEvent = uniqueName('quietZoneDone')
|
|
|
|
self.acceptOnce(self._quietZoneDoneEvent, self.handleQuietZoneDone)
|
2023-04-24 04:12:43 +00:00
|
|
|
self.quietZoneStateData = QuietZoneState(self._quietZoneDoneEvent)
|
2019-11-02 22:27:54 +00:00
|
|
|
self._enterWaitForSetZoneResponseMsg = self.quietZoneStateData.getEnterWaitForSetZoneResponseMsg()
|
|
|
|
self.acceptOnce(self._enterWaitForSetZoneResponseMsg, self.handleWaitForSetZoneResponse)
|
|
|
|
self._quietZoneLeftEvent = self.quietZoneStateData.getQuietZoneLeftEvent()
|
|
|
|
if base.placeBeforeObjects:
|
|
|
|
self.acceptOnce(self._quietZoneLeftEvent, self.handleLeftQuietZone)
|
2023-04-24 04:12:43 +00:00
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
self.quietZoneStateData.load()
|
|
|
|
self.quietZoneStateData.enter(requestStatus)
|
|
|
|
|
|
|
|
def exitQuietZone(self):
|
|
|
|
self.ignore(self._quietZoneDoneEvent)
|
|
|
|
self.ignore(self._quietZoneLeftEvent)
|
|
|
|
self.ignore(self._enterWaitForSetZoneResponseMsg)
|
|
|
|
del self._quietZoneDoneEvent
|
|
|
|
self.quietZoneStateData.exit()
|
|
|
|
self.quietZoneStateData.unload()
|
|
|
|
self.quietZoneStateData = None
|
|
|
|
|
|
|
|
def loadLoader(self, requestStatus):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def handleWaitForSetZoneResponse(self, requestStatus):
|
|
|
|
loaderName = requestStatus['loader']
|
|
|
|
if loaderName == 'safeZoneLoader':
|
2023-04-24 04:12:43 +00:00
|
|
|
if not base.loader.inBulkBlock:
|
|
|
|
base.loader.beginBulkLoad('hood', TTLocalizer.HeadingToPlayground, ToontownGlobals.safeZoneCountMap[self.id], 1, TTLocalizer.TIP_GENERAL)
|
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
self.loadLoader(requestStatus)
|
2023-04-24 04:12:43 +00:00
|
|
|
base.loader.endBulkLoad('hood')
|
2019-11-02 22:27:54 +00:00
|
|
|
elif loaderName == 'townLoader':
|
2023-04-24 04:12:43 +00:00
|
|
|
if not base.loader.inBulkBlock:
|
2019-11-02 22:27:54 +00:00
|
|
|
zoneId = requestStatus['zoneId']
|
2023-04-24 04:12:43 +00:00
|
|
|
toPhrase = ToontownGlobals.StreetNames[ZoneUtil.getCanonicalBranchZone(zoneId)][0]
|
|
|
|
streetName = ToontownGlobals.StreetNames[ZoneUtil.getCanonicalBranchZone(zoneId)][-1]
|
|
|
|
base.loader.beginBulkLoad('hood', TTLocalizer.HeadingToStreet % {'to': toPhrase,
|
|
|
|
'street': streetName}, ToontownGlobals.townCountMap[self.id], 1, TTLocalizer.TIP_STREET)
|
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
self.loadLoader(requestStatus)
|
2023-04-24 04:12:43 +00:00
|
|
|
base.loader.endBulkLoad('hood')
|
2019-11-02 22:27:54 +00:00
|
|
|
elif loaderName == 'minigame':
|
|
|
|
pass
|
|
|
|
elif loaderName == 'cogHQLoader':
|
2019-12-30 06:07:56 +00:00
|
|
|
print('should be loading HQ')
|
2019-11-02 22:27:54 +00:00
|
|
|
|
|
|
|
def handleLeftQuietZone(self):
|
|
|
|
status = self.quietZoneStateData.getRequestStatus()
|
|
|
|
teleportDebug(status, 'handleLeftQuietZone, status=%s' % status)
|
|
|
|
teleportDebug(status, 'requesting %s' % status['loader'])
|
|
|
|
self.fsm.request(status['loader'], [status])
|
|
|
|
|
|
|
|
def handleQuietZoneDone(self):
|
|
|
|
if not base.placeBeforeObjects:
|
|
|
|
status = self.quietZoneStateData.getRequestStatus()
|
|
|
|
self.fsm.request(status['loader'], [status])
|
|
|
|
|
|
|
|
def enterSafeZoneLoader(self, requestStatus):
|
|
|
|
self.accept(self.loaderDoneEvent, self.handleSafeZoneLoaderDone)
|
|
|
|
self.loader.enter(requestStatus)
|
|
|
|
self.spawnTitleText(requestStatus['zoneId'])
|
|
|
|
|
|
|
|
def exitSafeZoneLoader(self):
|
2019-11-17 21:29:23 +00:00
|
|
|
if self.titleTextSeq:
|
|
|
|
self.titleTextSeq.finish()
|
|
|
|
self.titleTextSeq = None
|
2023-04-24 04:12:43 +00:00
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
self.hideTitleText()
|
|
|
|
self.ignore(self.loaderDoneEvent)
|
|
|
|
self.loader.exit()
|
|
|
|
self.loader.unload()
|
|
|
|
del self.loader
|
|
|
|
|
|
|
|
def handleSafeZoneLoaderDone(self):
|
|
|
|
doneStatus = self.loader.getDoneStatus()
|
|
|
|
teleportDebug(doneStatus, 'handleSafeZoneLoaderDone, doneStatus=%s' % doneStatus)
|
|
|
|
if self.isSameHood(doneStatus) and doneStatus['where'] != 'party' or doneStatus['loader'] == 'minigame':
|
|
|
|
teleportDebug(doneStatus, 'same hood')
|
|
|
|
self.fsm.request('quietZone', [doneStatus])
|
|
|
|
else:
|
|
|
|
teleportDebug(doneStatus, 'different hood')
|
|
|
|
self.doneStatus = doneStatus
|
|
|
|
messenger.send(self.doneEvent)
|
|
|
|
|
|
|
|
def startSky(self):
|
2023-04-24 04:12:43 +00:00
|
|
|
self.sky.reparentTo(base.camera)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.sky.setZ(0.0)
|
|
|
|
self.sky.setHpr(0.0, 0.0, 0.0)
|
|
|
|
ce = CompassEffect.make(NodePath(), CompassEffect.PRot | CompassEffect.PZ)
|
|
|
|
self.sky.node().setEffect(ce)
|
|
|
|
|
|
|
|
def stopSky(self):
|
|
|
|
taskMgr.remove('skyTrack')
|
2023-04-24 04:12:43 +00:00
|
|
|
self.sky.reparentTo(base.hidden)
|
2019-11-02 22:27:54 +00:00
|
|
|
|
|
|
|
def startSpookySky(self):
|
|
|
|
if not self.spookySkyFile:
|
|
|
|
return
|
2023-04-24 04:12:43 +00:00
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
if hasattr(self, 'sky') and self.sky:
|
|
|
|
self.stopSky()
|
2023-04-24 04:12:43 +00:00
|
|
|
|
|
|
|
self.sky = base.loader.loadModel(self.spookySkyFile)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.sky.setTag('sky', 'Halloween')
|
|
|
|
self.sky.setColor(0.5, 0.5, 0.5, 1)
|
2023-04-24 04:12:43 +00:00
|
|
|
self.sky.reparentTo(base.camera)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.sky.setTransparency(TransparencyAttrib.MDual, 1)
|
|
|
|
fadeIn = self.sky.colorScaleInterval(1.5, Vec4(1, 1, 1, 1), startColorScale=Vec4(1, 1, 1, 0.25), blendType='easeInOut')
|
|
|
|
fadeIn.start()
|
|
|
|
self.sky.setZ(0.0)
|
|
|
|
self.sky.setHpr(0.0, 0.0, 0.0)
|
|
|
|
ce = CompassEffect.make(NodePath(), CompassEffect.PRot | CompassEffect.PZ)
|
|
|
|
self.sky.node().setEffect(ce)
|
|
|
|
|
|
|
|
def endSpookySky(self):
|
|
|
|
if hasattr(self, 'sky') and self.sky:
|
2023-04-24 04:12:43 +00:00
|
|
|
self.sky.reparentTo(base.hidden)
|
|
|
|
|
2019-11-02 22:27:54 +00:00
|
|
|
if hasattr(self, 'sky'):
|
2023-04-24 04:12:43 +00:00
|
|
|
self.sky = base.loader.loadModel(self.skyFile)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.sky.setTag('sky', 'Regular')
|
|
|
|
self.sky.setScale(1.0)
|
|
|
|
self.startSky()
|