1
0
mirror of synced 2025-01-22 03:24:05 +01:00
bemaniutils/bemani/backend/popn/tunestreet.py

622 lines
28 KiB
Python
Raw Normal View History

# vim: set fileencoding=utf-8
from typing import Dict, Any
from typing_extensions import Final
from bemani.backend.popn.base import PopnMusicBase
from bemani.backend.popn.stubs import PopnMusicSengokuRetsuden
from bemani.backend.base import Status
from bemani.common import Profile, VersionConstants
from bemani.data import Score, UserID
from bemani.protocol import Node
class PopnMusicTuneStreet(PopnMusicBase):
name: str = "Pop'n Music TUNE STREET"
version: int = VersionConstants.POPN_MUSIC_TUNE_STREET
# Play modes, as reported by profile save from the game
GAME_PLAY_MODE_CHALLENGE: Final[int] = 3
GAME_PLAY_MODE_CHO_CHALLENGE: Final[int] = 4
GAME_PLAY_MODE_TOWN_CHO_CHALLENGE: Final[int] = 15
# Play flags, as saved into/loaded from the DB
GAME_PLAY_FLAG_FAILED: Final[int] = 0
GAME_PLAY_FLAG_CLEARED: Final[int] = 1
GAME_PLAY_FLAG_FULL_COMBO: Final[int] = 2
GAME_PLAY_FLAG_PERFECT_COMBO: Final[int] = 3
# Chart type, as reported by profile save from the game
GAME_CHART_TYPE_NORMAL: Final[int] = 0
GAME_CHART_TYPE_HYPER: Final[int] = 1
GAME_CHART_TYPE_5_BUTTON: Final[int] = 2
GAME_CHART_TYPE_EX: Final[int] = 3
GAME_CHART_TYPE_BATTLE_NORMAL: Final[int] = 4
GAME_CHART_TYPE_BATTLE_HYPER: Final[int] = 5
GAME_CHART_TYPE_ENJOY_5_BUTTON: Final[int] = 6
GAME_CHART_TYPE_ENJOY_9_BUTTON: Final[int] = 7
# Extra chart types supported by Pop'n 19
CHART_TYPE_OLD_NORMAL: Final[int] = 4
CHART_TYPE_OLD_HYPER: Final[int] = 5
CHART_TYPE_OLD_EX: Final[int] = 6
CHART_TYPE_ENJOY_5_BUTTON: Final[int] = 7
CHART_TYPE_ENJOY_9_BUTTON: Final[int] = 8
CHART_TYPE_5_BUTTON: Final[int] = 9
# Chart type, as packed into a hiscore binary
GAME_CHART_TYPE_5_BUTTON_POSITION: Final[int] = 0
GAME_CHART_TYPE_NORMAL_POSITION: Final[int] = 1
GAME_CHART_TYPE_HYPER_POSITION: Final[int] = 2
GAME_CHART_TYPE_EX_POSITION: Final[int] = 3
GAME_CHART_TYPE_CHO_NORMAL_POSITION: Final[int] = 4
GAME_CHART_TYPE_CHO_HYPER_POSITION: Final[int] = 5
GAME_CHART_TYPE_CHO_EX_POSITION: Final[int] = 6
# Highest song ID we can represent
GAME_MAX_MUSIC_ID: Final[int] = 1045
def previous_version(self) -> PopnMusicBase:
return PopnMusicSengokuRetsuden(self.data, self.config, self.model)
@classmethod
def get_settings(cls) -> Dict[str, Any]:
"""
Return all of our front-end modifiably settings.
"""
return {
'ints': [
{
'name': 'Game Phase',
'tip': 'Game unlock phase for all players.',
'category': 'game_config',
'setting': 'game_phase',
'values': {
0: 'NO PHASE',
1: 'SECRET DATA RELEASE',
2: 'MAX: ALL DATA RELEASE',
}
},
{
'name': 'Town Mode Phase',
'tip': 'Town mode phase for all players.',
'category': 'game_config',
'setting': 'town_phase',
'values': {
0: 'town mode disabled',
1: 'town phase 1',
2: 'town phase 2',
3: 'Pop\'n Naan Festival',
# 4 seems to be a continuation of town phase 2. Intentionally leaving it out.
5: 'town phase 3',
6: 'town phase 4',
7: 'Miracle 4 + 1',
# 8 seems to be a continuation of town phase 4. Intentionally leaving it out.
9: 'town phase MAX',
10: 'Find your daughter!',
# 11 is a continuation of phase MAX after find your daughter, with Tanabata
# bamboo grass added as well.
11: 'town phase MAX+1',
12: 'Peruri-san visits',
# 13 is a continuation of phase MAX+1 after peruri-san visits, with Watermelon
# pattern tank added as well.
13: 'town phase MAX+2',
14: 'Find Deuil!',
# 15 is a continuation of phase MAX+2 after find deuil, with Tsukimi dumplings
# added as well.
15: 'town phase MAX+3',
16: 'Landmark stamp rally',
# 17 is a continuation of MAX+3 after landmark stamp rally ends, but offering
# no additional stuff.
}
},
],
'bools': [
{
'name': 'Force Song Unlock',
'tip': 'Force unlock all songs.',
'category': 'game_config',
'setting': 'force_unlock_songs',
},
{
'name': 'Force Customization Unlock',
'tip': 'Force unlock all theme and menu customizations.',
'category': 'game_config',
'setting': 'force_unlock_customizations',
},
],
}
def __format_flags_for_score(self, score: Score) -> int:
# Format song flags (cleared/not, combo flags)
playedflag = {
self.CHART_TYPE_5_BUTTON: 0x2000,
self.CHART_TYPE_OLD_NORMAL: 0x0800,
self.CHART_TYPE_OLD_HYPER: 0x1000,
self.CHART_TYPE_OLD_EX: 0x4000,
self.CHART_TYPE_NORMAL: 0x0800,
self.CHART_TYPE_HYPER: 0x1000,
self.CHART_TYPE_EX: 0x4000,
# We don't have a played flag for these, only cleared/no play
self.CHART_TYPE_ENJOY_5_BUTTON: 0,
self.CHART_TYPE_ENJOY_9_BUTTON: 0,
}[score.chart]
# Shift value for cleared/failed/combo indicators
shift = {
self.CHART_TYPE_5_BUTTON: 4,
self.CHART_TYPE_OLD_NORMAL: 0,
self.CHART_TYPE_OLD_HYPER: 2,
self.CHART_TYPE_OLD_EX: 6,
self.CHART_TYPE_NORMAL: 0,
self.CHART_TYPE_HYPER: 2,
self.CHART_TYPE_EX: 6,
self.CHART_TYPE_ENJOY_5_BUTTON: 9,
self.CHART_TYPE_ENJOY_9_BUTTON: 8,
}[score.chart]
flags = {
self.PLAY_MEDAL_CIRCLE_FAILED: self.GAME_PLAY_FLAG_FAILED,
self.PLAY_MEDAL_DIAMOND_FAILED: self.GAME_PLAY_FLAG_FAILED,
self.PLAY_MEDAL_STAR_FAILED: self.GAME_PLAY_FLAG_FAILED,
self.PLAY_MEDAL_EASY_CLEAR: self.GAME_PLAY_FLAG_CLEARED,
self.PLAY_MEDAL_CIRCLE_CLEARED: self.GAME_PLAY_FLAG_CLEARED,
self.PLAY_MEDAL_DIAMOND_CLEARED: self.GAME_PLAY_FLAG_CLEARED,
self.PLAY_MEDAL_STAR_CLEARED: self.GAME_PLAY_FLAG_CLEARED,
self.PLAY_MEDAL_CIRCLE_FULL_COMBO: self.GAME_PLAY_FLAG_FULL_COMBO,
self.PLAY_MEDAL_DIAMOND_FULL_COMBO: self.GAME_PLAY_FLAG_FULL_COMBO,
self.PLAY_MEDAL_STAR_FULL_COMBO: self.GAME_PLAY_FLAG_FULL_COMBO,
self.PLAY_MEDAL_PERFECT: self.GAME_PLAY_FLAG_PERFECT_COMBO,
}[score.data.get_int('medal')]
return (flags << shift) | playedflag
def format_profile(self, userid: UserID, profile: Profile) -> Node:
root = Node.void('playerdata')
# Format profile
binary_profile = [0] * 2198
# Copy name. We intentionally leave location 12 alone as it is
# the null termination for the name if it happens to be 12
# characters (6 shift-jis kana).
name_binary = profile.get_str('name', 'なし').encode('shift-jis')[0:12]
for name_pos, byte in enumerate(name_binary):
binary_profile[name_pos] = byte
# Copy game mode. Modes sent to the game are as follows.
# 0 - Enjoy mode.
# 1 - Challenge mode.
# 2 - Battle mode.
# 3 - Net ranking mode (enabled by setting netvs_phase in game.get).
# 4 - Cho challenge mode.
# 5 - Town mode (enabled by event_phase in game.get).
binary_profile[13] = {
0: 0,
1: 0,
2: 1,
3: 1,
4: 4,
5: 2,
13: 5,
14: 5,
15: 5,
}[profile.get_int('play_mode')]
# Copy miscelaneous values
binary_profile[15] = profile.get_int('last_play_flag') & 0xFF
binary_profile[16] = profile.get_int('medal_and_friend') & 0xFF
binary_profile[37] = profile.get_int('read_news') & 0xFF
binary_profile[38] = profile.get_int('skin_tex_note') & 0xFF
binary_profile[39] = profile.get_int('skin_tex_cmn') & 0xFF
binary_profile[40] = profile.get_int('skin_sd_bgm') & 0xFF
binary_profile[41] = profile.get_int('skin_sd_se') & 0xFF
binary_profile[44] = profile.get_int('option') & 0xFF
binary_profile[45] = (profile.get_int('option') >> 8) & 0xFF
binary_profile[46] = (profile.get_int('option') >> 16) & 0xFF
binary_profile[47] = (profile.get_int('option') >> 24) & 0xFF
binary_profile[48] = profile.get_int('jubeat_collabo') & 0xFF
binary_profile[49] = (profile.get_int('jubeat_collabo') >> 8) & 0xFF
# 52-56 and 56-60 make up two 32 bit colors found in color_3p_flag.
binary_profile[60] = profile.get_int('chara', -1) & 0xFF
binary_profile[61] = (profile.get_int('chara', -1) >> 8) & 0xFF
binary_profile[62] = profile.get_int('music') & 0xFF
binary_profile[63] = (profile.get_int('music') >> 8) & 0xFF
binary_profile[64] = profile.get_int('sheet') & 0xFF
binary_profile[65] = profile.get_int('category') & 0xFF
binary_profile[66] = profile.get_int('norma_point') & 0xFF
binary_profile[67] = (profile.get_int('norma_point') >> 8) & 0xFF
# Format Scores
hiscore_array = [0] * int((((self.GAME_MAX_MUSIC_ID * 7) * 17) + 7) / 8)
scores = self.data.remote.music.get_scores(self.game, self.version, userid)
for score in scores:
if score.id > self.GAME_MAX_MUSIC_ID:
continue
# Skip any scores for chart types we don't support
if score.chart in [
self.CHART_TYPE_EASY,
]:
continue
if score.data.get_int('medal') == self.PLAY_MEDAL_NO_PLAY:
continue
flags = self.__format_flags_for_score(score)
flags_index = score.id * 2
binary_profile[108 + flags_index] = binary_profile[108 + flags_index] | (flags & 0xFF)
binary_profile[109 + flags_index] = binary_profile[109 + flags_index] | ((flags >> 8) & 0xFF)
if score.chart in [
self.CHART_TYPE_ENJOY_5_BUTTON,
self.CHART_TYPE_ENJOY_9_BUTTON,
]:
# We don't return enjoy scores, just the flags that we played them
continue
# Format actual score, according to DB chart position
hiscore_index = (score.id * 7) + {
self.CHART_TYPE_5_BUTTON: self.GAME_CHART_TYPE_5_BUTTON_POSITION,
self.CHART_TYPE_OLD_NORMAL: self.GAME_CHART_TYPE_NORMAL_POSITION,
self.CHART_TYPE_OLD_HYPER: self.GAME_CHART_TYPE_HYPER_POSITION,
self.CHART_TYPE_OLD_EX: self.GAME_CHART_TYPE_EX_POSITION,
self.CHART_TYPE_NORMAL: self.GAME_CHART_TYPE_CHO_NORMAL_POSITION,
self.CHART_TYPE_HYPER: self.GAME_CHART_TYPE_CHO_HYPER_POSITION,
self.CHART_TYPE_EX: self.GAME_CHART_TYPE_CHO_EX_POSITION,
}[score.chart]
hiscore_byte_pos = int((hiscore_index * 17) / 8)
hiscore_bit_pos = int((hiscore_index * 17) % 8)
hiscore_value = score.points << hiscore_bit_pos
hiscore_array[hiscore_byte_pos] = hiscore_array[hiscore_byte_pos] | (hiscore_value & 0xFF)
hiscore_array[hiscore_byte_pos + 1] = hiscore_array[hiscore_byte_pos + 1] | ((hiscore_value >> 8) & 0xFF)
hiscore_array[hiscore_byte_pos + 2] = hiscore_array[hiscore_byte_pos + 2] | ((hiscore_value >> 16) & 0xFF)
# Format most played
most_played = [x[0] for x in self.data.local.music.get_most_played(self.game, self.version, userid, 20)]
while len(most_played) < 20:
most_played.append(-1)
profile_pos = 68
for musicid in most_played:
binary_profile[profile_pos] = musicid & 0xFF
binary_profile[profile_pos + 1] = (musicid >> 8) & 0xFF
profile_pos = profile_pos + 2
# Town purchases, including BGM/announcer changes and such.
# The town customization area will show up if the player owns
# one or more customization in any of the following four
# purchase locations. These are all purchased in town mode.
# - 4-7 are song unlock flags.
# - 8 appears to be purchased pop-kuns.
# - 9 appears to be purchased themes.
# - 10 appears to be purchased BGMs.
# - 11 appears to be purchased sound effects.
binary_town = [0] * 141
town = profile.get_dict('town')
# Last play flag, so the selection for 5/9/9+cool sticks.
binary_town[140] = town.get_int('play_type')
# Fill in basic town points, tracked here and returned in basic profile for some reason.
binary_town[0] = town.get_int('points') & 0xFF
binary_town[1] = (town.get_int('points') >> 8) & 0xFF
binary_town[2] = (town.get_int('points') >> 16) & 0xFF
binary_town[3] = (town.get_int('points') >> 24) & 0xFF
# Fill in purchase flags (this is for stuff like BGMs, SEs, Pop-kun customizations, etc).
bought_flg = town.get_int_array('bought_flg', 3)
game_config = self.get_game_config()
force_unlock_songs = game_config.get_bool('force_unlock_songs')
force_unlock_customizations = game_config.get_bool('force_unlock_customizations')
if force_unlock_songs:
bought_flg[0] = 0xFFFFFFFF
if force_unlock_customizations:
bought_flg[1] = 0xFFFFFFFF
for flg, off in enumerate([4, 8, 12]):
binary_town[off + 0] = bought_flg[flg] & 0xFF
binary_town[off + 1] = (bought_flg[flg] >> 8) & 0xFF
binary_town[off + 2] = (bought_flg[flg] >> 16) & 0xFF
binary_town[off + 3] = (bought_flg[flg] >> 24) & 0xFF
# Fill in build flags (presumably for what parcels of land have been bought and built on).
build_flg = town.get_int_array('build_flg', 8)
for flg, off in enumerate([16, 20, 24, 28, 32, 36, 40, 44]):
binary_town[off + 0] = build_flg[flg] & 0xFF
binary_town[off + 1] = (build_flg[flg] >> 8) & 0xFF
binary_town[off + 2] = (build_flg[flg] >> 16) & 0xFF
binary_town[off + 3] = (build_flg[flg] >> 24) & 0xFF
# Fill in character flags (presumably for character location, orientation, stats, etc).
chara_flg = town.get_int_array('chara_flg', 19)
for flg, off in enumerate([48, 52, 56, 60, 64, 68, 72, 76, 80, 84, 88, 92, 96, 100, 104, 108, 112, 116, 120]):
binary_town[off + 0] = chara_flg[flg] & 0xFF
binary_town[off + 1] = (chara_flg[flg] >> 8) & 0xFF
binary_town[off + 2] = (chara_flg[flg] >> 16) & 0xFF
binary_town[off + 3] = (chara_flg[flg] >> 24) & 0xFF
# Fill in miscellaneous event flags.
event_flg = town.get_int_array('event_flg', 4)
for flg, off in enumerate([124, 128, 132, 136]):
binary_town[off + 0] = event_flg[flg] & 0xFF
binary_town[off + 1] = (event_flg[flg] >> 8) & 0xFF
binary_town[off + 2] = (event_flg[flg] >> 16) & 0xFF
binary_town[off + 3] = (event_flg[flg] >> 24) & 0xFF
# Construct final profile
root.add_child(Node.binary('b', bytes(binary_profile)))
root.add_child(Node.binary('hiscore', bytes(hiscore_array)))
root.add_child(Node.binary('town', bytes(binary_town)))
return root
def unformat_profile(self, userid: UserID, request: Node, oldprofile: Profile) -> Profile:
newprofile = oldprofile.clone()
# Extract the playmode, important for scores later
playmode = int(request.attribute('play_mode'))
newprofile.replace_int('play_mode', playmode)
# Extract profile options
newprofile.replace_int('chara', int(request.attribute('chara_num')))
if 'option' in request.attributes:
newprofile.replace_int('option', int(request.attribute('option')))
if 'last_play_flag' in request.attributes:
newprofile.replace_int('last_play_flag', int(request.attribute('last_play_flag')))
if 'medal_and_friend' in request.attributes:
newprofile.replace_int('medal_and_friend', int(request.attribute('medal_and_friend')))
if 'music_num' in request.attributes:
newprofile.replace_int('music', int(request.attribute('music_num')))
if 'sheet_num' in request.attributes:
newprofile.replace_int('sheet', int(request.attribute('sheet_num')))
if 'category_num' in request.attributes:
newprofile.replace_int('category', int(request.attribute('category_num')))
if 'read_news_no_max' in request.attributes:
newprofile.replace_int('read_news', int(request.attribute('read_news_no_max')))
if 'jubeat_collabo' in request.attributes:
newprofile.replace_int('jubeat_collabo', int(request.attribute('jubeat_collabo')))
if 'norma_point' in request.attributes:
newprofile.replace_int('norma_point', int(request.attribute('norma_point')))
if 'skin_tex_note' in request.attributes:
newprofile.replace_int('skin_tex_note', int(request.attribute('skin_tex_note')))
if 'skin_tex_cmn' in request.attributes:
newprofile.replace_int('skin_tex_cmn', int(request.attribute('skin_tex_cmn')))
if 'skin_sd_bgm' in request.attributes:
newprofile.replace_int('skin_sd_bgm', int(request.attribute('skin_sd_bgm')))
if 'skin_sd_se' in request.attributes:
newprofile.replace_int('skin_sd_se', int(request.attribute('skin_sd_se')))
# Keep track of play statistics
self.update_play_statistics(userid)
# Extract scores
for node in request.children:
if node.name == 'music':
songid = int(node.attribute('music_num'))
chart = int(node.attribute('sheet_num'))
points = int(node.attribute('score'))
data = int(node.attribute('data'))
# We never save battle scores
if chart in [
self.GAME_CHART_TYPE_BATTLE_NORMAL,
self.GAME_CHART_TYPE_BATTLE_HYPER,
]:
continue
# Arrange order to be compatible with future mixes
if playmode in {self.GAME_PLAY_MODE_CHO_CHALLENGE, self.GAME_PLAY_MODE_TOWN_CHO_CHALLENGE}:
if chart in [
self.GAME_CHART_TYPE_5_BUTTON,
self.GAME_CHART_TYPE_ENJOY_5_BUTTON,
self.GAME_CHART_TYPE_ENJOY_9_BUTTON,
]:
# We don't save 5 button for cho scores, or enjoy modes
continue
chart = {
self.GAME_CHART_TYPE_NORMAL: self.CHART_TYPE_NORMAL,
self.GAME_CHART_TYPE_HYPER: self.CHART_TYPE_HYPER,
self.GAME_CHART_TYPE_EX: self.CHART_TYPE_EX,
}[chart]
else:
chart = {
self.GAME_CHART_TYPE_NORMAL: self.CHART_TYPE_OLD_NORMAL,
self.GAME_CHART_TYPE_HYPER: self.CHART_TYPE_OLD_HYPER,
self.GAME_CHART_TYPE_5_BUTTON: self.CHART_TYPE_5_BUTTON,
self.GAME_CHART_TYPE_EX: self.CHART_TYPE_OLD_EX,
self.GAME_CHART_TYPE_ENJOY_5_BUTTON: self.CHART_TYPE_ENJOY_5_BUTTON,
self.GAME_CHART_TYPE_ENJOY_9_BUTTON: self.CHART_TYPE_ENJOY_9_BUTTON,
}[chart]
# Extract play flags
shift = {
self.CHART_TYPE_5_BUTTON: 4,
self.CHART_TYPE_OLD_NORMAL: 0,
self.CHART_TYPE_OLD_HYPER: 2,
self.CHART_TYPE_OLD_EX: 6,
self.CHART_TYPE_NORMAL: 0,
self.CHART_TYPE_HYPER: 2,
self.CHART_TYPE_EX: 6,
self.CHART_TYPE_ENJOY_5_BUTTON: 9,
self.CHART_TYPE_ENJOY_9_BUTTON: 8,
}[chart]
if chart in [
self.CHART_TYPE_ENJOY_5_BUTTON,
self.CHART_TYPE_ENJOY_9_BUTTON,
]:
# We only store cleared or not played for enjoy mode
mask = 0x1
else:
# We store all data for regular charts
mask = 0x3
# Grab flags, map to medals in DB. Choose lowest one for each so
# a newer pop'n can still improve scores and medals.
flags = (data >> shift) & mask
medal = {
self.GAME_PLAY_FLAG_FAILED: self.PLAY_MEDAL_CIRCLE_FAILED,
self.GAME_PLAY_FLAG_CLEARED: self.PLAY_MEDAL_CIRCLE_CLEARED,
self.GAME_PLAY_FLAG_FULL_COMBO: self.PLAY_MEDAL_CIRCLE_FULL_COMBO,
self.GAME_PLAY_FLAG_PERFECT_COMBO: self.PLAY_MEDAL_PERFECT,
}[flags]
self.update_score(userid, songid, chart, points, medal)
# Update town mode data.
town = newprofile.get_dict('town')
# Basic stuff that's in the base node for no reason?
if 'tp' in request.attributes:
town.replace_int('points', int(request.attribute('tp')))
# Stuff that is in the town node
townnode = request.child('town')
if townnode is not None:
if 'play_type' in townnode.attributes:
town.replace_int('play_type', int(townnode.attribute('play_type')))
if 'base' in townnode.attributes:
town.replace_int_array('base', 4, [int(x) for x in townnode.attribute('base').split(',')])
if 'bought_flg' in townnode.attributes:
bought_array = [int(x) for x in townnode.attribute('bought_flg').split(',')]
if len(bought_array) == 3:
game_config = self.get_game_config()
force_unlock_songs = game_config.get_bool('force_unlock_songs')
force_unlock_customizations = game_config.get_bool('force_unlock_customizations')
old_bought_array = town.get_int_array('bought_flg', 3)
if force_unlock_songs:
# Don't save force unlocked flags, it'll clobber the profile.
bought_array[0] = old_bought_array[0]
if force_unlock_customizations:
# Don't save force unlocked flags, it'll clobber the profile.
bought_array[1] = old_bought_array[1]
town.replace_int_array('bought_flg', 3, bought_array)
if 'build_flg' in townnode.attributes:
town.replace_int_array('build_flg', 8, [int(x) for x in townnode.attribute('build_flg').split(',')])
if 'chara_flg' in townnode.attributes:
town.replace_int_array('chara_flg', 19, [int(x) for x in townnode.attribute('chara_flg').split(',')])
if 'event_flg' in townnode.attributes:
town.replace_int_array('event_flg', 4, [int(x) for x in townnode.attribute('event_flg').split(',')])
for bid in range(8):
if f'building_{bid}' in townnode.attributes:
town.replace_int_array(f'building_{bid}', 8, [int(x) for x in townnode.attribute(f'building_{bid}').split(',')])
newprofile.replace_dict('town', town)
return newprofile
def handle_game_get_request(self, request: Node) -> Node:
game_config = self.get_game_config()
game_phase = game_config.get_int('game_phase')
town_phase = game_config.get_int('town_phase')
root = Node.void('game')
root.set_attribute('game_phase', str(game_phase)) # Phase unlocks, for song availability.
root.set_attribute('boss_battle_point', '1')
root.set_attribute('boss_diff', '100,100,100,100,100,100,100,100,100,100')
root.set_attribute('card_phase', '3')
root.set_attribute('event_phase', str(town_phase)) # Town mode, for the main event.
root.set_attribute('gfdm_phase', '2')
root.set_attribute('ir_phase', '14')
root.set_attribute('jubeat_phase', '2')
root.set_attribute('local_matching_enable', '1')
root.set_attribute('matching_sec', '120')
root.set_attribute('netvs_phase', '0') # Net taisen mode phase, maximum 18 (no lobby support).
return root
def handle_game_active_request(self, request: Node) -> Node:
# Update the name of this cab for admin purposes
self.update_machine_name(request.attribute('shop_name'))
return Node.void('game')
def handle_playerdata_expire_request(self, request: Node) -> Node:
return Node.void('playerdata')
def handle_playerdata_logout_request(self, request: Node) -> Node:
return Node.void('playerdata')
def handle_playerdata_get_request(self, request: Node) -> Node:
modelstring = request.attribute('model')
refid = request.attribute('ref_id')
root = self.get_profile_by_refid(
refid,
self.NEW_PROFILE_ONLY if modelstring is None else self.OLD_PROFILE_ONLY,
)
if root is None:
root = Node.void('playerdata')
root.set_attribute('status', str(Status.NO_PROFILE))
return root
def handle_playerdata_town_request(self, request: Node) -> Node:
refid = request.attribute('ref_id')
root = Node.void('playerdata')
userid = self.data.remote.user.from_refid(self.game, self.version, refid)
if userid is None:
return root
profile = self.get_profile(userid)
if profile is None:
return root
town = profile.get_dict('town')
residence = Node.void('residence')
root.add_child(residence)
residence.set_attribute('id', str(town.get_int('residence')))
# It appears there can be up to 9 map nodes, not sure why. I'm only returning the
# first one. Perhaps if there's multiple towns, the residence ID lets you choose
# between them? Maybe it has to do with friends towns?
mapdata = [0] * 180
# Map over progress for base and buildings. Positions 173-176 are for base flags.
base = town.get_int_array('base', 4)
for i in range(4):
mapdata[173 + i] = base[i]
# Positions 42-105 are for building flags.
for bid, start in enumerate([42, 50, 58, 66, 74, 82, 90, 98]):
building = town.get_int_array(f'building_{bid}', 8)
for i in range(8):
mapdata[start + i] = building[i]
mapnode = Node.binary('map', bytes(mapdata))
root.add_child(mapnode)
mapnode.set_attribute('residence', '0')
return root
def handle_playerdata_new_request(self, request: Node) -> Node:
refid = request.attribute('ref_id')
name = request.attribute('name')
root = self.new_profile_by_refid(refid, name)
if root is None:
root = Node.void('playerdata')
root.set_attribute('status', str(Status.NO_PROFILE))
return root
def handle_playerdata_set_request(self, request: Node) -> Node:
refid = request.attribute('ref_id')
root = Node.void('playerdata')
if refid is None:
return root
userid = self.data.remote.user.from_refid(self.game, self.version, refid)
if userid is None:
return root
oldprofile = self.get_profile(userid) or Profile(self.game, self.version, refid, 0)
newprofile = self.unformat_profile(userid, request, oldprofile)
if newprofile is not None:
self.put_profile(userid, newprofile)
return root
def handle_lobby_request(self, request: Node) -> Node:
# Stub out the entire lobby service
return Node.void('lobby')