mirror of
https://github.com/DarkflameUniverse/DarkflameServer.git
synced 2025-01-09 06:17:10 +00:00
455f9470a5
* Move EntityManager to Game namespace * move initialization to later Need to wait for dZoneManager to be initialized. * Fix bugs - Cannot delete from a RandomAccessIterator while in a range based for loop. Touchup zone manager initialize replace magic numbers with better named constants replace magic zonecontrol id with a more readable hex alternative condense stack variables move initializers closer to their use initialize entity manager with zone control change initialize timings If zone is not zero we expect to initialize the entity manager during zone manager initialization Add constexpr for zone control LOT * Add proper error handling * revert vanity changes * Update WorldServer.cpp * Update dZoneManager.cpp
471 lines
18 KiB
C++
471 lines
18 KiB
C++
#include "LeaderboardManager.h"
|
|
#include <utility>
|
|
#include "Database.h"
|
|
#include "EntityManager.h"
|
|
#include "Character.h"
|
|
#include "Game.h"
|
|
#include "GameMessages.h"
|
|
#include "dLogger.h"
|
|
#include "dConfig.h"
|
|
#include "CDClientManager.h"
|
|
#include "GeneralUtils.h"
|
|
#include "Entity.h"
|
|
|
|
#include "CDActivitiesTable.h"
|
|
|
|
Leaderboard::Leaderboard(uint32_t gameID, uint32_t infoType, bool weekly, std::vector<LeaderboardEntry> entries,
|
|
LWOOBJID relatedPlayer, LeaderboardType leaderboardType) {
|
|
this->relatedPlayer = relatedPlayer;
|
|
this->gameID = gameID;
|
|
this->weekly = weekly;
|
|
this->infoType = infoType;
|
|
this->entries = std::move(entries);
|
|
this->leaderboardType = leaderboardType;
|
|
}
|
|
|
|
std::u16string Leaderboard::ToString() const {
|
|
std::string leaderboard;
|
|
|
|
leaderboard += "ADO.Result=7:1\n";
|
|
leaderboard += "Result.Count=1:1\n";
|
|
leaderboard += "Result[0].Index=0:RowNumber\n";
|
|
leaderboard += "Result[0].RowCount=1:" + std::to_string(entries.size()) + "\n";
|
|
|
|
auto index = 0;
|
|
for (const auto& entry : entries) {
|
|
leaderboard += "Result[0].Row[" + std::to_string(index) + "].LastPlayed=8:" + std::to_string(entry.lastPlayed) + "\n";
|
|
leaderboard += "Result[0].Row[" + std::to_string(index) + "].CharacterID=8:" + std::to_string(entry.playerID) + "\n";
|
|
leaderboard += "Result[0].Row[" + std::to_string(index) + "].NumPlayed=1:1\n";
|
|
leaderboard += "Result[0].Row[" + std::to_string(index) + "].RowNumber=8:" + std::to_string(entry.placement) + "\n";
|
|
leaderboard += "Result[0].Row[" + std::to_string(index) + "].Time=1:" + std::to_string(entry.time) + "\n";
|
|
|
|
// Only these minigames have a points system
|
|
if (leaderboardType == Survival || leaderboardType == ShootingGallery) {
|
|
leaderboard += "Result[0].Row[" + std::to_string(index) + "].Points=1:" + std::to_string(entry.score) + "\n";
|
|
} else if (leaderboardType == SurvivalNS) {
|
|
leaderboard += "Result[0].Row[" + std::to_string(index) + "].Wave=1:" + std::to_string(entry.score) + "\n";
|
|
}
|
|
|
|
leaderboard += "Result[0].Row[" + std::to_string(index) + "].name=0:" + entry.playerName + "\n";
|
|
index++;
|
|
}
|
|
|
|
return GeneralUtils::UTF8ToUTF16(leaderboard);
|
|
}
|
|
|
|
std::vector<LeaderboardEntry> Leaderboard::GetEntries() {
|
|
return entries;
|
|
}
|
|
|
|
uint32_t Leaderboard::GetGameID() const {
|
|
return gameID;
|
|
}
|
|
|
|
uint32_t Leaderboard::GetInfoType() const {
|
|
return infoType;
|
|
}
|
|
|
|
void Leaderboard::Send(LWOOBJID targetID) const {
|
|
auto* player = Game::entityManager->GetEntity(relatedPlayer);
|
|
if (player != nullptr) {
|
|
GameMessages::SendActivitySummaryLeaderboardData(targetID, this, player->GetSystemAddress());
|
|
}
|
|
}
|
|
|
|
void LeaderboardManager::SaveScore(LWOOBJID playerID, uint32_t gameID, uint32_t score, uint32_t time) {
|
|
const auto* player = Game::entityManager->GetEntity(playerID);
|
|
if (player == nullptr)
|
|
return;
|
|
|
|
auto* character = player->GetCharacter();
|
|
if (character == nullptr)
|
|
return;
|
|
|
|
auto* select = Database::CreatePreppedStmt("SELECT time, score FROM leaderboard WHERE character_id = ? AND game_id = ?;");
|
|
|
|
select->setUInt64(1, character->GetID());
|
|
select->setInt(2, gameID);
|
|
|
|
auto any = false;
|
|
auto* result = select->executeQuery();
|
|
auto leaderboardType = GetLeaderboardType(gameID);
|
|
|
|
// Check if the new score is a high score
|
|
while (result->next()) {
|
|
any = true;
|
|
|
|
const auto storedTime = result->getInt(1);
|
|
const auto storedScore = result->getInt(2);
|
|
auto highscore = true;
|
|
bool classicSurvivalScoring = Game::config->GetValue("classic_survival_scoring") == "1";
|
|
|
|
switch (leaderboardType) {
|
|
case ShootingGallery:
|
|
if (score <= storedScore)
|
|
highscore = false;
|
|
break;
|
|
case Racing:
|
|
if (time >= storedTime)
|
|
highscore = false;
|
|
break;
|
|
case MonumentRace:
|
|
if (time >= storedTime)
|
|
highscore = false;
|
|
break;
|
|
case FootRace:
|
|
if (time <= storedTime)
|
|
highscore = false;
|
|
break;
|
|
case Survival:
|
|
if (classicSurvivalScoring) {
|
|
if (time <= storedTime) { // Based on time (LU live)
|
|
highscore = false;
|
|
}
|
|
} else {
|
|
if (score <= storedScore) // Based on score (DLU)
|
|
highscore = false;
|
|
}
|
|
break;
|
|
case SurvivalNS:
|
|
if (!(score > storedScore || (time < storedTime && score >= storedScore)))
|
|
highscore = false;
|
|
break;
|
|
default:
|
|
highscore = false;
|
|
}
|
|
|
|
if (!highscore) {
|
|
delete select;
|
|
delete result;
|
|
return;
|
|
}
|
|
}
|
|
|
|
delete select;
|
|
delete result;
|
|
|
|
if (any) {
|
|
auto* statement = Database::CreatePreppedStmt("UPDATE leaderboard SET time = ?, score = ?, last_played=SYSDATE() WHERE character_id = ? AND game_id = ?;");
|
|
statement->setInt(1, time);
|
|
statement->setInt(2, score);
|
|
statement->setUInt64(3, character->GetID());
|
|
statement->setInt(4, gameID);
|
|
statement->execute();
|
|
|
|
delete statement;
|
|
} else {
|
|
// Note: last_played will be set to SYSDATE() by default when inserting into leaderboard
|
|
auto* statement = Database::CreatePreppedStmt("INSERT INTO leaderboard (character_id, game_id, time, score) VALUES (?, ?, ?, ?);");
|
|
statement->setUInt64(1, character->GetID());
|
|
statement->setInt(2, gameID);
|
|
statement->setInt(3, time);
|
|
statement->setInt(4, score);
|
|
statement->execute();
|
|
|
|
delete statement;
|
|
}
|
|
}
|
|
|
|
Leaderboard* LeaderboardManager::GetLeaderboard(uint32_t gameID, InfoType infoType, bool weekly, LWOOBJID playerID) {
|
|
auto leaderboardType = GetLeaderboardType(gameID);
|
|
|
|
std::string query;
|
|
bool classicSurvivalScoring = Game::config->GetValue("classic_survival_scoring") == "1";
|
|
switch (infoType) {
|
|
case InfoType::Standings:
|
|
switch (leaderboardType) {
|
|
case ShootingGallery:
|
|
query = standingsScoreQuery; // Shooting gallery is based on the highest score.
|
|
break;
|
|
case FootRace:
|
|
query = standingsTimeQuery; // The higher your time, the better for FootRace.
|
|
break;
|
|
case Survival:
|
|
query = classicSurvivalScoring ? standingsTimeQuery : standingsScoreQuery;
|
|
break;
|
|
case SurvivalNS:
|
|
query = standingsScoreQueryAsc; // BoNS is scored by highest wave (score) first, then time.
|
|
break;
|
|
default:
|
|
query = standingsTimeQueryAsc; // MonumentRace and Racing are based on the shortest time.
|
|
}
|
|
break;
|
|
case InfoType::Friends:
|
|
switch (leaderboardType) {
|
|
case ShootingGallery:
|
|
query = friendsScoreQuery; // Shooting gallery is based on the highest score.
|
|
break;
|
|
case FootRace:
|
|
query = friendsTimeQuery; // The higher your time, the better for FootRace.
|
|
break;
|
|
case Survival:
|
|
query = classicSurvivalScoring ? friendsTimeQuery : friendsScoreQuery;
|
|
break;
|
|
case SurvivalNS:
|
|
query = friendsScoreQueryAsc; // BoNS is scored by highest wave (score) first, then time.
|
|
break;
|
|
default:
|
|
query = friendsTimeQueryAsc; // MonumentRace and Racing are based on the shortest time.
|
|
}
|
|
break;
|
|
|
|
default:
|
|
switch (leaderboardType) {
|
|
case ShootingGallery:
|
|
query = topPlayersScoreQuery; // Shooting gallery is based on the highest score.
|
|
break;
|
|
case FootRace:
|
|
query = topPlayersTimeQuery; // The higher your time, the better for FootRace.
|
|
break;
|
|
case Survival:
|
|
query = classicSurvivalScoring ? topPlayersTimeQuery : topPlayersScoreQuery;
|
|
break;
|
|
case SurvivalNS:
|
|
query = topPlayersScoreQueryAsc; // BoNS is scored by highest wave (score) first, then time.
|
|
break;
|
|
default:
|
|
query = topPlayersTimeQueryAsc; // MonumentRace and Racing are based on the shortest time.
|
|
}
|
|
}
|
|
|
|
auto* statement = Database::CreatePreppedStmt(query);
|
|
statement->setUInt(1, gameID);
|
|
|
|
// Only the standings and friends leaderboards require the character ID to be set
|
|
if (infoType == Standings || infoType == Friends) {
|
|
auto characterID = 0;
|
|
|
|
const auto* player = Game::entityManager->GetEntity(playerID);
|
|
if (player != nullptr) {
|
|
auto* character = player->GetCharacter();
|
|
if (character != nullptr)
|
|
characterID = character->GetID();
|
|
}
|
|
|
|
statement->setUInt64(2, characterID);
|
|
}
|
|
|
|
auto* res = statement->executeQuery();
|
|
|
|
std::vector<LeaderboardEntry> entries{};
|
|
|
|
uint32_t index = 0;
|
|
while (res->next()) {
|
|
LeaderboardEntry entry;
|
|
entry.playerID = res->getUInt64(4);
|
|
entry.playerName = res->getString(5);
|
|
entry.time = res->getUInt(1);
|
|
entry.score = res->getUInt(2);
|
|
entry.placement = res->getUInt(3);
|
|
entry.lastPlayed = res->getUInt(6);
|
|
|
|
entries.push_back(entry);
|
|
index++;
|
|
}
|
|
|
|
delete res;
|
|
delete statement;
|
|
|
|
return new Leaderboard(gameID, infoType, weekly, entries, playerID, leaderboardType);
|
|
}
|
|
|
|
void LeaderboardManager::SendLeaderboard(uint32_t gameID, InfoType infoType, bool weekly, LWOOBJID targetID,
|
|
LWOOBJID playerID) {
|
|
const auto* leaderboard = LeaderboardManager::GetLeaderboard(gameID, infoType, weekly, playerID);
|
|
leaderboard->Send(targetID);
|
|
delete leaderboard;
|
|
}
|
|
|
|
LeaderboardType LeaderboardManager::GetLeaderboardType(uint32_t gameID) {
|
|
auto* activitiesTable = CDClientManager::Instance().GetTable<CDActivitiesTable>();
|
|
std::vector<CDActivities> activities = activitiesTable->Query([=](const CDActivities& entry) {
|
|
return (entry.ActivityID == gameID);
|
|
});
|
|
|
|
for (const auto& activity : activities) {
|
|
return static_cast<LeaderboardType>(activity.leaderboardType);
|
|
}
|
|
|
|
return LeaderboardType::None;
|
|
}
|
|
|
|
const std::string LeaderboardManager::topPlayersScoreQuery =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, "
|
|
"RANK() OVER ( ORDER BY l.score DESC, l.time DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
"INNER JOIN charinfo c ON l.character_id = c.id "
|
|
"WHERE l.game_id = ? "
|
|
"ORDER BY leaderboard_rank) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales LIMIT 11;";
|
|
|
|
const std::string LeaderboardManager::friendsScoreQuery =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, f.friend_id, f.player_id, "
|
|
" RANK() OVER ( ORDER BY l.score DESC, l.time DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
" INNER JOIN charinfo c ON l.character_id = c.id "
|
|
" INNER JOIN friends f ON f.player_id = c.id "
|
|
" WHERE l.game_id = ? "
|
|
" ORDER BY leaderboard_rank), "
|
|
" personal_values AS ( "
|
|
" SELECT id as related_player_id, "
|
|
" GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, "
|
|
" GREATEST(leaderboard_rank + 5, 11) AS max_rank "
|
|
" FROM leaderboard_vales WHERE leaderboard_vales.id = ? LIMIT 1) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales, personal_values "
|
|
"WHERE leaderboard_rank BETWEEN min_rank AND max_rank AND (player_id = related_player_id OR friend_id = related_player_id);";
|
|
|
|
const std::string LeaderboardManager::standingsScoreQuery =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, "
|
|
" RANK() OVER ( ORDER BY l.score DESC, l.time DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
" INNER JOIN charinfo c ON l.character_id = c.id "
|
|
" WHERE l.game_id = ? "
|
|
" ORDER BY leaderboard_rank), "
|
|
"personal_values AS ( "
|
|
" SELECT GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, "
|
|
" GREATEST(leaderboard_rank + 5, 11) AS max_rank "
|
|
" FROM leaderboard_vales WHERE id = ? LIMIT 1) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales, personal_values "
|
|
"WHERE leaderboard_rank BETWEEN min_rank AND max_rank;";
|
|
|
|
const std::string LeaderboardManager::topPlayersScoreQueryAsc =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, "
|
|
"RANK() OVER ( ORDER BY l.score DESC, l.time ASC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
"INNER JOIN charinfo c ON l.character_id = c.id "
|
|
"WHERE l.game_id = ? "
|
|
"ORDER BY leaderboard_rank) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales LIMIT 11;";
|
|
|
|
const std::string LeaderboardManager::friendsScoreQueryAsc =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, f.friend_id, f.player_id, "
|
|
" RANK() OVER ( ORDER BY l.score DESC, l.time ASC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
" INNER JOIN charinfo c ON l.character_id = c.id "
|
|
" INNER JOIN friends f ON f.player_id = c.id "
|
|
" WHERE l.game_id = ? "
|
|
" ORDER BY leaderboard_rank), "
|
|
" personal_values AS ( "
|
|
" SELECT id as related_player_id, "
|
|
" GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, "
|
|
" GREATEST(leaderboard_rank + 5, 11) AS max_rank "
|
|
" FROM leaderboard_vales WHERE leaderboard_vales.id = ? LIMIT 1) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales, personal_values "
|
|
"WHERE leaderboard_rank BETWEEN min_rank AND max_rank AND (player_id = related_player_id OR friend_id = related_player_id);";
|
|
|
|
const std::string LeaderboardManager::standingsScoreQueryAsc =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, "
|
|
" RANK() OVER ( ORDER BY l.score DESC, l.time ASC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
" INNER JOIN charinfo c ON l.character_id = c.id "
|
|
" WHERE l.game_id = ? "
|
|
" ORDER BY leaderboard_rank), "
|
|
"personal_values AS ( "
|
|
" SELECT GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, "
|
|
" GREATEST(leaderboard_rank + 5, 11) AS max_rank "
|
|
" FROM leaderboard_vales WHERE id = ? LIMIT 1) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales, personal_values "
|
|
"WHERE leaderboard_rank BETWEEN min_rank AND max_rank;";
|
|
|
|
const std::string LeaderboardManager::topPlayersTimeQuery =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, "
|
|
"RANK() OVER ( ORDER BY l.time DESC, l.score DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
"INNER JOIN charinfo c ON l.character_id = c.id "
|
|
"WHERE l.game_id = ? "
|
|
"ORDER BY leaderboard_rank) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales LIMIT 11;";
|
|
|
|
const std::string LeaderboardManager::friendsTimeQuery =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, f.friend_id, f.player_id, "
|
|
" RANK() OVER ( ORDER BY l.time DESC, l.score DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
" INNER JOIN charinfo c ON l.character_id = c.id "
|
|
" INNER JOIN friends f ON f.player_id = c.id "
|
|
" WHERE l.game_id = ? "
|
|
" ORDER BY leaderboard_rank), "
|
|
" personal_values AS ( "
|
|
" SELECT id as related_player_id, "
|
|
" GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, "
|
|
" GREATEST(leaderboard_rank + 5, 11) AS max_rank "
|
|
" FROM leaderboard_vales WHERE leaderboard_vales.id = ? LIMIT 1) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales, personal_values "
|
|
"WHERE leaderboard_rank BETWEEN min_rank AND max_rank AND (player_id = related_player_id OR friend_id = related_player_id);";
|
|
|
|
const std::string LeaderboardManager::standingsTimeQuery =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, "
|
|
" RANK() OVER ( ORDER BY l.time DESC, l.score DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
" INNER JOIN charinfo c ON l.character_id = c.id "
|
|
" WHERE l.game_id = ? "
|
|
" ORDER BY leaderboard_rank), "
|
|
"personal_values AS ( "
|
|
" SELECT GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, "
|
|
" GREATEST(leaderboard_rank + 5, 11) AS max_rank "
|
|
" FROM leaderboard_vales WHERE id = ? LIMIT 1) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales, personal_values "
|
|
"WHERE leaderboard_rank BETWEEN min_rank AND max_rank;";
|
|
|
|
const std::string LeaderboardManager::topPlayersTimeQueryAsc =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, "
|
|
"RANK() OVER ( ORDER BY l.time ASC, l.score DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
"INNER JOIN charinfo c ON l.character_id = c.id "
|
|
"WHERE l.game_id = ? "
|
|
"ORDER BY leaderboard_rank) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales LIMIT 11;";
|
|
|
|
const std::string LeaderboardManager::friendsTimeQueryAsc =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, f.friend_id, f.player_id, "
|
|
" RANK() OVER ( ORDER BY l.time ASC, l.score DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
" INNER JOIN charinfo c ON l.character_id = c.id "
|
|
" INNER JOIN friends f ON f.player_id = c.id "
|
|
" WHERE l.game_id = ? "
|
|
" ORDER BY leaderboard_rank), "
|
|
" personal_values AS ( "
|
|
" SELECT id as related_player_id, "
|
|
" GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, "
|
|
" GREATEST(leaderboard_rank + 5, 11) AS max_rank "
|
|
" FROM leaderboard_vales WHERE leaderboard_vales.id = ? LIMIT 1) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales, personal_values "
|
|
"WHERE leaderboard_rank BETWEEN min_rank AND max_rank AND (player_id = related_player_id OR friend_id = related_player_id);";
|
|
|
|
const std::string LeaderboardManager::standingsTimeQueryAsc =
|
|
"WITH leaderboard_vales AS ( "
|
|
" SELECT l.time, l.score, UNIX_TIMESTAMP(l.last_played) last_played, c.name, c.id, "
|
|
" RANK() OVER ( ORDER BY l.time ASC, l.score DESC, last_played ) leaderboard_rank "
|
|
" FROM leaderboard l "
|
|
" INNER JOIN charinfo c ON l.character_id = c.id "
|
|
" WHERE l.game_id = ? "
|
|
" ORDER BY leaderboard_rank), "
|
|
"personal_values AS ( "
|
|
" SELECT GREATEST(CAST(leaderboard_rank AS SIGNED) - 5, 1) AS min_rank, "
|
|
" GREATEST(leaderboard_rank + 5, 11) AS max_rank "
|
|
" FROM leaderboard_vales WHERE id = ? LIMIT 1) "
|
|
"SELECT time, score, leaderboard_rank, id, name, last_played "
|
|
"FROM leaderboard_vales, personal_values "
|
|
"WHERE leaderboard_rank BETWEEN min_rank AND max_rank;";
|