# vim: set fileencoding=utf-8 import re from typing import Any, Dict from flask import Blueprint, request, Response, url_for, abort from bemani.common import GameConstants from bemani.data import UserID from bemani.frontend.app import loginrequired, jsonify, render_react from bemani.frontend.danevo.danevo import DanceEvolutionFrontend from bemani.frontend.templates import templates_location from bemani.frontend.static import static_location from bemani.frontend.types import g danevo_pages = Blueprint( "danevo_pages", __name__, url_prefix=f"/{GameConstants.DANCE_EVOLUTION.value}", template_folder=templates_location, static_folder=static_location, ) @danevo_pages.route("/scores") @loginrequired def viewnetworkscores() -> Response: # Only load the last 100 results for the initial fetch, so we can render faster frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) network_scores = frontend.get_network_scores(limit=100) if len(network_scores["attempts"]) > 10: network_scores["attempts"] = frontend.round_to_ten(network_scores["attempts"]) return render_react( "Global Dance Evolution Scores", "danevo/scores.react.js", { "attempts": network_scores["attempts"], "songs": frontend.get_all_songs(), "players": network_scores["players"], "versions": {version: name for (game, version, name) in frontend.all_games()}, "shownames": True, "shownewrecords": False, }, { "refresh": url_for("danevo_pages.listnetworkscores"), "player": url_for("danevo_pages.viewplayer", userid=-1), "individual_score": url_for("danevo_pages.viewtopscores", musicid=-1), }, ) @danevo_pages.route("/scores/list") @jsonify @loginrequired def listnetworkscores() -> Dict[str, Any]: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) return frontend.get_network_scores() @danevo_pages.route("/scores/") @loginrequired def viewscores(userid: UserID) -> Response: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) info = frontend.get_latest_player_info([userid]).get(userid) if info is None: abort(404) scores = frontend.get_scores(userid, limit=100) if len(scores) > 10: scores = frontend.round_to_ten(scores) return render_react( f'{info["name"]}\'s Dance Evolution Scores', "danevo/scores.react.js", { "attempts": scores, "songs": frontend.get_all_songs(), "players": {}, "versions": {version: name for (game, version, name) in frontend.all_games()}, "shownames": False, "shownewrecords": True, }, { "refresh": url_for("danevo_pages.listscores", userid=userid), "player": url_for("danevo_pages.viewplayer", userid=-1), "individual_score": url_for("danevo_pages.viewtopscores", musicid=-1), }, ) @danevo_pages.route("/scores//list") @jsonify @loginrequired def listscores(userid: UserID) -> Dict[str, Any]: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) return { "attempts": frontend.get_scores(userid), "players": {}, } @danevo_pages.route("/records") @loginrequired def viewnetworkrecords() -> Response: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) network_records = frontend.get_network_records() versions = {version: name for (game, version, name) in frontend.all_games()} return render_react( "Global Dance Evolution Records", "danevo/records.react.js", { "records": network_records["records"], "songs": frontend.get_all_songs(), "players": network_records["players"], "versions": versions, "shownames": True, "showpersonalsort": False, "filterempty": False, }, { "refresh": url_for("danevo_pages.listnetworkrecords"), "player": url_for("danevo_pages.viewplayer", userid=-1), "individual_score": url_for("danevo_pages.viewtopscores", musicid=-1), }, ) @danevo_pages.route("/records/list") @jsonify @loginrequired def listnetworkrecords() -> Dict[str, Any]: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) return frontend.get_network_records() @danevo_pages.route("/records/") @loginrequired def viewrecords(userid: UserID) -> Response: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) info = frontend.get_latest_player_info([userid]).get(userid) if info is None: abort(404) versions = {version: name for (game, version, name) in frontend.all_games()} return render_react( f'{info["name"]}\'s Dance Evolution Records', "danevo/records.react.js", { "records": frontend.get_records(userid), "songs": frontend.get_all_songs(), "players": {}, "versions": versions, "shownames": False, "showpersonalsort": True, "filterempty": True, }, { "refresh": url_for("danevo_pages.listrecords", userid=userid), "player": url_for("danevo_pages.viewplayer", userid=-1), "individual_score": url_for("danevo_pages.viewtopscores", musicid=-1), }, ) @danevo_pages.route("/records//list") @jsonify @loginrequired def listrecords(userid: UserID) -> Dict[str, Any]: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) return { "records": frontend.get_records(userid), "players": {}, } @danevo_pages.route("/topscores/") @loginrequired def viewtopscores(musicid: int) -> Response: # We just want to find the latest mix that this song exists in frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) versions = sorted( [version for (game, version, name) in frontend.all_games()], reverse=True, ) name = None artist = None genre = None kcal = None levels = [0, 0, 0, 0, 0] for version in versions: for chart in [0, 1, 2, 3, 4]: details = g.data.local.music.get_song(GameConstants.DANCE_EVOLUTION, version, musicid, chart) if details is not None: if name is None: name = details.name if artist is None: artist = details.artist if genre is None: genre = details.genre if kcal is None: kcal = details.data.get_float("kcal") if levels[chart] == 0: levels[chart] = details.data.get_int("level") if name is None or not [x for x in levels if x > 0]: # Not a real song! abort(404) top_scores = frontend.get_top_scores(musicid) return render_react( f"Top Dance Evolution Scores for {artist} - {name}", "danevo/topscores.react.js", { "name": name, "artist": artist, "genre": genre, "levels": levels, "kcal": kcal, "players": top_scores["players"], "topscores": top_scores["topscores"], }, { "refresh": url_for("danevo_pages.listtopscores", musicid=musicid), "player": url_for("danevo_pages.viewplayer", userid=-1), }, ) @danevo_pages.route("/topscores//list") @jsonify @loginrequired def listtopscores(musicid: int) -> Dict[str, Any]: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) return frontend.get_top_scores(musicid) @danevo_pages.route("/players") @loginrequired def viewplayers() -> Response: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) return render_react( "All Dance Evolution Players", "danevo/allplayers.react.js", {"players": frontend.get_all_players()}, { "refresh": url_for("danevo_pages.listplayers"), "player": url_for("danevo_pages.viewplayer", userid=-1), }, ) @danevo_pages.route("/players/list") @jsonify @loginrequired def listplayers() -> Dict[str, Any]: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) return { "players": frontend.get_all_players(), } @danevo_pages.route("/players/") @loginrequired def viewplayer(userid: UserID) -> Response: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) info = frontend.get_all_player_info([userid])[userid] if not info: abort(404) latest_version = sorted(info.keys(), reverse=True)[0] return render_react( f'{info[latest_version]["name"]}\'s Dance Evolution Profile', "danevo/player.react.js", { "playerid": userid, "own_profile": userid == g.userID, "player": info, "versions": {version: name for (game, version, name) in frontend.all_games()}, }, { "refresh": url_for("danevo_pages.listplayer", userid=userid), "records": url_for("danevo_pages.viewrecords", userid=userid), "scores": url_for("danevo_pages.viewscores", userid=userid), }, ) @danevo_pages.route("/players//list") @jsonify @loginrequired def listplayer(userid: UserID) -> Dict[str, Any]: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) info = frontend.get_all_player_info([userid])[userid] return { "player": info, } @danevo_pages.route("/options") @loginrequired def viewsettings() -> Response: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) userid = g.userID info = frontend.get_all_player_info([userid])[userid] if not info: abort(404) return render_react( "Dance Evolution Game Settings", "danevo/settings.react.js", { "player": info, "versions": {version: name for (game, version, name) in frontend.all_games()}, }, { "updatename": url_for("danevo_pages.updatename"), }, ) @danevo_pages.route("/options/name/update", methods=["POST"]) @jsonify @loginrequired def updatename() -> Dict[str, Any]: version = int(request.get_json()["version"]) name = request.get_json()["name"] user = g.data.local.user.get_user(g.userID) if user is None: raise Exception("Unable to find user to update!") # Grab profile and update name profile = g.data.local.user.get_profile(GameConstants.DANCE_EVOLUTION, version, user.id) if profile is None: raise Exception("Unable to find profile to update!") if len(name) == 0 or len(name) > 10: raise Exception("Invalid profile name!") if ( re.match( "^[" + "\uff20-\uff3a" # widetext A-Z and @ + "\uff40-\uff5a" # widetext a-z and ` + "\uff10-\uff19" # widetext 0-9 + "\uff0c\uff0e\uff3f\u0437\u2200\u2207" # ,._ and face symbols + "\u3041-\u308d\u308f\u3092\u3093" # hiragana + "\u30a1-\u30ed\u30ef\u30f2\u30f3\u30fc" # katakana + "\u2605\u266a\uff01\uff1f\uff0b" # allowed symbols + "\u2212\u00d7\u00f7\uff03\u3002" # allowed symbols + "\u2267\u2266\u0434\u0398\u25a1" # allowed symbols + "\u76bf\uff1b\uff1a\u301c\uff0a" # allowed symbols + "\u00b4\uff3e\u309c\uff1c\uff1e" # allowed symbols + "\uff08\uff09\u8278\u30fb\uff0f" # allowed symbols + "\u309d\u03c9\u03b5" # allowed symbols + "]*$", name, ) is None ): raise Exception("Invalid profile name!") profile.replace_str("name", name) g.data.local.user.put_profile(GameConstants.DANCE_EVOLUTION, version, user.id, profile) # Return that we updated return { "version": version, "name": name, } @danevo_pages.route("/dancemates/") @loginrequired def viewdancemates(userid: UserID) -> Response: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) info = frontend.get_latest_player_info([userid]).get(userid) if info is None: abort(404) # Since we have one version of DanEvo this is an ugly hack. dancemates_by_version, profiles = frontend.get_rivals(userid) dancemates = [] for version, actual_dancemates in dancemates_by_version.items(): dancemates.extend(actual_dancemates) return render_react( f'{info["name"]}\'s Dance Evolution Dance Mates', "danevo/dancemates.react.js", { "name": info["name"], "version": version, "dancemates": dancemates, "profiles": profiles, }, { "refresh": url_for("danevo_pages.listdancemates", userid=userid), }, ) @danevo_pages.route("/dancemates//list") @jsonify @loginrequired def listdancemates(userid: UserID) -> Dict[str, Any]: frontend = DanceEvolutionFrontend(g.data, g.config, g.cache) info = frontend.get_latest_player_info([userid]).get(userid) if info is None: abort(404) # Since we have one version of DanEvo this is an ugly hack. dancemates_by_version, profiles = frontend.get_rivals(userid) dancemates = [] for version, actual_dancemates in dancemates_by_version.items(): dancemates.extend(actual_dancemates) return { "name": info["name"], "version": version, "dancemates": dancemates, "profiles": profiles, }