2019-11-02 22:27:54 +00:00
|
|
|
from pandac.PandaModules import *
|
|
|
|
from direct.distributed.ClockDelta import *
|
|
|
|
from direct.interval.IntervalGlobal import *
|
|
|
|
from direct.gui.DirectGui import *
|
|
|
|
from pandac.PandaModules import *
|
|
|
|
from direct.fsm import FSM
|
|
|
|
from direct.distributed import DistributedSmoothNode
|
|
|
|
from direct.interval.IntervalGlobal import *
|
|
|
|
from direct.showbase.PythonUtil import clampScalar
|
|
|
|
from otp.otpbase import OTPGlobals
|
|
|
|
from otp.avatar import ShadowCaster
|
|
|
|
from toontown.racing import Kart
|
|
|
|
from toontown.racing.KartDNA import *
|
|
|
|
from toontown.toonbase import ToontownGlobals
|
|
|
|
from toontown.toonbase import TTLocalizer
|
|
|
|
from toontown.effects.Drift import Drift
|
|
|
|
from toontown.effects.Sparks import Sparks
|
|
|
|
from direct.interval.ProjectileInterval import *
|
|
|
|
from toontown.battle.BattleProps import *
|
|
|
|
import random
|
|
|
|
from direct.showbase.PythonUtil import randFloat
|
|
|
|
from direct.task.Task import Task
|
|
|
|
import math
|
|
|
|
iceTurnFactor = 0.25
|
|
|
|
iceAccelFactor = 0.4
|
|
|
|
|
|
|
|
class DistributedVehicle(DistributedSmoothNode.DistributedSmoothNode, Kart.Kart, FSM.FSM):
|
|
|
|
notify = DirectNotifyGlobal.directNotify.newCategory('DistributedVehicle')
|
|
|
|
cheatFactor = 1.0
|
|
|
|
proRacer = 0
|
|
|
|
physicsCalculationsPerSecond = 60
|
|
|
|
maxPhysicsDt = 1.0
|
|
|
|
physicsDt = 1.0 / float(physicsCalculationsPerSecond)
|
|
|
|
maxPhysicsFrames = maxPhysicsDt * physicsCalculationsPerSecond
|
|
|
|
maxSpeed = 200 * cheatFactor
|
|
|
|
turnRatio = 1.0 / 0.025 * math.sqrt(cheatFactor)
|
|
|
|
accelerationMult = 35
|
|
|
|
accelerationBase = 20
|
|
|
|
if proRacer:
|
|
|
|
accelerationMult = 35
|
|
|
|
accelerationBase = 30
|
|
|
|
surfaceModifiers = {'asphalt': {'shake': 0.1,
|
|
|
|
'driftMin': 65,
|
|
|
|
'windResistance': 0.2,
|
|
|
|
'particleColor': Vec4(0.7, 0.7, 0.7, 1.0)},
|
|
|
|
'gravel': {'shake': 0.2,
|
|
|
|
'driftMin': 65,
|
|
|
|
'windResistance': 0.2,
|
|
|
|
'particleColor': Vec4(0.53, 0.53, 0.53, 1.0)},
|
|
|
|
'dirt': {'shake': 0.4,
|
|
|
|
'driftMin': 35,
|
|
|
|
'windResistance': 0.3,
|
|
|
|
'particleColor': Vec4(1.0, 1.0, 1.0, 1.0)},
|
|
|
|
'grass': {'shake': 0.8,
|
|
|
|
'driftMin': 15,
|
|
|
|
'windResistance': 0.4,
|
|
|
|
'particleColor': Vec4(0.8, 0.42, 0.8, 1.0)},
|
|
|
|
'ice': {'shake': 0.0,
|
|
|
|
'driftMin': 0,
|
|
|
|
'windResistance': 0.01,
|
|
|
|
'particleColor': Vec4(1.0, 1.0, 1.0, 1.0)},
|
|
|
|
'': {'shake': 0,
|
|
|
|
'driftMin': 1,
|
|
|
|
'windResistance': 0.2,
|
|
|
|
'particleColor': Vec4(1.0, 1.0, 1.0, 1.0)}}
|
|
|
|
SFX_BaseDir = 'phase_6/audio/sfx/'
|
|
|
|
SFX_WallHits = [SFX_BaseDir + 'KART_Hitting_Wood_Fence.mp3',
|
|
|
|
SFX_BaseDir + 'KART_Hitting_Wood_Fence_1.mp3',
|
|
|
|
SFX_BaseDir + 'KART_Hitting_Metal_Fence.mp3',
|
|
|
|
SFX_BaseDir + 'KART_Hitting_Wall.mp3']
|
|
|
|
SFX_SkidLoop_Grass = SFX_BaseDir + 'KART_Skidding_On_Grass.wav'
|
|
|
|
SFX_SkidLoop_Asphalt = SFX_BaseDir + 'KART_Skidding_On_Asphalt.wav'
|
|
|
|
SFX_TurboStart = SFX_BaseDir + 'KART_turboStart.mp3'
|
|
|
|
SFX_TurboLoop = SFX_BaseDir + 'KART_turboLoop.wav'
|
|
|
|
AvId2kart = {}
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def getKartFromAvId(cls, avId):
|
|
|
|
return cls.AvId2kart.get(avId)
|
|
|
|
|
|
|
|
def __init__(self, cr):
|
|
|
|
DistributedSmoothNode.DistributedSmoothNode.__init__(self, cr)
|
|
|
|
FSM.FSM.__init__(self, 'DistributedVehicle')
|
|
|
|
Kart.Kart.__init__(self)
|
|
|
|
if base.config.GetBool('want-racer', 0) == 1:
|
|
|
|
DistributedVehicle.proRacer = 1
|
|
|
|
DistributedVehicle.accelerationMult = 35
|
|
|
|
DistributedVehicle.accelerationBase = 30
|
|
|
|
self.speedometer = None
|
|
|
|
self.speedGauge = None
|
|
|
|
self.leanAmount = 0
|
|
|
|
self.leftHeld = 0
|
|
|
|
self.stopped = False
|
|
|
|
self.rightHeld = 0
|
|
|
|
self.canRace = False
|
|
|
|
self.groundType = 'gravel'
|
|
|
|
self.offGround = 0
|
|
|
|
self.turbo = False
|
|
|
|
self.ownerId = 0
|
|
|
|
self.cameraTrack = None
|
|
|
|
self.curSpeed = 0
|
|
|
|
self.acceleration = 0
|
|
|
|
self.imHitMult = 1
|
|
|
|
self.skidding = False
|
|
|
|
self.hittingWall = 0
|
|
|
|
base.kart = self
|
|
|
|
self.wipeOut = None
|
|
|
|
self.gagMovie = None
|
|
|
|
self.spinAnim = None
|
|
|
|
self.wrongWay = False
|
|
|
|
self.wallCollideTrack = None
|
|
|
|
self.wheelMaxTurn = 1.0
|
|
|
|
self.wheelMinTurn = 0.15
|
|
|
|
self.speedMaxTurn = 90
|
|
|
|
self.speedMinTurn = 300
|
|
|
|
self.wheelTurnTime = 0.6
|
|
|
|
self.wheelReturnTime = 0.3
|
|
|
|
self.wheelFightTime = 0.2
|
|
|
|
self.wheelPosition = 0.0
|
|
|
|
self.outPutCounter = 0
|
|
|
|
self.proCameraHeading = 0
|
|
|
|
self.proCameraDummyNode = None
|
|
|
|
self.cameraArmNode = None
|
|
|
|
self.armSwingSpeedPerp = 0.2
|
|
|
|
self.armSwingSpeedPara = 0.7
|
|
|
|
self.cameraTightener = 3.0
|
|
|
|
self.cameraArmBase = 0
|
|
|
|
self.cameraArmExtend = 20
|
|
|
|
self.pieCount = 0
|
|
|
|
self.numPieChunks = 6
|
|
|
|
self.pieSlideSpeed = []
|
|
|
|
for piece in range(self.numPieChunks):
|
|
|
|
self.pieSlideSpeed.append(randFloat(0.0, 0.2))
|
|
|
|
|
|
|
|
self.wantSmoke = ConfigVariableBool('want-kart-smoke', 1).getValue()
|
|
|
|
self.wantSparks = ConfigVariableBool('want-kart-sparks', 1).getValue()
|
|
|
|
self.__loadTextures()
|
|
|
|
return
|
|
|
|
|
|
|
|
def __loadTextures(self):
|
|
|
|
self.pieSplatter = loader.loadModel('phase_6/models/karting/pie_splat_1.bam')
|
|
|
|
|
|
|
|
def announceGenerate(self):
|
|
|
|
DistributedSmoothNode.DistributedSmoothNode.announceGenerate(self)
|
|
|
|
self.generateKart()
|
|
|
|
self.name = self.uniqueName('vehicle')
|
|
|
|
self.posHprBroadcastName = self.uniqueName('vehicleBroadcast')
|
|
|
|
self.LODnode.forceSwitch(self.LODnode.getHighestSwitch())
|
|
|
|
self.cameraNode = NodePath('cameraNode')
|
|
|
|
self.cameraNode.reparentTo(self)
|
|
|
|
if self.proRacer:
|
|
|
|
self.cameraArmNode = render.attachNewNode('cameraArm')
|
|
|
|
self.cameraArmNode.reparentTo(self)
|
|
|
|
self.cameraNode.reparentTo(self.cameraArmNode)
|
|
|
|
self.proCameraDummyNode = render.attachNewNode('proCameraDummy')
|
|
|
|
self.proCameraDummyNode.reparentTo(render)
|
|
|
|
self.localVehicle = self.ownerId == localAvatar.doId
|
|
|
|
if self.localVehicle:
|
|
|
|
self.setupPhysics()
|
|
|
|
self.setupParticles()
|
|
|
|
self.__enableCollisions()
|
|
|
|
self.forward = NodePath('forward')
|
|
|
|
self.forward.setPos(0, 1, 0)
|
|
|
|
self.wallHitsSfx = []
|
|
|
|
for wallHit in self.SFX_WallHits:
|
2019-11-23 22:07:27 +00:00
|
|
|
self.wallHitsSfx.append(base.loader.loadSfx(wallHit))
|
2019-11-02 22:27:54 +00:00
|
|
|
|
2019-11-23 22:07:27 +00:00
|
|
|
self.skidLoopAsphaltSfx = base.loader.loadSfx(self.SFX_SkidLoop_Asphalt)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.skidLoopAsphaltSfx.setLoop()
|
2019-11-23 22:07:27 +00:00
|
|
|
self.skidLoopGrassSfx = base.loader.loadSfx(self.SFX_SkidLoop_Grass)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.skidLoopGrassSfx.setLoop()
|
2019-11-23 22:07:27 +00:00
|
|
|
self.turboStartSfx = base.loader.loadSfx(self.SFX_TurboStart)
|
|
|
|
self.turboLoopSfx = base.loader.loadSfx(self.SFX_TurboLoop)
|
2019-11-02 22:27:54 +00:00
|
|
|
self.turboLoopSfx.setLoop()
|
|
|
|
self.forward.reparentTo(self.geom[0])
|
|
|
|
self.anvil = globalPropPool.getProp('anvil')
|
|
|
|
self.anvil.setScale(5)
|
|
|
|
self.anvil.reparentTo(hidden)
|
|
|
|
self.anvil.setColor(1, 1, 1, 1)
|
|
|
|
self.anvil.setTransparency(1)
|
|
|
|
self.reparentTo(render)
|
|
|
|
|
|
|
|
def setupPhysics(self):
|
|
|
|
self.__setupCollisions()
|
|
|
|
self.physicsMgr = PhysicsManager()
|
|
|
|
self.physicsEpoch = globalClock.getFrameTime()
|
|
|
|
self.lastPhysicsFrame = 0
|
|
|
|
integrator = LinearEulerIntegrator()
|
|
|
|
self.physicsMgr.attachLinearIntegrator(integrator)
|
|
|
|
fn = ForceNode('windResistance')
|
|
|
|
fnp = NodePath(fn)
|
|
|
|
fnp.reparentTo(render)
|
|
|
|
windResistance = LinearFrictionForce(0.2)
|
|
|
|
fn.addForce(windResistance)
|
|
|
|
self.physicsMgr.addLinearForce(windResistance)
|
|
|
|
self.windResistance = windResistance
|
|
|
|
fn = ForceNode('engine')
|
|
|
|
fnp = NodePath(fn)
|
|
|
|
fnp.reparentTo(self)
|
|
|
|
engine = LinearVectorForce(0, 0, 0)
|
|
|
|
fn.addForce(engine)
|
|
|
|
self.physicsMgr.addLinearForce(engine)
|
|
|
|
self.engine = engine
|
|
|
|
|
|
|
|
def disable(self):
|
|
|
|
DistributedVehicle.AvId2kart.pop(self.ownerId)
|
|
|
|
self.finishMovies()
|
|
|
|
self.request('Off')
|
|
|
|
self.stopSkid()
|
|
|
|
self.stopSmooth()
|
|
|
|
if self.localVehicle:
|
|
|
|
self.__disableCollisions()
|
|
|
|
self.__undoCollisions()
|
|
|
|
self.physicsMgr.clearLinearForces()
|
|
|
|
self.detachNode()
|
|
|
|
DistributedSmoothNode.DistributedSmoothNode.disable(self)
|
|
|
|
taskMgr.remove('slidePies')
|
|
|
|
|
|
|
|
def delete(self):
|
|
|
|
self.stopSmooth()
|
|
|
|
Kart.Kart.delete(self)
|
|
|
|
DistributedSmoothNode.DistributedSmoothNode.delete(self)
|
|
|
|
if self.localVehicle:
|
|
|
|
self.ignoreAll()
|
|
|
|
if hasattr(self, 'piePieces'):
|
|
|
|
for piece in self.piePieces:
|
|
|
|
del piece
|
|
|
|
|
|
|
|
def getVelocity(self):
|
|
|
|
return self.actorNode.getPhysicsObject().getVelocity()
|
|
|
|
|
|
|
|
def __updateWallCollision(self, entry = None):
|
|
|
|
vol = self.curSpeed / 160
|
|
|
|
for wallHit in self.wallHitsSfx:
|
|
|
|
wallHit.setVolume(vol)
|
|
|
|
|
|
|
|
def __wallCollisionStart(self, entry):
|
|
|
|
self.entry = entry
|
|
|
|
self.__wallCollisionStop()
|
|
|
|
self.hittingWall = 1
|
|
|
|
hitToPlay = random.randrange(0, len(self.wallHitsSfx))
|
|
|
|
self.wallCollideTrack = Parallel(Func(self.wallHitsSfx[hitToPlay].play))
|
|
|
|
self.__updateWallCollision()
|
|
|
|
self.wallCollideTrack.start()
|
|
|
|
curSpeed = self.actorNode.getPhysicsObject().getVelocity().length()
|
|
|
|
if self.wantSparks and curSpeed > 10:
|
|
|
|
if entry.getSurfaceNormal(self)[0] < 0:
|
|
|
|
self.fireSparkParticles('right')
|
|
|
|
else:
|
|
|
|
self.fireSparkParticles('left')
|
|
|
|
|
|
|
|
def __wallCollisionStop(self, entry = None):
|
|
|
|
self.hittingWall = 0
|
|
|
|
if self.wallCollideTrack:
|
|
|
|
self.wallCollideTrack.pause()
|
|
|
|
|
|
|
|
def __setupCollisions(self):
|
|
|
|
self.cWallTrav = CollisionTraverser('KartWall')
|
|
|
|
self.cWallTrav.setRespectPrevTransform(True)
|
|
|
|
self.collisionNode = CollisionNode(self.uniqueName('vehicle'))
|
|
|
|
self.collisionNode.setFromCollideMask(OTPGlobals.WallBitmask)
|
|
|
|
self.collisionNode.setIntoCollideMask(ToontownGlobals.PieBitmask)
|
|
|
|
cs = CollisionSphere(0, 0, 4, 4)
|
|
|
|
self.collisionNode.addSolid(cs)
|
|
|
|
self.collisionNodePath = NodePath(self.collisionNode)
|
|
|
|
self.wallHandler = PhysicsCollisionHandler()
|
|
|
|
self.wallHandler.setStaticFrictionCoef(0.0)
|
|
|
|
self.wallHandler.setDynamicFrictionCoef(0.1)
|
|
|
|
self.wallHandler.addCollider(self.collisionNodePath, self)
|
|
|
|
self.wallHandler.setInPattern('enterWallCollision')
|
|
|
|
self.wallHandler.setOutPattern('exitWallCollision')
|
|
|
|
self.cWallTrav.addCollider(self.collisionNodePath, self.wallHandler)
|
|
|
|
self.accept('enterWallCollision', self.__wallCollisionStart)
|
|
|
|
self.accept('exitWallCollision', self.__wallCollisionStop)
|
|
|
|
cRay = CollisionRay(0.0, 0.0, 40000.0, 0.0, 0.0, -1.0)
|
|
|
|
cRayNode = CollisionNode(self.uniqueName('vehicle-FloorRay'))
|
|
|
|
cRayNode.addSolid(cRay)
|
|
|
|
cRayNode.setFromCollideMask(OTPGlobals.FloorBitmask)
|
|
|
|
cRayNode.setIntoCollideMask(BitMask32.allOff())
|
|
|
|
self.cRayNodePath = self.attachNewNode(cRayNode)
|
|
|
|
self.lifter = CollisionHandlerGravity()
|
|
|
|
self.lifter.setGravity(32.174 * 3.0)
|
|
|
|
self.lifter.addInPattern('floorCollision')
|
|
|
|
self.lifter.addAgainPattern('floorCollision')
|
|
|
|
self.lifter.setOffset(OTPGlobals.FloorOffset)
|
|
|
|
self.lifter.setReach(40.0)
|
|
|
|
self.lifter.addCollider(self.cRayNodePath, self)
|
|
|
|
base.cTrav.addCollider(self.cRayNodePath, self.lifter)
|
|
|
|
self.accept('floorCollision', self.__groundCollision)
|
|
|
|
self.accept('imIn-banana', self.hitBanana)
|
|
|
|
base.localAvatar.collisionsOff()
|
|
|
|
|
|
|
|
def __undoCollisions(self):
|
|
|
|
base.cTrav.removeCollider(self.cRayNodePath)
|
|
|
|
base.localAvatar.collisionsOn()
|
|
|
|
|
|
|
|
def __groundCollision(self, entry):
|
|
|
|
ground = entry.getIntoNodePath()
|
|
|
|
if ground.getNetTag('surface') == '':
|
|
|
|
return
|
|
|
|
self.groundType = ground.getNetTag('surface')
|
|
|
|
|
|
|
|
def __enableCollisions(self):
|
|
|
|
self.cQueue = []
|
|
|
|
self.cRays = NodePath('stickVehicleToFloor')
|
|
|
|
self.cRays.reparentTo(self.geom[0])
|
|
|
|
for wheelNode in self.wheelBases:
|
|
|
|
rayNode = CollisionNode(wheelNode.getName())
|
|
|
|
x = wheelNode.getX()
|
|
|
|
y = wheelNode.getY()
|
|
|
|
ray = CollisionRay(x, y, 40000.0, 0.0, 0.0, -1.0)
|
|
|
|
rayNode.addSolid(ray)
|
|
|
|
rayNode.setFromCollideMask(OTPGlobals.FloorBitmask)
|
|
|
|
rayNode.setIntoCollideMask(BitMask32.allOff())
|
|
|
|
rayNodePath = self.cRays.attachNewNode(rayNode)
|
|
|
|
cQueue = CollisionHandlerQueue()
|
|
|
|
self.cWallTrav.addCollider(rayNodePath, cQueue)
|
|
|
|
self.cQueue.append(cQueue)
|
|
|
|
|
|
|
|
self.collisionNodePath.reparentTo(self)
|
|
|
|
|
|
|
|
def setupLapCollisions(self):
|
|
|
|
self.lapBit = BitMask32(32768)
|
|
|
|
self.lapHandler = CollisionHandlerEvent()
|
|
|
|
self.lapHandler.addInPattern('imIn-%in')
|
|
|
|
self.cSphere = CollisionSphere(0, 0, 0, 3)
|
|
|
|
self.lapNode = CollisionNode('lapChecker')
|
|
|
|
self.lapNode.addSolid(self.cSphere)
|
|
|
|
self.lapNode.setIntoCollideMask(BitMask32.allOff())
|
|
|
|
self.lapNode.setFromCollideMask(self.lapBit)
|
|
|
|
self.lapNodePath = NodePath(self.lapNode)
|
|
|
|
self.lapNodePath.reparentTo(self)
|
|
|
|
self.cWallTrav.addCollider(self.lapNodePath, self.lapHandler)
|
|
|
|
|
|
|
|
def __disableCollisions(self):
|
|
|
|
self.ignore('imIn-startLine')
|
|
|
|
self.ignore('imIn-quarterLine')
|
|
|
|
self.ignore('imIn-midLine')
|
|
|
|
|
|
|
|
def __handleCollisionSphereEnter(self, collEntry = None):
|
|
|
|
self.d_requestControl()
|
|
|
|
|
|
|
|
def setupParticles(self):
|
|
|
|
if self.wantSmoke:
|
|
|
|
self.setupDriftParticles()
|
|
|
|
if self.wantSparks:
|
|
|
|
self.setupSparkParticles()
|
|
|
|
|
|
|
|
def updateParticles(self, leanAmount):
|
|
|
|
if self.wantSmoke:
|
|
|
|
self.updateDriftParticles(leanAmount)
|
|
|
|
|
|
|
|
def cleanupParticles(self):
|
|
|
|
if self.wantSmoke:
|
|
|
|
self.cleanupDriftParticles()
|
|
|
|
if self.wantSparks:
|
|
|
|
self.cleanupSparkParticles()
|
|
|
|
|
|
|
|
def setupDriftParticles(self):
|
|
|
|
smokeMount = self.geom[0].attachNewNode('Smoke Effect')
|
|
|
|
backLeft = smokeMount.attachNewNode('Back Left Smokemount')
|
|
|
|
backLeft.setPos(self.geom[0].find('**/' + self.wheelData[self.LRWHEEL]['node']).getPos() + Vec3(-0.25, -1.0, -0.5))
|
|
|
|
backRight = smokeMount.attachNewNode('Back Right Smokemount')
|
|
|
|
backRight.setPos(self.geom[0].find('**/' + self.wheelData[self.RRWHEEL]['node']).getPos() + Vec3(0.25, -1.0, -0.5))
|
|
|
|
driftEffects = (Drift(backLeft, backLeft), Drift(backRight, backRight))
|
|
|
|
for x in driftEffects:
|
|
|
|
x.start()
|
|
|
|
|
|
|
|
self.drifts = driftEffects
|
|
|
|
self.driftSeq = Sequence(Func(backLeft.show), Func(backRight.show), Wait(1000000), Func(self.drifts[0].effect.getParticlesNamed('particles-1').getRenderer().setColor, Vec4(1.0, 1.0, 1.0, 1.0)), Func(self.drifts[1].effect.getParticlesNamed('particles-1').getRenderer().setColor, Vec4(1.0, 1.0, 1.0, 1.0)), Func(backLeft.hide), Func(backRight.hide))
|
|
|
|
self.driftSeqStarted = False
|
|
|
|
self.driftParticleForces = [ x.effect.getParticlesNamed('particles-1').getLinearForce(0) for x in self.drifts ]
|
|
|
|
self.smokeMount = smokeMount
|
|
|
|
backLeft.hide()
|
|
|
|
backRight.hide()
|
|
|
|
|
|
|
|
def updateDriftParticles(self, leanAmount):
|
|
|
|
for x in self.driftParticleForces:
|
|
|
|
x.setAmplitude(leanAmount * 30)
|
|
|
|
|
|
|
|
if self.skidding and not self.driftSeqStarted:
|
|
|
|
self.driftSeqStarted = True
|
|
|
|
self.driftSeq.start()
|
|
|
|
elif not self.skidding and self.driftSeqStarted:
|
|
|
|
self.driftSeqStarted = False
|
|
|
|
self.driftSeq.finish()
|
|
|
|
|
|
|
|
def cleanupDriftParticles(self):
|
|
|
|
self.driftSeq.finish()
|
|
|
|
for x in self.drifts:
|
|
|
|
x.destroy()
|
|
|
|
|
2019-11-23 22:51:14 +00:00
|
|
|
self.smokeMount.removeNode()
|
2019-11-02 22:27:54 +00:00
|
|
|
del self.driftSeq
|
|
|
|
del self.driftParticleForces
|
|
|
|
del self.drifts
|
|
|
|
del self.smokeMount
|
|
|
|
|
|
|
|
def setupSparkParticles(self):
|
|
|
|
bodyType = self.kartDNA[KartDNA.bodyType]
|
|
|
|
endPts = KartDict[bodyType][7]
|
|
|
|
self.sparkMount = self.geom[0].attachNewNode('Spark Effect')
|
|
|
|
left = self.sparkMount.attachNewNode('Left Sparkmount')
|
|
|
|
left.setPos((self.geom[0].find('**/' + self.wheelData[self.LFWHEEL]['node']).getPos() + self.geom[0].find('**/' + self.wheelData[self.LRWHEEL]['node']).getPos()) / 2.0 + Vec3(0.0, -1.0, 1.0))
|
|
|
|
left.setScale(0.5)
|
|
|
|
right = self.sparkMount.attachNewNode('Right Sparkmount')
|
|
|
|
right.setPos((self.geom[0].find('**/' + self.wheelData[self.RFWHEEL]['node']).getPos() + self.geom[0].find('**/' + self.wheelData[self.RRWHEEL]['node']).getPos()) / 2.0 + Vec3(0.0, -1.0, 1.0))
|
|
|
|
right.setScale(0.5)
|
|
|
|
self.sparks = (Sparks(right, self.sparkMount), Sparks(left, self.sparkMount))
|
|
|
|
self.sparks[0].effect.getParticlesNamed('particles-1').getEmitter().setEndpoint1(endPts[0])
|
|
|
|
self.sparks[0].effect.getParticlesNamed('particles-1').getEmitter().setEndpoint2(endPts[1])
|
|
|
|
self.sparks[0].effect.getParticlesNamed('particles-1').getLinearForce(0).setAmplitude(-50)
|
|
|
|
self.sparks[1].effect.getParticlesNamed('particles-1').getEmitter().setEndpoint1(Point3(-endPts[0][0], endPts[0][1], endPts[0][2]))
|
|
|
|
self.sparks[1].effect.getParticlesNamed('particles-1').getEmitter().setEndpoint2(Point3(-endPts[1][0], endPts[1][1], endPts[1][2]))
|
|
|
|
self.sparks[1].effect.getParticlesNamed('particles-1').getLinearForce(0).setAmplitude(50)
|
|
|
|
|
|
|
|
def fireSparkParticles(self, side):
|
|
|
|
spark = self.sparks[{'right': 0,
|
|
|
|
'left': 1}[side]]
|
|
|
|
if not spark.effect.isEnabled():
|
|
|
|
spark.effect.getParticlesNamed('particles-1').setBirthRate(0.02)
|
|
|
|
spark.start()
|
|
|
|
taskMgr.doMethodLater(0.25, self.stopSparkParticles, 'sparkTimer-' + side, extraArgs=[side])
|
|
|
|
|
|
|
|
def stopSparkParticles(self, side = None):
|
|
|
|
sides = {0: 'right',
|
|
|
|
1: 'left'}
|
|
|
|
if side == None:
|
2019-12-30 06:07:56 +00:00
|
|
|
for x in list(sides.keys()):
|
2019-11-02 22:27:54 +00:00
|
|
|
self.sparks[x].effect.getParticlesNamed('particles-1').setBirthRate(1000)
|
|
|
|
taskMgr.doMethodLater(0.75, self.sparks[x].stop, 'stopSparks-' + sides[x], extraArgs=[])
|
|
|
|
|
|
|
|
else:
|
|
|
|
spark = self.sparks[{'right': 0,
|
|
|
|
'left': 1}[side]]
|
|
|
|
spark.effect.getParticlesNamed('particles-1').setBirthRate(1000)
|
|
|
|
taskMgr.doMethodLater(0.75, spark.stop, 'stopSparks-' + side, extraArgs=[])
|
|
|
|
return
|
|
|
|
|
|
|
|
def cleanupSparkParticles(self):
|
|
|
|
taskMgr.remove('sparkTimer-left')
|
|
|
|
taskMgr.remove('sparkTimer-right')
|
|
|
|
taskMgr.remove('stopSparks-left')
|
|
|
|
taskMgr.remove('stopSparks-right')
|
|
|
|
for x in self.sparks:
|
|
|
|
x.destroy()
|
|
|
|
|
2019-11-23 22:51:14 +00:00
|
|
|
self.sparkMount.removeNode()
|
2019-11-02 22:27:54 +00:00
|
|
|
del self.sparks
|
|
|
|
del self.sparkMount
|
|
|
|
|
|
|
|
def setState(self, state, avId):
|
|
|
|
self.notify.debug('SetState received: %s avId: %s' % (state, avId))
|
|
|
|
if state == 'C':
|
|
|
|
self.demand('Controlled', avId)
|
|
|
|
elif state == 'P':
|
|
|
|
self.demand('Parked')
|
|
|
|
else:
|
|
|
|
self.notify.error('Invalid state from AI: %s' % state)
|
|
|
|
|
|
|
|
def d_requestControl(self):
|
|
|
|
self.sendUpdate('requestControl')
|
|
|
|
|
|
|
|
def d_requestParked(self):
|
|
|
|
self.sendUpdate('requestParked')
|
|
|
|
|
|
|
|
def enterOff(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def exitOff(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def enterParked(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def exitParked(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def enterControlled(self, avId):
|
|
|
|
self.avId = avId
|
|
|
|
self.toon = base.cr.doId2do.get(avId, None)
|
|
|
|
if self.toon:
|
|
|
|
self.toon.stopSmooth()
|
|
|
|
self.toon.stopPosHprBroadcast()
|
|
|
|
for feet in self.toon.getPieces(('legs', 'feet')):
|
|
|
|
feet.hide()
|
|
|
|
|
|
|
|
self.toon.reparentTo(self.toonSeat)
|
|
|
|
self.toon.dropShadow.hide()
|
|
|
|
self.notify.debug('setting pos of toon%s' % self.toon.doId)
|
|
|
|
self.toon.setPosHpr(0, 0, 0, 0, 0, 0)
|
|
|
|
self.toon.loop('sit')
|
|
|
|
if self.toon.doId == localAvatar.doId:
|
|
|
|
self.notify.debug('calling send currentPosition')
|
|
|
|
self.toon.sendCurrentPosition()
|
|
|
|
self.doHeadScale(self.toon, 1.75)
|
|
|
|
self.toon.setShear(0, 0, 0)
|
|
|
|
NametagGlobals.setOnscreenChatForced(1)
|
|
|
|
if self.localVehicle:
|
|
|
|
camera.reparentTo(self.cameraNode)
|
|
|
|
camera.setPosHpr(0, -33, 16, 0, -10, 0)
|
|
|
|
self.physicsMgr.attachPhysicalNode(self.node())
|
|
|
|
self.__enableControlInterface()
|
|
|
|
self.__createPieWindshield()
|
|
|
|
self.startPosHprBroadcast()
|
|
|
|
self.engineSfxTrack = self.generateEngineStartTrack()
|
|
|
|
self.engineSfxTrack.start()
|
|
|
|
else:
|
|
|
|
self.startSmooth()
|
|
|
|
taskName = 'updateNonlocalVehicle-%s' % avId
|
|
|
|
self.updateNonLocalTask = taskMgr.add(self.__updateNonlocalVehicle, taskName)
|
|
|
|
return
|
|
|
|
|
|
|
|
def exitControlled(self):
|
|
|
|
if self.localVehicle:
|
|
|
|
self.stopPosHprBroadcast()
|
|
|
|
self.__disableControlInterface()
|
|
|
|
self.physicsMgr.removePhysicalNode(self.node())
|
|
|
|
self.cleanupParticles()
|
|
|
|
camera.reparentTo(localAvatar)
|
|
|
|
camera.setPos(localAvatar.cameraPositions[0][0])
|
|
|
|
camera.setHpr(0, 0, 0)
|
|
|
|
self.engineSfxTrack.finish()
|
|
|
|
self.engineSfxTrack = self.generateEngineStopTrack()
|
|
|
|
else:
|
|
|
|
self.stopSmooth()
|
|
|
|
taskMgr.remove(self.updateNonLocalTask)
|
|
|
|
if self.toon and not self.toon.isDisabled() and not self.toon.isEmpty():
|
|
|
|
self.toon.dropShadow.show()
|
|
|
|
self.doHeadScale(self.toon, None)
|
|
|
|
self.toon.setPosHpr(self.geom[0], 0, 8, 0, 0, 0, 0)
|
|
|
|
for feet in self.toon.getPieces(('legs', 'feet')):
|
|
|
|
feet.show()
|
|
|
|
|
|
|
|
self.toon.reparentTo(render)
|
|
|
|
self.toon.loop('neutral')
|
|
|
|
self.toon.startSmooth()
|
|
|
|
NametagGlobals.setOnscreenChatForced(0)
|
|
|
|
return
|
|
|
|
|
|
|
|
def doHeadScale(self, model, scale):
|
|
|
|
if scale == None:
|
|
|
|
scale = ToontownGlobals.toonHeadScales[model.style.getAnimal()]
|
|
|
|
base.localAvatar.clearCheesyEffect()
|
|
|
|
for hi in range(model.headParts.getNumPaths()):
|
|
|
|
head = model.headParts[hi]
|
|
|
|
head.setScale(scale)
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
def __createPieWindshield(self):
|
|
|
|
self.piePieces = []
|
|
|
|
for piece in range(self.numPieChunks):
|
|
|
|
self.piePieces.append(DirectLabel(relief=None, pos=(0.0, 0.0, 0.0), image=self.pieSplatter, image_scale=(0.5, 0.5, 0.5), text=' ', text_scale=0.18, text_fg=(1, 0, 1, 1), text_pos=(-0.0, 0.0, 0), text_font=ToontownGlobals.getSignFont(), textMayChange=1))
|
|
|
|
self.piePieces[piece].hide()
|
|
|
|
|
|
|
|
return
|
|
|
|
|
|
|
|
def showPieces(self):
|
|
|
|
xRange = 0.3
|
|
|
|
for piece in self.piePieces:
|
|
|
|
piece.setPos(randFloat(-xRange, xRange), randFloat(-0.1, 3.5), randFloat(-0.9, 0.9))
|
|
|
|
piece.setScale(randFloat(1.1, 2.0), 1, randFloat(1.1, 2.0))
|
|
|
|
piece.show()
|
|
|
|
xRange += 2.5 / self.numPieChunks
|
|
|
|
|
|
|
|
for piece in range(self.numPieChunks):
|
|
|
|
self.pieSlideSpeed[piece] = randFloat(0.0, 0.2)
|
|
|
|
|
|
|
|
def splatPie(self):
|
|
|
|
self.showPieces()
|
|
|
|
if self.pieCount == 0:
|
|
|
|
self.pieCount = 15
|
|
|
|
taskMgr.doMethodLater(1, self.__countPies, ' ', extraArgs=[])
|
|
|
|
taskMgr.add(self.__slidePies, 'slidePies', priority=25)
|
|
|
|
else:
|
|
|
|
self.pieCount = 15
|
|
|
|
|
|
|
|
def __countPies(self):
|
|
|
|
if self.pieCount > 0:
|
|
|
|
taskMgr.doMethodLater(1, self.__countPies, ' ', extraArgs=[])
|
|
|
|
self.pieCount -= 1
|
|
|
|
else:
|
|
|
|
for piece in self.piePieces:
|
|
|
|
piece.hide()
|
|
|
|
|
|
|
|
def __slidePies(self, optional = None):
|
|
|
|
dt = globalClock.getDt()
|
|
|
|
for piece in range(self.numPieChunks):
|
|
|
|
self.pieSlideSpeed[piece] += randFloat(0.0, 0.25 * dt)
|
|
|
|
pieSpeed = self.pieSlideSpeed[piece] * dt
|
|
|
|
self.piePieces[piece].setPos(self.piePieces[piece].getPos()[0], self.piePieces[piece].getPos()[1] - pieSpeed, self.piePieces[piece].getPos()[2] - pieSpeed)
|
|
|
|
|
|
|
|
if self.pieCount == 0:
|
|
|
|
return Task.done
|
|
|
|
else:
|
|
|
|
return Task.cont
|
|
|
|
|
|
|
|
def __enableControlInterface(self):
|
|
|
|
if self.canRace:
|
|
|
|
self.enableControls()
|
|
|
|
taskMgr.remove('watchVehicleControls')
|
|
|
|
taskMgr.add(self.__watchControls, 'watchVehicleControls', priority=25)
|
|
|
|
if not self.speedometer:
|
|
|
|
cm = CardMaker('speed')
|
|
|
|
cm.setFrame(-0.5, 0.5, -0.5, 0.5)
|
|
|
|
self.speedometerImages = aspect2d.attachNewNode('SpeedometerImages')
|
|
|
|
self.speedometerImages.setTransparency(True)
|
|
|
|
self.speedometerImages.setPos(1.24, 0.0, -0.98)
|
|
|
|
self.speedometerImages.setScale(0.75)
|
|
|
|
m = loader.loadModel('phase_6/models/karting/speedometer')
|
|
|
|
if self.getBodyColor() == InvalidEntry:
|
|
|
|
bodyColor = getDefaultColor()
|
|
|
|
else:
|
|
|
|
bodyColor = getAccessory(self.getBodyColor())
|
|
|
|
if self.getAccessoryColor() == InvalidEntry:
|
|
|
|
accColor = getDefaultColor()
|
|
|
|
else:
|
|
|
|
accColor = getAccessory(self.getAccessoryColor())
|
|
|
|
self.speedometerImages.attachNewNode(m.find('**/*gauge').node()).setColorScale(1, 1, 1, 1)
|
|
|
|
self.speedGauge = self.speedometerImages.attachNewNode(m.find('**/*spin').node())
|
|
|
|
self.speedGauge.setColor(1, 1, 1, 1)
|
|
|
|
c = (Vec4(1, 1, 1, 1) + accColor) / 2.0
|
|
|
|
c.setW(1.0)
|
|
|
|
self.speedometerImages.attachNewNode(m.find('**/*face').node()).setColorScale(c)
|
|
|
|
c = (Vec4(2, 2, 2, 2) - accColor) / 2.0
|
|
|
|
c = (bodyColor + Vec4(1, 1, 1, 1)) / 2.0
|
|
|
|
c.setW(1.0)
|
|
|
|
self.speedometerImages.attachNewNode(m.find('**/*tics').node()).setColorScale(c)
|
|
|
|
c = (bodyColor + Vec4(1, 1, 1, 1)) / 2.0
|
|
|
|
c.setW(1.0)
|
|
|
|
self.speedometerImages.attachNewNode(m.find('**/*ring').node()).setColorScale(c)
|
|
|
|
self.speedometer = DirectLabel(relief=None, pos=(1.24, 0.0, -0.98), text=str(0), text_scale=0.18, text_fg=bodyColor, text_pos=(-0.04, 0.02, 0), text_font=ToontownGlobals.getSignFont())
|
|
|
|
else:
|
|
|
|
self.showSpeedometer()
|
|
|
|
self.arrowVert = 0
|
|
|
|
self.arrowHorz = 0
|
|
|
|
return
|
|
|
|
|
|
|
|
def showSpeedometer(self):
|
|
|
|
if self.speedometer:
|
|
|
|
self.speedometer.show()
|
|
|
|
self.speedometerImages.show()
|
|
|
|
|
|
|
|
def hideSpeedometer(self):
|
|
|
|
if self.speedometer:
|
|
|
|
self.speedometer.hide()
|
|
|
|
self.speedometerImages.hide()
|
|
|
|
|
|
|
|
def __disableControlInterface(self):
|
|
|
|
self.hideSpeedometer()
|
|
|
|
self.disableControls()
|
|
|
|
self.stopSmooth()
|
|
|
|
taskMgr.remove('watchVehicleControls')
|
|
|
|
|
|
|
|
def __deleteControlInterface(self):
|
|
|
|
if self.speedometer:
|
|
|
|
self.speedometer.destroy()
|
|
|
|
self.speedometer = None
|
|
|
|
self.speedGauge = None
|
|
|
|
self.speedometerImages.removeNode()
|
|
|
|
self.speedometerImages = None
|
|
|
|
return
|
|
|
|
|
|
|
|
def setTurbo(self, setting):
|
|
|
|
self.turbo = setting
|
|
|
|
if self.turbo:
|
|
|
|
self.turboLoopSfx.play()
|
|
|
|
else:
|
|
|
|
self.turboLoopSfx.stop()
|
|
|
|
|
|
|
|
def interruptTurbo(self):
|
|
|
|
self.__stopTurbo()
|
|
|
|
if self.cameraTrack:
|
|
|
|
self.cameraTrack.finish()
|
|
|
|
self.cameraTrack = None
|
|
|
|
return
|
|
|
|
|
|
|
|
def startTurbo(self):
|
|
|
|
newCameraPos = Point3(0, -25, 16)
|
|
|
|
newCameraFov = 90
|
|
|
|
turboDuration = 3
|
|
|
|
startFov = base.camLens.getFov().getX()
|
|
|
|
if self.cameraTrack:
|
|
|
|
self.cameraTrack.pause()
|
|
|
|
cameraZoomIn = Parallel(LerpPosInterval(camera, 2, newCameraPos), LerpFunc(base.camLens.setFov, fromData=startFov, toData=newCameraFov, duration=2))
|
|
|
|
cameraToNormal = Parallel(LerpPosInterval(camera, 1, Point3(0, -33, 16), newCameraPos), LerpFunc(base.camLens.setFov, fromData=newCameraFov, toData=ToontownGlobals.DefaultCameraFov, duration=1))
|
|
|
|
self.cameraTrack = Sequence(Func(self.turboStartSfx.play), cameraZoomIn, Func(lambda : self.setTurbo(True)), Wait(turboDuration), Func(self.__stopTurbo), cameraToNormal)
|
|
|
|
self.cameraTrack.start()
|
|
|
|
|
|
|
|
def __stopTurbo(self, task = None):
|
|
|
|
self.setTurbo(False)
|
|
|
|
|
|
|
|
def __controlPressed(self):
|
|
|
|
self.__throwGag()
|
|
|
|
|
|
|
|
def __controlReleased(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def __upArrow(self, pressed):
|
|
|
|
if pressed:
|
|
|
|
self.arrowVert = 1
|
|
|
|
elif self.arrowVert > 0:
|
|
|
|
self.arrowVert = 0
|
|
|
|
|
|
|
|
def __downArrow(self, pressed):
|
|
|
|
if pressed:
|
|
|
|
self.arrowVert = -1
|
|
|
|
elif self.arrowVert < 0:
|
|
|
|
self.arrowVert = 0
|
|
|
|
|
|
|
|
def __rightArrow(self, pressed):
|
|
|
|
if pressed:
|
|
|
|
self.arrowHorz = 1
|
|
|
|
self.rightHeld += 1
|
|
|
|
elif self.arrowHorz > 0:
|
|
|
|
self.arrowHorz = 0
|
|
|
|
self.rightHeld = 0
|
|
|
|
|
|
|
|
def __leftArrow(self, pressed):
|
|
|
|
if pressed:
|
|
|
|
self.arrowHorz = -1
|
|
|
|
self.leftHeld += 1
|
|
|
|
elif self.arrowHorz < 0:
|
|
|
|
self.arrowHorz = 0
|
|
|
|
self.leftHeld = 0
|
|
|
|
|
|
|
|
def __updateNonlocalVehicle(self, task = None):
|
|
|
|
self.curSpeed = self.smoother.getSmoothForwardVelocity()
|
|
|
|
rotSpeed = -1 * self.smoother.getSmoothRotationalVelocity()
|
|
|
|
self.leanAmount = self.curSpeed * rotSpeed / 500.0
|
|
|
|
self.leanAmount = clampScalar(self.leanAmount, -10, 10)
|
|
|
|
self.__animate()
|
|
|
|
return Task.cont
|
|
|
|
|
|
|
|
def __animate(self):
|
|
|
|
speed = self.curSpeed
|
|
|
|
self.spinWheels(speed / 10)
|
|
|
|
enginePitch = clampScalar(speed / 120.0, 0.5, 15)
|
|
|
|
self.kartLoopSfx.setPlayRate(enginePitch)
|
|
|
|
if not self.localVehicle:
|
|
|
|
dist = (self.getPos() - localAvatar.getPos()).length()
|
|
|
|
if self.localVehicle:
|
|
|
|
if self.lifter.isOnGround():
|
|
|
|
if self.offGround > 10:
|
|
|
|
kart = self.geom[0].find('**/main*')
|
|
|
|
bumpDown1 = kart.posInterval(0.1, Vec3(0, 0, -1))
|
|
|
|
bumpUp1 = kart.posInterval(0.15, Vec3(0, 0, 0))
|
|
|
|
bumpDown2 = kart.posInterval(0.5, Vec3(0, 0, -.4))
|
|
|
|
bumpUp2 = kart.posInterval(0.7, Vec3(0, 0, 0))
|
|
|
|
bumpSeq = Sequence(bumpDown1, bumpUp1, bumpDown2, bumpUp2)
|
|
|
|
bumpSeq.start()
|
|
|
|
self.offGround = 0
|
|
|
|
else:
|
|
|
|
self.offGround += 1
|
|
|
|
if self.offGround == 0:
|
|
|
|
modifier = self.surfaceModifiers[self.groundType]['shake'] * (speed / 50.0)
|
|
|
|
else:
|
|
|
|
modifier = 1
|
|
|
|
roll = self.leanAmount * 1.5
|
|
|
|
roll += (random.random() * 2 - 1) * modifier
|
|
|
|
pitch = self.acceleration * -.005
|
|
|
|
pitch += (random.random() * 2 - 1) * modifier
|
|
|
|
self.rollSuspension(roll)
|
|
|
|
self.pitchSuspension(pitch)
|
|
|
|
return Task.cont
|
|
|
|
|
|
|
|
def __clampPosition(self):
|
|
|
|
retVal = False
|
|
|
|
if self.getX() < -3276:
|
|
|
|
self.notify.debug('clamping X %d' % self.getX())
|
|
|
|
self.setX(-3276)
|
|
|
|
retVal = True
|
|
|
|
if self.getX() > 3276:
|
|
|
|
self.notify.debug('clamping X %d' % self.getX())
|
|
|
|
self.setX(3276)
|
|
|
|
retVal = True
|
|
|
|
if self.getY() < -3276:
|
|
|
|
self.notify.debug('clamping Y %d' % self.getY())
|
|
|
|
self.setY(-3276)
|
|
|
|
retVal = True
|
|
|
|
if self.getY() > 3276:
|
|
|
|
self.notify.debug('clamping Y %d' % self.getY())
|
|
|
|
self.setY(3276)
|
|
|
|
retVal = True
|
|
|
|
if self.getZ() < -3276:
|
|
|
|
self.notify.debug('clamping Z %d' % self.getZ())
|
|
|
|
self.setZ(-3276)
|
|
|
|
retVal = True
|
|
|
|
if self.getZ() > 3276:
|
|
|
|
self.notify.debug('clamping Z %d' % self.getZ())
|
|
|
|
self.setZ(3276)
|
|
|
|
retVal = True
|
|
|
|
return retVal
|
|
|
|
|
|
|
|
def amIClampingPosition(self):
|
|
|
|
return self.getX() == -3276 or self.getX() == 3276 or self.getY() == -3276 or self.getY() == 3276 or self.getZ() == -3276 or self.getZ() == 3276
|
|
|
|
|
|
|
|
def __computeTurnRatio(self, speed):
|
|
|
|
effectiveSpeed = speed
|
|
|
|
if effectiveSpeed > self.speedMinTurn:
|
|
|
|
effectiveSpeed = self.speedMinTurn
|
|
|
|
if effectiveSpeed < self.speedMaxTurn:
|
|
|
|
turnRatio = 1.0 * effectiveSpeed
|
|
|
|
elif effectiveSpeed >= self.speedMaxTurn:
|
|
|
|
turnRatio = 1.0 * (effectiveSpeed * (self.speedMaxTurn / speed))
|
|
|
|
|
|
|
|
def __updateWheelPos(self, dt, curSpeed):
|
|
|
|
ratioIntoMax = (curSpeed - self.speedMinTurn) / (self.speedMaxTurn - self.speedMinTurn)
|
|
|
|
if ratioIntoMax > 1.0:
|
|
|
|
ratioIntoMax = 1.0
|
|
|
|
if ratioIntoMax < 0.0:
|
|
|
|
ratioIntoMax = 0.0
|
|
|
|
maxWheelDeflection = self.wheelMaxTurn * ratioIntoMax + self.wheelMinTurn * (1.0 - ratioIntoMax)
|
|
|
|
if self.wheelPosition * self.arrowHorz > 0:
|
|
|
|
self.wheelPosition += self.arrowHorz * dt / self.wheelTurnTime
|
|
|
|
else:
|
|
|
|
self.wheelPosition += self.arrowHorz * dt / self.wheelFightTime
|
|
|
|
if not self.arrowHorz:
|
|
|
|
if self.wheelPosition > 0:
|
|
|
|
self.wheelPosition -= dt / self.wheelReturnTime
|
|
|
|
if self.wheelPosition < 0:
|
|
|
|
self.wheelPosition = 0
|
|
|
|
else:
|
|
|
|
self.wheelPosition += dt / self.wheelReturnTime
|
|
|
|
if self.wheelPosition > 0:
|
|
|
|
self.wheelPosition = 0
|
|
|
|
if abs(self.wheelPosition) >= maxWheelDeflection:
|
|
|
|
if self.wheelPosition > 0:
|
|
|
|
self.wheelPosition = maxWheelDeflection
|
|
|
|
else:
|
|
|
|
self.wheelPosition = -maxWheelDeflection
|
|
|
|
|
|
|
|
def __watchControls(self, task):
|
|
|
|
dt = globalClock.getDt()
|
|
|
|
curVelocity = self.actorNode.getPhysicsObject().getVelocity()
|
|
|
|
curSpeed = curVelocity.length()
|
|
|
|
fvec = self.forward.getPos(render) - self.getPos(render)
|
|
|
|
fvec.normalize()
|
|
|
|
dotProduct = curVelocity.dot(fvec)
|
|
|
|
goingBack = -1
|
|
|
|
if dotProduct < 0:
|
|
|
|
goingBack = 1
|
|
|
|
if self.proRacer:
|
|
|
|
self.__computeTurnRatio(curSpeed)
|
|
|
|
self.__updateWheelPos(dt, curSpeed)
|
|
|
|
newHForTurning = self.getH()
|
|
|
|
if self.proRacer or not self.stopped and self.arrowHorz and curSpeed > 1:
|
|
|
|
if self.proRacer:
|
|
|
|
turnHelp = 0
|
|
|
|
if self.hittingWall or goingBack == 1:
|
|
|
|
turnHelp = self.arrowHorz
|
|
|
|
effectiveSpeed = curSpeed
|
|
|
|
if effectiveSpeed > self.speedMinTurn:
|
|
|
|
effectiveSpeed = self.speedMinTurn
|
|
|
|
rotation = -goingBack * (self.wheelPosition * dt * self.turnRatio * -1.8 * curSpeed / 100 + turnHelp * dt * self.turnRatio * -1.2)
|
|
|
|
self.outPutCounter += 1
|
|
|
|
if self.outPutCounter > 5:
|
|
|
|
self.outPutCounter = 0
|
|
|
|
else:
|
|
|
|
rotation = self.arrowHorz * dt * self.turnRatio * -1.2
|
|
|
|
oldH = self.getH()
|
|
|
|
newH = (oldH + rotation) % 360
|
|
|
|
self.setH(newH)
|
|
|
|
newHForTurning = newH
|
|
|
|
if self.groundType == 'ice':
|
|
|
|
newHForTurning = (oldH + rotation * iceTurnFactor) % 360
|
|
|
|
pitch = -self.getP() + 5
|
|
|
|
accelBase = self.accelerationBase
|
|
|
|
pitch += accelBase
|
|
|
|
pitch = clampScalar(pitch, accelBase - 5, accelBase + 5)
|
|
|
|
self.accelerationMult = pitch * 2
|
|
|
|
if self.groundType == 'ice':
|
|
|
|
self.accelerationMult *= iceAccelFactor
|
|
|
|
if self.stopped:
|
|
|
|
self.acceleration = 0
|
|
|
|
else:
|
|
|
|
self.acceleration = self.arrowVert * self.accelerationMult * self.cheatFactor
|
|
|
|
if self.proRacer:
|
|
|
|
if self.skidding:
|
|
|
|
self.acceleration = self.arrowVert * self.accelerationMult * self.cheatFactor * 0.5
|
|
|
|
if self.turbo:
|
|
|
|
self.acceleration += self.accelerationMult * 1.5
|
|
|
|
self.engine.setVector(Vec3(0, self.acceleration, 0))
|
|
|
|
if self.groundType == 'ice':
|
|
|
|
rotMat = Mat3.rotateMatNormaxis(newHForTurning, Vec3.up())
|
|
|
|
else:
|
|
|
|
rotMat = Mat3.rotateMatNormaxis(self.getH(), Vec3.up())
|
|
|
|
curHeading = rotMat.xform(Vec3.forward())
|
|
|
|
push = (3 - self.getP()) * 0.02
|
|
|
|
curHeading.setZ(curHeading.getZ() - min(0.2, max(-.2, push)))
|
|
|
|
onScreenDebug.append('vehicle curHeading = %s\n' % curHeading.pPrintValues())
|
|
|
|
onScreenDebug.append('vehicle H = %s newHForTurning=%f\n' % (self.getH(), newHForTurning))
|
|
|
|
windResistance = self.surfaceModifiers[self.groundType]['windResistance']
|
|
|
|
self.windResistance.setCoef(windResistance)
|
|
|
|
physicsFrame = int((globalClock.getFrameTime() - self.physicsEpoch) * self.physicsCalculationsPerSecond)
|
|
|
|
numFrames = min(physicsFrame - self.lastPhysicsFrame, self.maxPhysicsFrames)
|
|
|
|
self.lastPhysicsFrame = physicsFrame
|
|
|
|
leanIncrement = self.arrowHorz * self.physicsDt * self.turnRatio
|
|
|
|
if self.stopped:
|
|
|
|
leanIncrement = 0
|
|
|
|
driftMin = self.surfaceModifiers[self.groundType]['driftMin']
|
|
|
|
if self.proRacer:
|
|
|
|
driftMin = self.surfaceModifiers[self.groundType]['driftMin'] * 0.2
|
|
|
|
if self.skidding:
|
|
|
|
driftMin = self.surfaceModifiers[self.groundType]['driftMin']
|
|
|
|
for i in range(numFrames):
|
|
|
|
self.physicsMgr.doPhysics(self.physicsDt)
|
|
|
|
curVelocity = self.actorNode.getPhysicsObject().getVelocity()
|
|
|
|
idealVelocity = curHeading * curSpeed
|
|
|
|
curVelocity *= self.imHitMult
|
|
|
|
driftVal = abs(self.leanAmount) * 16 / self.cheatFactor + 15 / self.cheatFactor
|
|
|
|
curVelocity = Vec3((curVelocity * driftVal + idealVelocity) / (driftVal + 1))
|
|
|
|
curSpeed = curVelocity.length()
|
|
|
|
curVelocity.normalize()
|
|
|
|
curVelocity *= min(curSpeed, 600)
|
|
|
|
self.actorNode.getPhysicsObject().setVelocity(curVelocity)
|
|
|
|
curSpeed = curVelocity.length()
|
|
|
|
speedFactor = min(curSpeed, 150) / 162.0
|
|
|
|
self.leanAmount = (self.leanAmount + leanIncrement) * speedFactor
|
|
|
|
self.leanAmount = clampScalar(self.leanAmount, -10, 10)
|
|
|
|
|
|
|
|
self.cWallTrav.traverse(render)
|
|
|
|
self.curSpeed = curSpeed
|
|
|
|
if self.proRacer:
|
|
|
|
self.turnWheels(self.wheelPosition * -45)
|
|
|
|
else:
|
|
|
|
self.turnWheels(self.arrowHorz * -10)
|
|
|
|
self.__animate()
|
|
|
|
if self.proRacer:
|
|
|
|
speedProporation = 1.0
|
|
|
|
if curSpeed < self.speedMaxTurn:
|
|
|
|
speedProporation = 0.0
|
|
|
|
else:
|
|
|
|
speedProporation = (curSpeed - self.speedMaxTurn) / self.speedMinTurn
|
|
|
|
cameraDist = self.cameraArmBase + self.cameraArmExtend * speedProporation
|
|
|
|
cameraOffset = Point3(0, -cameraDist, 0)
|
|
|
|
self.cameraNode.setPos(cameraOffset)
|
|
|
|
behindPos = render.getRelativePoint(self, Point3(0, -30, 0))
|
|
|
|
self.proCameraDummyNode.setPos(render, behindPos)
|
|
|
|
self.proCameraDummyNode.lookAt(self)
|
|
|
|
self.cameraNode.lookAt(self)
|
|
|
|
dir1 = self.proCameraDummyNode.getH()
|
|
|
|
dir2 = self.proCameraHeading
|
|
|
|
if dir1 > 180:
|
|
|
|
dir1 -= 360
|
|
|
|
elif dir1 < -180:
|
|
|
|
dir1 += 360
|
|
|
|
if dir2 > 180:
|
|
|
|
dir2 -= 360
|
|
|
|
elif dir2 < -180:
|
|
|
|
dir2 += 360
|
|
|
|
self.proCameraHeading = dir2
|
|
|
|
dif = dir1 - dir2
|
|
|
|
if dif > 180:
|
|
|
|
dif -= 360
|
|
|
|
elif dif < -180:
|
|
|
|
dif += 360
|
|
|
|
speedDif = abs(dif)
|
|
|
|
if speedDif > 90:
|
|
|
|
speedDif = 90
|
|
|
|
cameraTightener = 1.0
|
|
|
|
if curSpeed > self.speedMinTurn:
|
|
|
|
cameraTightener = self.cameraTightener
|
|
|
|
else:
|
|
|
|
cameraTightener = 1.0 + curSpeed / self.speedMinTurn * (self.cameraTightener - 1.0)
|
|
|
|
swingSpeedRatio = speedDif / 90
|
|
|
|
swingSpeed = self.armSwingSpeedPerp * swingSpeedRatio + self.armSwingSpeedPara * (1 - swingSpeedRatio)
|
|
|
|
self.proCameraHeading += dif * cameraTightener * (dt / swingSpeed)
|
|
|
|
self.cameraArmNode.setH(self.proCameraHeading - self.getH())
|
|
|
|
elif not self.stopped:
|
|
|
|
self.cameraNode.setH(self.leanAmount)
|
|
|
|
self.updateParticles(self.leanAmount)
|
|
|
|
if (self.leanAmount > 8 or self.leanAmount < -8) and self.offGround <= 0:
|
|
|
|
self.startSkid()
|
|
|
|
else:
|
|
|
|
self.stopSkid()
|
|
|
|
if self.speedometer:
|
|
|
|
self.speedometer['text'] = str(int(curSpeed / 3))
|
|
|
|
self.speedGauge.setR(min(110, max(0, curSpeed / 3 / 120 * 110)))
|
|
|
|
if not self.stopped:
|
|
|
|
self.stickCarToGround()
|
|
|
|
if self.__clampPosition():
|
|
|
|
self.notify.debug('did a clampPosition on %d' % self.doId)
|
|
|
|
return Task.cont
|
|
|
|
|
|
|
|
def enableControls(self):
|
|
|
|
self.canRace = True
|
|
|
|
self.accept('control', self.__controlPressed)
|
|
|
|
self.accept('control-up', self.__controlReleased)
|
|
|
|
self.accept('InputState-forward', self.__upArrow)
|
|
|
|
self.accept('InputState-reverse', self.__downArrow)
|
|
|
|
self.accept('InputState-turnLeft', self.__leftArrow)
|
|
|
|
self.accept('InputState-turnRight', self.__rightArrow)
|
|
|
|
|
|
|
|
def disableControls(self):
|
|
|
|
self.arrowVert = 0
|
|
|
|
self.arrowHorz = 0
|
|
|
|
self.ignore('control')
|
|
|
|
self.ignore('control-up')
|
|
|
|
self.ignore('tab')
|
|
|
|
self.ignore('InputState-forward')
|
|
|
|
self.ignore('InputState-reverse')
|
|
|
|
self.ignore('InputState-turnLeft')
|
|
|
|
self.ignore('InputState-turnRight')
|
|
|
|
|
|
|
|
def setInput(self, on):
|
|
|
|
if localAvatar.doId == self.ownerId:
|
|
|
|
if on:
|
|
|
|
self.enableControls()
|
|
|
|
else:
|
|
|
|
self.disableControls()
|
|
|
|
|
|
|
|
def shakeWheel(self, nWheel, floorNode):
|
|
|
|
groundType = floorNode.getNetTag('surface')
|
|
|
|
modifier = self.surfaceModifiers[groundType]['shake']
|
|
|
|
shakeFactor = (random.random() * 2 - 1) * modifier
|
|
|
|
shakeFactor *= self.curSpeed / 200
|
|
|
|
node = self.wheelCenters[nWheel].find('*Wheel')
|
|
|
|
node.setZ(shakeFactor)
|
|
|
|
|
|
|
|
def stickCarToGround(self):
|
|
|
|
posList = []
|
|
|
|
nWheels = len(self.wheelData)
|
|
|
|
for nWheel in range(nWheels):
|
|
|
|
cQueue = self.cQueue[nWheel]
|
|
|
|
cQueue.sortEntries()
|
|
|
|
if cQueue.getNumEntries() == 0:
|
|
|
|
return
|
|
|
|
entry = cQueue.getEntry(0)
|
|
|
|
self.shakeWheel(nWheel, entry.getIntoNodePath())
|
|
|
|
pos = entry.getSurfacePoint(render)
|
|
|
|
wheelPos = self.wheelBases[nWheel].getPos(render)
|
|
|
|
pos = wheelPos - pos
|
|
|
|
posList.append(pos)
|
|
|
|
cQueue.clearEntries()
|
|
|
|
|
|
|
|
rf = posList[0].getZ()
|
|
|
|
lf = posList[1].getZ()
|
|
|
|
rr = posList[2].getZ()
|
|
|
|
lr = posList[3].getZ()
|
|
|
|
right = (rf + rr) / 2
|
|
|
|
left = (lf + lr) / 2
|
|
|
|
rollVal = right - left
|
|
|
|
rollVal = clampScalar(rollVal, -1, 1)
|
|
|
|
curRoll = self.getR()
|
|
|
|
newRoll = curRoll + rollVal * 2.0
|
|
|
|
self.setR(newRoll)
|
|
|
|
if not self.stopped:
|
|
|
|
camera.setR(-newRoll)
|
|
|
|
front = (rf + lf) / 2
|
|
|
|
rear = (rr + lr) / 2
|
|
|
|
center = (front + rear) / 2
|
|
|
|
pitchVal = front - rear
|
|
|
|
pitchVal = clampScalar(pitchVal, -1, 1)
|
|
|
|
curPitch = self.getP()
|
|
|
|
newPitch = curPitch - pitchVal * 2.0
|
|
|
|
self.setP((newPitch + curPitch) / 2.0)
|
|
|
|
if self.proRacer:
|
|
|
|
self.cameraNode.setP(-newPitch)
|
|
|
|
elif not self.stopped:
|
|
|
|
self.cameraNode.setP(-newPitch)
|
|
|
|
|
|
|
|
def setBodyType(self, bodyType):
|
|
|
|
self.kartDNA[KartDNA.bodyType] = bodyType
|
|
|
|
|
|
|
|
def setBodyColor(self, bodyColor):
|
|
|
|
self.kartDNA[KartDNA.bodyColor] = bodyColor
|
|
|
|
|
|
|
|
def setAccessoryColor(self, accColor):
|
|
|
|
self.kartDNA[KartDNA.accColor] = accColor
|
|
|
|
|
|
|
|
def setEngineBlockType(self, ebType):
|
|
|
|
self.kartDNA[KartDNA.ebType] = ebType
|
|
|
|
|
|
|
|
def setSpoilerType(self, spType):
|
|
|
|
self.kartDNA[KartDNA.spType] = spType
|
|
|
|
|
|
|
|
def setFrontWheelWellType(self, fwwType):
|
|
|
|
self.kartDNA[KartDNA.fwwType] = fwwType
|
|
|
|
|
|
|
|
def setBackWheelWellType(self, bwwType):
|
|
|
|
self.kartDNA[KartDNA.bwwType] = bwwType
|
|
|
|
|
|
|
|
def setRimType(self, rimsType):
|
|
|
|
self.kartDNA[KartDNA.rimsType] = rimsType
|
|
|
|
|
|
|
|
def setDecalType(self, decalType):
|
|
|
|
self.kartDNA[KartDNA.decalType] = decalType
|
|
|
|
|
|
|
|
def setOwner(self, avId):
|
|
|
|
self.ownerId = avId
|
|
|
|
DistributedVehicle.AvId2kart[avId] = self
|
|
|
|
|
|
|
|
def stopCar(self, level):
|
|
|
|
self.imHitMult = level
|
|
|
|
if hasattr(self, 'cameraTrack') and self.cameraTrack:
|
|
|
|
self.cameraTrack.pause()
|
|
|
|
cameraToNormal = Parallel(LerpPosInterval(camera, 0.05, Point3(0, -33, 16), startPos=camera.getPos()), LerpFunc(base.camLens.setFov, fromData=base.camLens.getFov()[0], toData=ToontownGlobals.DefaultCameraFov, duration=0.05))
|
|
|
|
cameraToNormal.start()
|
|
|
|
self.__stopTurbo()
|
|
|
|
self.stopped = True
|
|
|
|
|
|
|
|
def startCar(self):
|
|
|
|
self.imHitMult = 1
|
|
|
|
self.stopped = False
|
|
|
|
|
|
|
|
def hideAnvil(self):
|
|
|
|
self.anvil.reparentTo(hidden)
|
|
|
|
self.anvil.setAlphaScale(1)
|
|
|
|
self.dropShadow.setScale(self.ShadowScale)
|
|
|
|
|
|
|
|
def spinCar(self, spin):
|
|
|
|
self.geom[0].setH(2 * spin)
|
|
|
|
|
|
|
|
def playSpin(self, timeStamp):
|
|
|
|
delta = -globalClockDelta.networkToLocalTime(timeStamp, globalClock.getFrameTime(), 16, 100) + globalClock.getFrameTime()
|
|
|
|
self.spinAnim = LerpFunc(self.spinCar, fromData=0, toData=360, duration=min(1 - delta, 0))
|
|
|
|
self.spinAnim.start()
|
|
|
|
|
|
|
|
def __throwGag(self):
|
|
|
|
base.race.shootGag()
|
|
|
|
|
|
|
|
def dropOnMe(self, timestamp):
|
|
|
|
self.anvil.setScale(7, 9, 7)
|
|
|
|
self.anvil.setPos(0, 0, 75)
|
|
|
|
self.anvil.reparentTo(self)
|
|
|
|
anvilDrop = ProjectileInterval(self.anvil, startPos=Point3(0, 0, 100), endPos=Point3(0, 0, 0), duration=1)
|
|
|
|
shadowScale = self.dropShadow.scaleInterval(1, self.ShadowScale * 2)
|
|
|
|
fadeOut = LerpFunc(self.anvil.setAlphaScale, fromData=1, toData=0, duration=0.5)
|
|
|
|
flattenMe = self.geom[0].scaleInterval(0.08, Vec3(1, 1, 0.1))
|
|
|
|
unFlatten = Sequence(self.geom[0].scaleInterval(0.2, Vec3(1, 1, 2)), self.geom[0].scaleInterval(0.1, 1))
|
|
|
|
anvilSquish = Parallel(Parallel(anvilDrop, shadowScale), Sequence(Wait(0.92), flattenMe))
|
|
|
|
anvilStretchFade = Sequence(Parallel(self.anvil.scaleInterval(0.2, Vec3(7, 9, 5)), Func(self.lookUp)), fadeOut)
|
|
|
|
self.gagMovie = Sequence(anvilSquish, Func(self.stopCar, 0.9), anvilStretchFade, Wait(0.5), Parallel(unFlatten, Func(self.lookNormal)), Func(self.startCar), Func(self.hideAnvil))
|
|
|
|
self.gagMovie.start()
|
|
|
|
|
|
|
|
def lookUp(self):
|
|
|
|
if self.toon and self.toon.headParts:
|
|
|
|
headParts = self.toon.headParts
|
|
|
|
for hi in range(headParts.getNumPaths()):
|
|
|
|
head = headParts[hi]
|
|
|
|
head.setP(90)
|
|
|
|
|
|
|
|
def lookNormal(self):
|
|
|
|
if self.toon and self.toon.headParts:
|
|
|
|
headParts = self.toon.headParts
|
|
|
|
for hi in range(headParts.getNumPaths()):
|
|
|
|
head = headParts[hi]
|
|
|
|
head.setP(0)
|
|
|
|
|
|
|
|
def flattenMe(self):
|
|
|
|
self.geom[0].setScale(3, 3, 0.1)
|
|
|
|
|
|
|
|
def unFlattenMe(self):
|
|
|
|
self.geom[0].setScale(2)
|
|
|
|
|
|
|
|
def startSkid(self):
|
|
|
|
if self.skidding == False:
|
|
|
|
self.skidding = True
|
|
|
|
self.skidLoopAsphaltSfx.play()
|
|
|
|
|
|
|
|
def stopSkid(self):
|
|
|
|
if self.skidding == True:
|
|
|
|
self.skidding = False
|
|
|
|
self.skidLoopAsphaltSfx.stop()
|
|
|
|
self.skidLoopGrassSfx.stop()
|
|
|
|
|
|
|
|
def updateSkid(self):
|
|
|
|
if self.skidding:
|
|
|
|
if self.groundType == 'grass':
|
|
|
|
self.skidLoopGrassSfx.setVolume(1)
|
|
|
|
self.skidLoopAsphaltSfx.setVolume(0)
|
|
|
|
else:
|
|
|
|
self.skidLoopGrassSfx.setVolume(0)
|
|
|
|
self.skidLoopAsphaltSfx.setVolume(1)
|
|
|
|
|
|
|
|
def dropOnHim(self, timestamp):
|
|
|
|
self.anvil.setScale(10, 13, 10)
|
|
|
|
self.anvil.setPos(0, 0, 75)
|
|
|
|
self.anvil.reparentTo(self)
|
|
|
|
anvilDrop = ProjectileInterval(self.anvil, startPos=Point3(0, 0, 100), endPos=Point3(0, 0, 0), duration=1)
|
|
|
|
shadowScale = self.dropShadow.scaleInterval(1, self.ShadowScale * 4)
|
|
|
|
fadeOut = LerpFunc(self.anvil.setAlphaScale, fromData=1, toData=0, duration=0.5)
|
|
|
|
flattenMe = self.geom[0].scaleInterval(0.08, Vec3(1, 1, 0.1))
|
|
|
|
unFlatten = Sequence(self.geom[0].scaleInterval(0.2, Vec3(1, 1, 2)), self.geom[0].scaleInterval(0.1, 1))
|
|
|
|
anvilSquish = Parallel(Parallel(anvilDrop, shadowScale), Sequence(Wait(0.92), flattenMe))
|
|
|
|
anvilStretchFade = Sequence(Parallel(self.anvil.scaleInterval(0.2, Vec3(10, 13, 4)), Func(self.lookUp)), fadeOut)
|
|
|
|
self.gagMovie = Sequence(anvilSquish, anvilStretchFade, Wait(0.5), Parallel(unFlatten, Func(self.lookNormal)), Func(self.hideAnvil))
|
|
|
|
self.gagMovie.start()
|
|
|
|
|
|
|
|
def hitBanana(self):
|
|
|
|
if self.wipeOut:
|
|
|
|
self.wipeOut.pause()
|
|
|
|
spinAnim = LerpFunc(self.spinCar, fromData=self.geom[0].getH(), toData=360, duration=1)
|
|
|
|
else:
|
|
|
|
spinAnim = LerpFunc(self.spinCar, fromData=0, toData=360, duration=1)
|
|
|
|
self.wipeOut = Sequence(Func(self.stopCar, 0.99), spinAnim, Func(self.startCar))
|
|
|
|
self.wipeOut.start()
|
|
|
|
|
|
|
|
def hitPie(self):
|
2019-12-30 06:07:56 +00:00
|
|
|
print('yar, got Me with pi!')
|
2019-11-02 22:27:54 +00:00
|
|
|
self.splatPie()
|
|
|
|
if self.wipeOut:
|
|
|
|
self.wipeOut.pause()
|
|
|
|
spinAnim = LerpFunc(self.spinCar, fromData=self.geom[0].getH(), toData=1080, duration=0.5)
|
|
|
|
else:
|
|
|
|
spinAnim = LerpFunc(self.spinCar, fromData=0, toData=1080, duration=0.5)
|
|
|
|
self.wipeOut = Sequence(Func(self.stopCar, 0.99), spinAnim, Func(self.startCar))
|
|
|
|
self.wipeOut.start()
|
|
|
|
|
|
|
|
def finishMovies(self):
|
|
|
|
if self.gagMovie:
|
|
|
|
self.gagMovie.finish()
|
|
|
|
self.gagMovie = None
|
|
|
|
if self.wipeOut:
|
|
|
|
self.wipeOut.finish()
|
|
|
|
self.wipeOut = None
|
|
|
|
if self.spinAnim:
|
|
|
|
self.spinAnim.finish()
|
|
|
|
self.spinAnim = None
|
|
|
|
return
|