2015-07-12 19:42:55 +00:00
|
|
|
from direct.directnotify import DirectNotifyGlobal
|
|
|
|
from direct.distributed.ClockDelta import *
|
2015-03-03 22:10:12 +00:00
|
|
|
from toontown.estate.DistributedFurnitureItemAI import DistributedFurnitureItemAI
|
2015-07-12 19:42:55 +00:00
|
|
|
from PhoneGlobals import *
|
|
|
|
|
2015-03-03 22:10:12 +00:00
|
|
|
from toontown.toonbase import ToontownGlobals
|
2015-07-17 11:50:16 +00:00
|
|
|
from toontown.catalog import CatalogItem, CatalogInvalidItem, GiftAvatar
|
2015-03-03 22:10:12 +00:00
|
|
|
from toontown.catalog.CatalogItemList import CatalogItemList
|
2015-07-12 19:42:55 +00:00
|
|
|
from toontown.uberdog import TopToonsGlobals
|
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
import json
|
2015-03-03 22:10:12 +00:00
|
|
|
import time
|
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
class LoadGiftAvatar:
|
|
|
|
|
|
|
|
def __init__(self, phone, avId, targetId, optional, callback):
|
|
|
|
self.air = phone.air
|
|
|
|
self.phone = phone
|
|
|
|
self.avId = avId
|
|
|
|
self.targetId = targetId
|
|
|
|
self.optional = optional
|
|
|
|
self.callback = callback
|
|
|
|
|
|
|
|
def start(self):
|
|
|
|
self.air.dbInterface.queryObject(self.air.dbId, self.targetId, self.__gotAvatar)
|
|
|
|
|
|
|
|
def copyDict(self, dict, *keys):
|
|
|
|
return {key: dict[key] for key in keys}
|
|
|
|
|
|
|
|
def __gotAvatar(self, dclass, fields):
|
|
|
|
if dclass != self.air.dclassesByName['DistributedToonAI']:
|
|
|
|
return
|
|
|
|
|
|
|
|
for key in ('setDNAString', 'setMailboxContents', 'setGiftSchedule', 'setDeliverySchedule'):
|
2015-07-17 16:48:30 +00:00
|
|
|
fields[key] = fields[key][0].encode('base64')
|
2015-07-17 11:50:16 +00:00
|
|
|
|
|
|
|
newDict = self.copyDict(fields, 'setDNAString', 'setMailboxContents', 'setGiftSchedule', 'setDeliverySchedule', 'setHat', 'setGlasses', 'setBackpack',
|
|
|
|
'setShoes', 'setHatList', 'setGlassesList', 'setBackpackList', 'setShoes', 'setShoesList', 'setCustomMessages', 'setEmoteAccess',
|
2015-07-17 21:39:11 +00:00
|
|
|
'setClothesTopsList', 'setClothesBottomsList', 'setPetTrickPhrases', 'setNametagStyles')
|
2015-07-17 11:50:16 +00:00
|
|
|
|
|
|
|
self.callback(self.avId, self.targetId, newDict, self.optional)
|
|
|
|
del self.phone.fsms[self.avId]
|
|
|
|
|
2015-03-03 22:10:12 +00:00
|
|
|
class DistributedPhoneAI(DistributedFurnitureItemAI):
|
2015-07-12 19:42:55 +00:00
|
|
|
notify = DirectNotifyGlobal.directNotify.newCategory("DistributedPhoneAI")
|
|
|
|
|
|
|
|
def __init__(self, air, furnitureMgr, catalogItem):
|
|
|
|
DistributedFurnitureItemAI.__init__(self, air, furnitureMgr, catalogItem)
|
2015-07-17 11:50:16 +00:00
|
|
|
self.fsms = {}
|
2015-07-16 13:26:34 +00:00
|
|
|
self.initialScale = (0.8, 0.8, 0.8)
|
2015-07-12 19:42:55 +00:00
|
|
|
self.inUse = False
|
|
|
|
self.currAvId = 0
|
|
|
|
|
|
|
|
def calcHouseItems(self, avatar):
|
|
|
|
houseId = avatar.houseId
|
|
|
|
|
|
|
|
if not houseId:
|
|
|
|
self.notify.warning('Avatar %s has no houseId associated.' % avatar.doId)
|
|
|
|
return 0
|
|
|
|
|
|
|
|
house = simbase.air.doId2do.get(houseId)
|
|
|
|
if not house:
|
|
|
|
self.notify.warning('House %s (for avatar %s) not instantiated.' % (houseId, avatar.doId))
|
|
|
|
return 0
|
|
|
|
|
|
|
|
mgr = house.interior.furnitureManager
|
|
|
|
attic = (mgr.atticItems, mgr.atticWallpaper, mgr.atticWindows)
|
|
|
|
numHouseItems = len(CatalogItemList(house.getInteriorItems(), store=CatalogItem.Customization | CatalogItem.Location))
|
|
|
|
numAtticItems = sum(len(x) for x in attic)
|
|
|
|
|
|
|
|
return numHouseItems + numAtticItems
|
|
|
|
|
|
|
|
def setInitialScale(self, scale):
|
|
|
|
self.initialScale = scale
|
|
|
|
|
2015-03-03 22:10:12 +00:00
|
|
|
def getInitialScale(self):
|
2015-07-12 19:42:55 +00:00
|
|
|
return self.initialScale
|
2015-03-03 22:10:12 +00:00
|
|
|
|
|
|
|
def avatarEnter(self):
|
|
|
|
avId = self.air.getAvatarIdFromSender()
|
2015-07-12 19:42:55 +00:00
|
|
|
if self.inUse:
|
|
|
|
self.ejectAvatar(avId)
|
2015-03-03 22:10:12 +00:00
|
|
|
return
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-03-03 22:10:12 +00:00
|
|
|
av = self.air.doId2do.get(avId)
|
2015-07-12 19:42:55 +00:00
|
|
|
if av:
|
|
|
|
self.setInUse(avId)
|
|
|
|
self.sendUpdateToAvatarId(avId, 'setLimits', [self.calcHouseItems(av)])
|
|
|
|
self.d_setMovie(PHONE_MOVIE_PICKUP, avId)
|
|
|
|
av.b_setCatalogNotify(0, av.mailboxNotify)
|
|
|
|
|
2015-07-22 14:49:36 +00:00
|
|
|
self.air.questManager.toonCalledClarabelle(av)
|
|
|
|
|
2015-03-03 22:10:12 +00:00
|
|
|
def avatarExit(self):
|
2015-07-12 19:42:55 +00:00
|
|
|
if not self.inUse:
|
|
|
|
self.notify.warning('Requested avatarExit but phone isn\'t in use!')
|
|
|
|
return
|
2015-03-03 22:10:12 +00:00
|
|
|
avId = self.air.getAvatarIdFromSender()
|
2015-07-12 19:42:55 +00:00
|
|
|
if avId != self.currAvId:
|
|
|
|
self.notify.warning('Requested avatarExit from unknown avatar %s' %avId)
|
2015-03-03 22:10:12 +00:00
|
|
|
return
|
2015-07-12 19:42:55 +00:00
|
|
|
self.d_setMovie(PHONE_MOVIE_HANGUP, avId)
|
|
|
|
taskMgr.doMethodLater(1, self.resetMovie, self.taskName('resetMovie'))
|
|
|
|
self.setFree()
|
|
|
|
|
|
|
|
def setFree(self):
|
|
|
|
self.inUse = False
|
|
|
|
self.currAvId = 0
|
|
|
|
|
|
|
|
def setInUse(self, avId):
|
|
|
|
self.inUse = True
|
|
|
|
self.currAvId = avId
|
|
|
|
|
|
|
|
def d_setMovie(self, movie, avId):
|
|
|
|
self.sendUpdate('setMovie', args=[movie, avId, globalClockDelta.getRealNetworkTime(bits=32)])
|
|
|
|
|
|
|
|
def ejectAvatar(self, avId):
|
|
|
|
self.sendUpdateToAvatarId(avId, 'freeAvatar', [])
|
|
|
|
|
|
|
|
def __getCaller(self):
|
2015-03-03 22:10:12 +00:00
|
|
|
avId = self.air.getAvatarIdFromSender()
|
2015-07-12 19:42:55 +00:00
|
|
|
if avId != self.currAvId:
|
|
|
|
self.air.writeServerEvent('suspicious', avId, 'tried purchasing item, but not using phone')
|
|
|
|
self.notify.warning('%d tried purchasing item, but not using phone' % avId)
|
2015-03-03 22:10:12 +00:00
|
|
|
return
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-03-03 22:10:12 +00:00
|
|
|
av = self.air.doId2do.get(avId)
|
|
|
|
if not av:
|
2015-07-12 19:42:55 +00:00
|
|
|
self.air.writeServerEvent('suspicious', avId, 'tried purchasing item, but not on shard')
|
|
|
|
self.notify.warning('%d tried purchasing item, but not on shard' % avId)
|
2015-03-03 22:10:12 +00:00
|
|
|
return
|
2015-07-12 19:42:55 +00:00
|
|
|
|
|
|
|
return av
|
2015-07-17 11:50:16 +00:00
|
|
|
|
|
|
|
def checkPurchaseLimit(self, recipient, item):
|
2015-07-17 12:29:08 +00:00
|
|
|
if len(recipient.mailboxContents) + len(recipient.onOrder) + len(recipient.onGiftOrder) + 1 >= ToontownGlobals.MaxMailboxContents:
|
2015-07-17 11:50:16 +00:00
|
|
|
return ToontownGlobals.P_MailboxFull
|
|
|
|
elif item.reachedPurchaseLimit(recipient):
|
|
|
|
return ToontownGlobals.P_ReachedPurchaseLimit
|
2015-07-17 12:29:08 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
return ToontownGlobals.P_ItemOnOrder
|
|
|
|
|
|
|
|
def chargeAvatar(self, av, money, emblems):
|
|
|
|
av.takeMoney(money)
|
|
|
|
av.subtractEmblems(emblems)
|
|
|
|
|
|
|
|
def attemptPurchase(self, context, av, blob, optional, gifting=False):
|
|
|
|
avId = av.doId
|
2015-07-12 19:42:55 +00:00
|
|
|
item = CatalogItem.getItem(blob, CatalogItem.Customization)
|
2015-07-17 11:50:16 +00:00
|
|
|
|
2015-07-12 19:42:55 +00:00
|
|
|
if isinstance(item, CatalogInvalidItem.CatalogInvalidItem):
|
|
|
|
self.air.writeServerEvent('suspicious', avId, 'tried purchasing invalid item')
|
|
|
|
self.notify.warning('%d tried purchasing invalid item' % avId)
|
2015-07-23 20:00:50 +00:00
|
|
|
return ToontownGlobals.P_NotInCatalog
|
2015-07-17 11:50:16 +00:00
|
|
|
elif (not item.hasEmblemPrices()) and item not in av.backCatalog and item not in av.weeklyCatalog and item not in av.monthlyCatalog:
|
2015-07-12 19:42:55 +00:00
|
|
|
self.air.writeServerEvent('suspicious', avId, 'tried purchasing non-existing item')
|
|
|
|
self.notify.warning('%d tried purchasing non-existing item' % avId)
|
|
|
|
return ToontownGlobals.P_NotInCatalog
|
2015-07-17 11:50:16 +00:00
|
|
|
|
|
|
|
if gifting and not item.isGift():
|
|
|
|
return ToontownGlobals.P_NotAGift
|
|
|
|
|
|
|
|
price = item.getPrice(CatalogItem.CatalogTypeBackorder if item in av.backCatalog else 0)
|
|
|
|
|
|
|
|
if price > av.getTotalMoney() or (item.hasEmblemPrices() and not av.isEnoughEmblemsToBuy(item.getEmblemPrices())):
|
|
|
|
return ToontownGlobals.P_NotEnoughMoney
|
|
|
|
|
|
|
|
if item.getDeliveryTime() or gifting:
|
|
|
|
deliveryTime = 0 if config.GetBool('want-instant-delivery', False) else item.getDeliveryTime()
|
|
|
|
item.deliveryDate = int(time.time() / 60. + deliveryTime + .5)
|
|
|
|
|
|
|
|
if gifting:
|
|
|
|
return self.requestGiftAvatarOperation(avId, gifting, [context, item, price], self.attemptGiftPurchase)
|
|
|
|
else:
|
|
|
|
returnCode = self.checkPurchaseLimit(av, item)
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
if returnCode != ToontownGlobals.P_ItemOnOrder:
|
|
|
|
return returnCode
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
if item.getDeliveryTime():
|
|
|
|
self.chargeAvatar(av, price, item.getEmblemPrices())
|
|
|
|
av.onOrder.append(item)
|
|
|
|
av.b_setDeliverySchedule(av.onOrder)
|
|
|
|
else:
|
|
|
|
returnCode = item.recordPurchase(av, optional)
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
if returnCode == ToontownGlobals.P_ItemAvailable:
|
|
|
|
self.chargeAvatar(av, price, item.getEmblemPrices())
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
return returnCode
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
def attemptGiftPurchase(self, avId, targetId, avatar, optional):
|
|
|
|
av = self.air.doId2do.get(avId)
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
if not av:
|
|
|
|
return
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
recipient = GiftAvatar.createFromFields(avatar)
|
|
|
|
context = optional[0]
|
|
|
|
item = optional[1]
|
|
|
|
returnCode = self.checkPurchaseLimit(recipient, item)
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
if returnCode != ToontownGlobals.P_ItemOnOrder:
|
|
|
|
self.sendGiftPurchaseResponse(context, avId, returnCode)
|
|
|
|
return
|
|
|
|
|
|
|
|
item.giftTag = avId
|
|
|
|
self.chargeAvatar(av, optional[2], item.getEmblemPrices())
|
|
|
|
recipient.onGiftOrder.append(item)
|
|
|
|
|
|
|
|
dg = self.air.dclassesByName['DistributedToonAI'].aiFormatUpdate('setGiftSchedule', targetId, targetId, self.air.ourChannel, [recipient.getGiftScheduleBlob()])
|
|
|
|
self.air.send(dg)
|
|
|
|
self.sendGiftPurchaseResponse(context, avId, ToontownGlobals.P_ItemOnOrder)
|
|
|
|
|
|
|
|
def sendGiftPurchaseResponse(self, context, avId, returnCode):
|
|
|
|
if returnCode in (ToontownGlobals.P_ItemOnOrder, ToontownGlobals.P_ItemAvailable):
|
|
|
|
messenger.send('topToonsManager-event', [avId, TopToonsGlobals.CAT_CATALOG | TopToonsGlobals.CAT_GIFTS, 1])
|
|
|
|
|
|
|
|
self.sendUpdateToAvatarId(avId, 'requestGiftPurchaseResponse', [context, returnCode])
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
def requestPurchaseMessage(self, context, blob, optional):
|
2015-07-12 19:42:55 +00:00
|
|
|
av = self.__getCaller()
|
2015-07-17 11:50:16 +00:00
|
|
|
|
|
|
|
if not av:
|
|
|
|
return
|
|
|
|
|
|
|
|
returnCode = self.attemptPurchase(context, av, blob, optional)
|
2015-07-12 19:42:55 +00:00
|
|
|
|
2015-07-17 11:50:16 +00:00
|
|
|
if returnCode in (ToontownGlobals.P_ItemOnOrder, ToontownGlobals.P_ItemAvailable):
|
|
|
|
messenger.send('topToonsManager-event', [av.doId, TopToonsGlobals.CAT_CATALOG, 1])
|
|
|
|
|
|
|
|
self.sendUpdateToAvatarId(av.doId, 'requestPurchaseResponse', [context, returnCode])
|
|
|
|
|
|
|
|
def requestGiftPurchaseMessage(self, context, targetId, blob, optional):
|
2015-07-12 19:42:55 +00:00
|
|
|
av = self.__getCaller()
|
2015-07-17 11:50:16 +00:00
|
|
|
|
|
|
|
if not av:
|
|
|
|
return
|
|
|
|
|
|
|
|
returnCode = self.attemptPurchase(context, av, blob, optional, gifting=targetId)
|
|
|
|
|
|
|
|
if returnCode:
|
|
|
|
self.sendGiftPurchaseResponse(context, av.doId, returnCode)
|
|
|
|
|
|
|
|
def requestGiftAvatar(self, doId):
|
|
|
|
self.requestGiftAvatarOperation(self.air.getAvatarIdFromSender(), doId, None, self.sendGiftAvatarResponse)
|
|
|
|
|
|
|
|
def requestGiftAvatarOperation(self, avId, doId, optional, callback):
|
|
|
|
if avId in self.fsms:
|
2015-07-17 12:54:12 +00:00
|
|
|
return ToontownGlobals.P_TooFast
|
2015-07-17 11:50:16 +00:00
|
|
|
|
|
|
|
loadOperation = LoadGiftAvatar(self, avId, doId, optional, callback)
|
|
|
|
loadOperation.start()
|
|
|
|
self.fsms[avId] = loadOperation
|
|
|
|
return None
|
|
|
|
|
|
|
|
def sendGiftAvatarResponse(self, avId, targetId, avatar, optional):
|
|
|
|
self.sendUpdateToAvatarId(avId, 'setGiftAvatar', [json.dumps(avatar)])
|
2015-07-12 19:42:55 +00:00
|
|
|
|
|
|
|
def resetMovie(self, task):
|
|
|
|
self.d_setMovie(PHONE_MOVIE_CLEAR, 0)
|
|
|
|
return task.done
|