feat: complete scoreboard
Add log in tip, loading screen & player record highlighting
This commit is contained in:
parent
6a5831ef55
commit
662b65a605
@ -73,6 +73,34 @@ AnimationClip:
|
||||
path: Restart Button
|
||||
classID: 1
|
||||
script: {fileID: 0}
|
||||
- curve:
|
||||
serializedVersion: 2
|
||||
m_Curve:
|
||||
- serializedVersion: 3
|
||||
time: 0
|
||||
value: 0.5019608
|
||||
inSlope: 0
|
||||
outSlope: 0
|
||||
tangentMode: 136
|
||||
weightedMode: 0
|
||||
inWeight: 0.33333334
|
||||
outWeight: 0.33333334
|
||||
- serializedVersion: 3
|
||||
time: 1
|
||||
value: 0
|
||||
inSlope: 0
|
||||
outSlope: 0
|
||||
tangentMode: 136
|
||||
weightedMode: 0
|
||||
inWeight: 0.33333334
|
||||
outWeight: 0.33333334
|
||||
m_PreInfinity: 2
|
||||
m_PostInfinity: 2
|
||||
m_RotationOrder: 4
|
||||
attribute: m_EffectColor.a
|
||||
path: Scoreboard Window Image
|
||||
classID: 114
|
||||
script: {fileID: 11500000, guid: e19747de3f5aca642ab2be37e372fb86, type: 3}
|
||||
m_PPtrCurves: []
|
||||
m_SampleRate: 60
|
||||
m_WrapMode: 0
|
||||
@ -95,6 +123,13 @@ AnimationClip:
|
||||
typeID: 1
|
||||
customType: 0
|
||||
isPPtrCurve: 0
|
||||
- serializedVersion: 2
|
||||
path: 72073735
|
||||
attribute: 1760254681
|
||||
script: {fileID: 11500000, guid: e19747de3f5aca642ab2be37e372fb86, type: 3}
|
||||
typeID: 114
|
||||
customType: 0
|
||||
isPPtrCurve: 0
|
||||
pptrCurveMapping: []
|
||||
m_AnimationClipSettings:
|
||||
serializedVersion: 2
|
||||
@ -173,6 +208,34 @@ AnimationClip:
|
||||
path: Restart Button
|
||||
classID: 1
|
||||
script: {fileID: 0}
|
||||
- curve:
|
||||
serializedVersion: 2
|
||||
m_Curve:
|
||||
- serializedVersion: 3
|
||||
time: 0
|
||||
value: 0.5019608
|
||||
inSlope: 0
|
||||
outSlope: 0
|
||||
tangentMode: 136
|
||||
weightedMode: 0
|
||||
inWeight: 0.33333334
|
||||
outWeight: 0.33333334
|
||||
- serializedVersion: 3
|
||||
time: 1
|
||||
value: 0
|
||||
inSlope: 0
|
||||
outSlope: 0
|
||||
tangentMode: 136
|
||||
weightedMode: 0
|
||||
inWeight: 0.33333334
|
||||
outWeight: 0.33333334
|
||||
m_PreInfinity: 2
|
||||
m_PostInfinity: 2
|
||||
m_RotationOrder: 4
|
||||
attribute: m_EffectColor.a
|
||||
path: Scoreboard Window Image
|
||||
classID: 114
|
||||
script: {fileID: 11500000, guid: e19747de3f5aca642ab2be37e372fb86, type: 3}
|
||||
m_EulerEditorCurves: []
|
||||
m_HasGenericRootTransform: 0
|
||||
m_HasMotionFloatCurves: 0
|
||||
|
1500
Assets/Scenes/SampleScene.unity
generated
1500
Assets/Scenes/SampleScene.unity
generated
File diff suppressed because it is too large
Load Diff
@ -106,12 +106,17 @@ public class ScoreManager : MonoBehaviour
|
||||
|
||||
_highScore = _currentScore;
|
||||
SessionStore.HighScore = _highScore;
|
||||
|
||||
if (SessionStore.UserData == null)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
await HttpClient.Post<ScoreboardRecordDto>(
|
||||
$"{SessionStore.ApiUrl}/scoreboard",
|
||||
new ScoreboardRecordDto
|
||||
{
|
||||
PostTime = DateTime.UtcNow, Score = SessionStore.HighScore,
|
||||
PostTime = DateTime.UtcNow, Score = _highScore,
|
||||
User = SessionStore.UserData.ToDto()
|
||||
});
|
||||
}
|
||||
@ -136,7 +141,7 @@ public class ScoreManager : MonoBehaviour
|
||||
break;
|
||||
case GameState.GameOver:
|
||||
await SaveHighScore();
|
||||
_scoreboardManager.SpawnScoreboardRecords();
|
||||
await _scoreboardManager.SpawnScoreboardRecords();
|
||||
break;
|
||||
default:
|
||||
throw new ArgumentOutOfRangeException(nameof(newGameState), newGameState, null);
|
||||
|
@ -1,4 +1,5 @@
|
||||
using System;
|
||||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using DatabaseModels.DataTransferObjets;
|
||||
using TMPro;
|
||||
@ -21,9 +22,12 @@ public class UIManager : MonoBehaviour
|
||||
|
||||
[Header("Scoreboard")]
|
||||
[SerializeField] private RectTransform _scoreboardScrollViewContent;
|
||||
[SerializeField] public GameObject _scoreboardLoadingScreen;
|
||||
[SerializeField] public GameObject _scoreboardLoginTip;
|
||||
[SerializeField] private GameObject _scoreboardRecordPrefab;
|
||||
[SerializeField] private Color _scoreboardRecordColor1;
|
||||
[SerializeField] private Color _scoreboardRecordColor2;
|
||||
[SerializeField] private Color _scoreboardPlayerRecordColor;
|
||||
|
||||
|
||||
private void Awake()
|
||||
@ -63,17 +67,38 @@ public class UIManager : MonoBehaviour
|
||||
isFull = Math.Abs(_experienceSlider.value - _experienceSlider.maxValue) < 0.1f;
|
||||
}
|
||||
|
||||
public void InstantiateScoreboardRecords(ScoreboardRecordDto[] records)
|
||||
public void InstantiateScoreboardRecords(ScoreboardRecordDto[] records, int firstRecordIndex)
|
||||
{
|
||||
_scoreboardScrollViewContent.sizeDelta = new Vector2(0, records.Length * 100);
|
||||
_scoreboardScrollViewContent.localPosition = new Vector3(0, _scoreboardScrollViewContent.sizeDelta.y / records.Length * 2f);
|
||||
|
||||
float yPos = 0;
|
||||
|
||||
if (records.Last().User.Username == SessionStore.UserData.Username)
|
||||
{
|
||||
yPos = _scoreboardScrollViewContent.sizeDelta.y / records.Length / 100;
|
||||
}
|
||||
else
|
||||
{
|
||||
yPos = _scoreboardScrollViewContent.sizeDelta.y / records.Length * 2f;
|
||||
}
|
||||
|
||||
_scoreboardScrollViewContent.localPosition = new Vector3(0, yPos);
|
||||
|
||||
for (int i = 0; i < records.Length; i++)
|
||||
{
|
||||
var record = Instantiate(_scoreboardRecordPrefab, Vector3.zero, Quaternion.identity, _scoreboardScrollViewContent.transform);
|
||||
record.GetComponent<RectTransform>().localPosition = new Vector2(218, -50 - 100 * i);
|
||||
record.GetComponent<Image>().color = i % 2 == 0 ? _scoreboardRecordColor1 : _scoreboardRecordColor2;
|
||||
record.GetComponentInChildren<TextMeshProUGUI>().text = $"{i + 1}. {records[i].User.Username}: {records[i].Score}";
|
||||
|
||||
if (records[i].User.Username == SessionStore.UserData.Username)
|
||||
{
|
||||
record.GetComponent<Image>().color = _scoreboardPlayerRecordColor;
|
||||
}
|
||||
else
|
||||
{
|
||||
record.GetComponent<Image>().color = i % 2 == 0 ? _scoreboardRecordColor1 : _scoreboardRecordColor2;
|
||||
}
|
||||
|
||||
record.GetComponentInChildren<TextMeshProUGUI>().text = $"{firstRecordIndex + i + 1}. {records[i].User.Username}: {records[i].Score}";
|
||||
}
|
||||
}
|
||||
|
||||
@ -81,7 +106,7 @@ public class UIManager : MonoBehaviour
|
||||
{
|
||||
for (int i = 0; i < _scoreboardScrollViewContent.transform.childCount; i++)
|
||||
{
|
||||
Destroy(_scoreboardScrollViewContent.transform.GetChild(i).gameObject);
|
||||
DestroyImmediate(_scoreboardScrollViewContent.transform.GetChild(i).gameObject);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -3,39 +3,41 @@ using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using DatabaseModels.DataTransferObjets;
|
||||
using TMPro;
|
||||
using UnityEngine;
|
||||
using UnityEngine.UI;
|
||||
|
||||
public class ScoreboardManager : MonoBehaviour
|
||||
{
|
||||
[SerializeField] private UIManager _uiManager;
|
||||
|
||||
public async void SpawnScoreboardRecords()
|
||||
public async Task SpawnScoreboardRecords()
|
||||
{
|
||||
if (SessionStore.UserData == null)
|
||||
{
|
||||
// Login to display online scoreboard
|
||||
|
||||
_uiManager._scoreboardLoginTip.SetActive(true);
|
||||
return;
|
||||
}
|
||||
|
||||
var filteredScoreboardRecords = await GetFilteredScoreboardRecords();
|
||||
|
||||
_uiManager._scoreboardLoadingScreen.SetActive(true);
|
||||
_uiManager.DestroyAllScoreboardRecords();
|
||||
_uiManager.InstantiateScoreboardRecords(filteredScoreboardRecords);
|
||||
var filteredScoreboard = await GetFilteredScoreboard();
|
||||
_uiManager.InstantiateScoreboardRecords(filteredScoreboard.records, filteredScoreboard.firstRecordIndex);
|
||||
_uiManager._scoreboardLoadingScreen.SetActive(false);
|
||||
|
||||
async Task<ScoreboardRecordDto[]> GetFilteredScoreboardRecords()
|
||||
async Task<(ScoreboardRecordDto[] records, int firstRecordIndex)> GetFilteredScoreboard()
|
||||
{
|
||||
int spareCount = 5;
|
||||
|
||||
var localRecords = await HttpClient.Get<List<ScoreboardRecordDto>>($"{SessionStore.ApiUrl}/scoreboard");
|
||||
var currentUserRecord = localRecords.First(r => r.User.Username == SessionStore.UserData.Username);
|
||||
|
||||
var firstRecordNum = Mathf.Clamp(localRecords.IndexOf(currentUserRecord) - spareCount, 0, localRecords.Count);
|
||||
|
||||
var filteredRecords = localRecords
|
||||
.SkipWhile(r => Math.Abs(localRecords.IndexOf(r) - localRecords.IndexOf(currentUserRecord)) >= 5)
|
||||
.TakeWhile(r => Math.Abs(localRecords.IndexOf(r) - localRecords.IndexOf(currentUserRecord)) <= 5)
|
||||
.SkipWhile(r => Math.Abs(localRecords.IndexOf(r) - localRecords.IndexOf(currentUserRecord)) >= spareCount)
|
||||
.TakeWhile(r => Math.Abs(localRecords.IndexOf(r) - localRecords.IndexOf(currentUserRecord)) <= spareCount)
|
||||
.ToArray();
|
||||
|
||||
return filteredRecords.ToArray();
|
||||
return (filteredRecords, firstRecordNum);
|
||||
}
|
||||
}
|
||||
}
|
@ -1,14 +1,15 @@
|
||||
using System.Linq;
|
||||
using DatabaseModels.DataTransferObjets;
|
||||
using DatabaseModels.Requests;
|
||||
using DatabaseModels.Responses;
|
||||
|
||||
public static class SessionStore
|
||||
{
|
||||
public static string ApiUrl { get; set; }
|
||||
|
||||
public static string Jwt { get; set; }
|
||||
|
||||
public static UserData UserData { get; set; }
|
||||
public static int HighScore { get; set; }
|
||||
|
||||
public static ScoreboardRecordDto[] ScoreboardRecords;
|
||||
|
||||
public static async void Save()
|
||||
{
|
||||
@ -21,17 +22,18 @@ public static class SessionStore
|
||||
UserData = await SaveSystem.LoadFromJsonAsync<UserData>("userData.json");
|
||||
HighScore = await SaveSystem.LoadFromBinaryAsync<int>("HighScore.bin");
|
||||
|
||||
ScoreboardRecords = await HttpClient.Get<ScoreboardRecordDto[]>($"{ApiUrl}/scoreboard");
|
||||
|
||||
if (UserData == null)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
int? dbHighScore = ScoreboardRecords?.FirstOrDefault(sbr => sbr.User.Username == UserData.Username)?.Score;
|
||||
if (dbHighScore != null && dbHighScore > HighScore)
|
||||
|
||||
var authResponse = await HttpClient.Post<AuthenticationResponse>($"{ApiUrl}/auth/login", new AuthenticationRequest { Username = UserData.Username, Password = UserData.Password } );
|
||||
Jwt = authResponse.Token;
|
||||
|
||||
var dbHighScore = await HttpClient.Get<ScoreboardRecordDto>($"{ApiUrl}/scoreboard/{UserData.Username}");
|
||||
if (dbHighScore?.Score != null && dbHighScore.Score > HighScore)
|
||||
{
|
||||
HighScore = (int) dbHighScore;
|
||||
HighScore = dbHighScore.Score;
|
||||
}
|
||||
}
|
||||
}
|
@ -2,12 +2,11 @@ using System;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using Newtonsoft.Json;
|
||||
using UnityEngine;
|
||||
using UnityEngine.Networking;
|
||||
|
||||
public static class HttpClient
|
||||
{
|
||||
private static string _jwt = "";
|
||||
|
||||
public static async Task<T> Get<T>(string endpoint)
|
||||
{
|
||||
var getRequest = CreateRequest(endpoint, RequestType.GET);
|
||||
@ -22,8 +21,15 @@ public static class HttpClient
|
||||
{
|
||||
return JsonConvert.DeserializeObject<T>(getRequest.downloadHandler.text);
|
||||
}
|
||||
catch (Exception)
|
||||
catch (Exception e)
|
||||
{
|
||||
Debug.LogWarning($"HttpClient: GET from {endpoint}" +
|
||||
$"\nSent object type: {typeof(T)}" +
|
||||
$"\nError: {getRequest.error}" +
|
||||
$"\n\n" +
|
||||
$"See details in a exception logged below." +
|
||||
$"\n\n");
|
||||
Debug.LogException(e);
|
||||
return default(T);
|
||||
}
|
||||
}
|
||||
@ -42,13 +48,23 @@ public static class HttpClient
|
||||
{
|
||||
await Task.Delay(10);
|
||||
}
|
||||
|
||||
return JsonConvert.DeserializeObject<T>(postRequest.downloadHandler.text);
|
||||
}
|
||||
|
||||
public static void SetJwt(string jwt)
|
||||
{
|
||||
_jwt = jwt;
|
||||
try
|
||||
{
|
||||
return JsonConvert.DeserializeObject<T>(postRequest.downloadHandler.text);
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
Debug.LogWarning($"HttpClient: POST to {endpoint}" +
|
||||
$"\nSent object type: {payload.GetType()}" +
|
||||
$"\nRetrieved object type: {typeof(T)}" +
|
||||
$"\nError: {postRequest.error}" +
|
||||
$"\n\n" +
|
||||
$"See details in a exception logged below." +
|
||||
$"\n\n");
|
||||
Debug.LogException(e);
|
||||
return default(T);
|
||||
}
|
||||
}
|
||||
|
||||
private static UnityWebRequest CreateRequest(string path, RequestType type, object data = null)
|
||||
@ -64,9 +80,9 @@ public static class HttpClient
|
||||
request.downloadHandler = new DownloadHandlerBuffer();
|
||||
request.SetRequestHeader("Content-Type", "application/json");
|
||||
|
||||
if (_jwt != null)
|
||||
if (SessionStore.Jwt != null)
|
||||
{
|
||||
request.SetRequestHeader("Authorization", $"Bearer {_jwt}");
|
||||
request.SetRequestHeader("Authorization", $"Bearer {SessionStore.Jwt}");
|
||||
}
|
||||
|
||||
request.certificateHandler = new CertificateWhore();
|
||||
|
Loading…
Reference in New Issue
Block a user