2023-10-09 18:48:50 +00:00
|
|
|
import 'dart:io';
|
2023-10-10 20:20:27 +00:00
|
|
|
import 'package:flutter/foundation.dart';
|
2023-07-07 20:32:57 +00:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:intl/intl.dart';
|
2023-09-02 22:48:50 +00:00
|
|
|
import 'package:tetra_stats/data_objects/tetrio.dart';
|
2023-07-15 16:22:25 +00:00
|
|
|
import 'package:tetra_stats/gen/strings.g.dart';
|
2023-07-07 20:32:57 +00:00
|
|
|
import 'package:tetra_stats/services/tetrio_crud.dart';
|
2023-07-09 16:50:17 +00:00
|
|
|
import 'package:tetra_stats/views/main_view.dart';
|
2023-08-14 21:26:20 +00:00
|
|
|
import 'package:tetra_stats/views/rank_averages_view.dart';
|
2023-07-09 16:50:17 +00:00
|
|
|
import 'package:tetra_stats/views/ranks_averages_view.dart';
|
2023-10-09 18:48:50 +00:00
|
|
|
import 'package:window_manager/window_manager.dart';
|
2023-07-07 20:32:57 +00:00
|
|
|
|
2023-10-26 22:38:03 +00:00
|
|
|
final TetrioService _teto = TetrioService();
|
|
|
|
List<DropdownMenuItem> _itemStats = [for (MapEntry e in chartsShortTitles.entries) DropdownMenuItem(value: e.key, child: Text(e.value))];
|
|
|
|
Stats _sortBy = Stats.tr;
|
2023-09-02 22:48:50 +00:00
|
|
|
bool reversed = false;
|
2023-10-26 22:38:03 +00:00
|
|
|
List<DropdownMenuItem> _itemCountries = [for (MapEntry e in t.countries.entries) DropdownMenuItem(value: e.key, child: Text(e.value))];
|
|
|
|
String _country = "";
|
|
|
|
late String _oldWindowTitle;
|
|
|
|
final NumberFormat _f4 = NumberFormat.decimalPatternDigits(locale: LocaleSettings.currentLocale.languageCode, decimalDigits: 4);
|
2023-07-07 20:32:57 +00:00
|
|
|
|
|
|
|
class TLLeaderboardView extends StatefulWidget {
|
2024-01-13 18:49:36 +00:00
|
|
|
const TLLeaderboardView({super.key});
|
2023-07-07 20:32:57 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => TLLeaderboardState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class TLLeaderboardState extends State<TLLeaderboardView> {
|
2023-10-09 18:48:50 +00:00
|
|
|
@override
|
|
|
|
void initState() {
|
2023-10-26 22:38:03 +00:00
|
|
|
if (!kIsWeb && !Platform.isAndroid && !Platform.isIOS) windowManager.getTitle().then((value) => _oldWindowTitle = value);
|
2023-10-09 18:48:50 +00:00
|
|
|
super.initState();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
2023-10-26 22:38:03 +00:00
|
|
|
if (!kIsWeb && !Platform.isAndroid && !Platform.isIOS) windowManager.setTitle(_oldWindowTitle);
|
2023-10-09 18:48:50 +00:00
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
2023-07-07 20:32:57 +00:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2023-07-15 16:22:25 +00:00
|
|
|
final t = Translations.of(context);
|
2024-01-13 18:49:36 +00:00
|
|
|
final NumberFormat f2 = NumberFormat.decimalPattern(LocaleSettings.currentLocale.languageCode)..maximumFractionDigits = 2;
|
2023-07-07 20:32:57 +00:00
|
|
|
return Scaffold(
|
|
|
|
appBar: AppBar(
|
2023-07-15 16:22:25 +00:00
|
|
|
title: Text(t.tlLeaderboard),
|
2023-07-09 16:50:17 +00:00
|
|
|
actions: [
|
|
|
|
IconButton(
|
|
|
|
onPressed: () {
|
|
|
|
Navigator.push(
|
|
|
|
context,
|
|
|
|
MaterialPageRoute(
|
|
|
|
builder: (context) => const RankAveragesView(),
|
|
|
|
maintainState: false,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
icon: const Icon(Icons.compress),
|
2023-07-15 16:22:25 +00:00
|
|
|
tooltip: t.averages,
|
2023-07-09 16:50:17 +00:00
|
|
|
),
|
|
|
|
],
|
2023-07-07 20:32:57 +00:00
|
|
|
),
|
|
|
|
backgroundColor: Colors.black,
|
|
|
|
body: SafeArea(
|
|
|
|
child: FutureBuilder(
|
2023-10-26 22:38:03 +00:00
|
|
|
future: _teto.fetchTLLeaderboard(),
|
2023-07-07 20:32:57 +00:00
|
|
|
builder: (context, snapshot) {
|
|
|
|
switch (snapshot.connectionState) {
|
|
|
|
case ConnectionState.none:
|
|
|
|
case ConnectionState.waiting:
|
|
|
|
case ConnectionState.active:
|
|
|
|
return const Center(child: Text('Fetching...'));
|
|
|
|
case ConnectionState.done:
|
2023-10-26 22:38:03 +00:00
|
|
|
final allPlayers = snapshot.data?.getStatRanking(snapshot.data!.leaderboard, _sortBy, reversed: reversed, country: _country);
|
2023-10-10 20:20:27 +00:00
|
|
|
if (!kIsWeb && !Platform.isAndroid && !Platform.isIOS) windowManager.setTitle("Tetra Stats: ${t.tlLeaderboard} - ${t.players(n: allPlayers!.length)}");
|
2023-07-07 20:32:57 +00:00
|
|
|
return NestedScrollView(
|
|
|
|
headerSliverBuilder: (context, value) {
|
|
|
|
String howManyPlayers(int numberOfPlayers) => Intl.plural(
|
|
|
|
numberOfPlayers,
|
2023-07-15 16:22:25 +00:00
|
|
|
zero: t.lbViewZeroEntrys,
|
|
|
|
one: t.lbViewOneEntry,
|
2023-08-20 21:57:01 +00:00
|
|
|
other: t.lbViewManyEntrys(numberOfPlayers: t.players(n: numberOfPlayers)),
|
2023-07-07 20:32:57 +00:00
|
|
|
name: 'howManyPeople',
|
|
|
|
args: [numberOfPlayers],
|
|
|
|
desc: 'Description of how many people are seen in a place.',
|
|
|
|
examples: const {'numberOfPeople': 3},
|
|
|
|
);
|
|
|
|
return [
|
|
|
|
SliverToBoxAdapter(
|
|
|
|
child: Padding(
|
|
|
|
padding: const EdgeInsets.only(left: 16),
|
2023-08-21 16:28:00 +00:00
|
|
|
child: Wrap(
|
|
|
|
direction: Axis.horizontal,
|
|
|
|
alignment: WrapAlignment.spaceBetween,
|
2023-08-14 21:26:20 +00:00
|
|
|
children: [
|
|
|
|
Text(
|
2023-07-07 20:32:57 +00:00
|
|
|
howManyPlayers(allPlayers.length),
|
|
|
|
style: const TextStyle(color: Colors.white, fontSize: 25),
|
|
|
|
),
|
2023-08-14 21:26:20 +00:00
|
|
|
TextButton(onPressed: (){
|
|
|
|
Navigator.push(
|
|
|
|
context,
|
|
|
|
MaterialPageRoute(
|
|
|
|
builder: (context) => RankView(rank: snapshot.data!.getAverageOfRank("")),
|
|
|
|
),
|
|
|
|
);
|
2023-08-20 21:57:01 +00:00
|
|
|
}, child: Text(t.everyoneAverages,
|
2023-08-14 21:26:20 +00:00
|
|
|
style: const TextStyle(fontSize: 25)))
|
|
|
|
],)
|
2023-07-07 20:32:57 +00:00
|
|
|
)),
|
2023-09-02 22:48:50 +00:00
|
|
|
SliverToBoxAdapter(child: Padding(
|
|
|
|
padding: const EdgeInsets.only(left: 16),
|
|
|
|
child: Wrap(
|
|
|
|
direction: Axis.horizontal,
|
|
|
|
alignment: WrapAlignment.start,
|
|
|
|
crossAxisAlignment: WrapCrossAlignment.center,
|
|
|
|
spacing: 16,
|
|
|
|
children: [
|
|
|
|
Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.baseline,
|
|
|
|
textBaseline: TextBaseline.alphabetic,
|
|
|
|
children: [
|
|
|
|
Text("${t.sortBy}: ",
|
|
|
|
style: const TextStyle(color: Colors.white, fontSize: 25)),
|
2023-10-26 22:38:03 +00:00
|
|
|
DropdownButton(items: _itemStats, value: _sortBy, onChanged: ((value) {
|
|
|
|
_sortBy = value;
|
2023-09-02 22:48:50 +00:00
|
|
|
setState(() {});
|
|
|
|
}),),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.baseline,
|
|
|
|
textBaseline: TextBaseline.alphabetic,
|
|
|
|
children: [
|
|
|
|
Text("${t.reversed}: ",
|
|
|
|
style: const TextStyle(color: Colors.white, fontSize: 25)),
|
|
|
|
Padding(
|
|
|
|
padding: const EdgeInsets.fromLTRB(0, 5.5, 0, 7.5),
|
|
|
|
child: Checkbox(value: reversed,
|
|
|
|
checkColor: Colors.black,
|
|
|
|
onChanged: ((value) {
|
|
|
|
reversed = value!;
|
|
|
|
setState(() {});
|
|
|
|
}),),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.baseline,
|
|
|
|
textBaseline: TextBaseline.alphabetic,
|
|
|
|
children: [
|
|
|
|
Text("${t.country}: ",
|
|
|
|
style: const TextStyle(color: Colors.white, fontSize: 25)),
|
2023-10-26 22:38:03 +00:00
|
|
|
DropdownButton(items: _itemCountries, value: _country, onChanged: ((value) {
|
|
|
|
_country = value;
|
2023-09-02 22:48:50 +00:00
|
|
|
setState(() {});
|
|
|
|
}),),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),),
|
2023-07-07 20:32:57 +00:00
|
|
|
const SliverToBoxAdapter(child: Divider())
|
|
|
|
];
|
|
|
|
},
|
|
|
|
body: ListView.builder(
|
|
|
|
itemCount: allPlayers!.length,
|
|
|
|
itemBuilder: (context, index) {
|
2023-07-09 16:50:17 +00:00
|
|
|
bool bigScreen = MediaQuery.of(context).size.width > 768;
|
2023-07-07 20:32:57 +00:00
|
|
|
return ListTile(
|
2023-07-09 16:50:17 +00:00
|
|
|
leading: Text((index+1).toString(), style: bigScreen ? const TextStyle(fontFamily: "Eurostile Round Extended", fontSize: 28) : null),
|
|
|
|
title: Text(allPlayers[index].username, style: const TextStyle(fontFamily: "Eurostile Round Extended")),
|
2024-01-13 18:49:36 +00:00
|
|
|
subtitle: Text(_sortBy == Stats.tr ? "${f2.format(allPlayers[index].apm)} APM, ${f2.format(allPlayers[index].pps)} PPS, ${f2.format(allPlayers[index].vs)} VS, ${f2.format(allPlayers[index].nerdStats.app)} APP, ${f2.format(allPlayers[index].nerdStats.vsapm)} VS/APM" : "${_f4.format(allPlayers[index].getStatByEnum(_sortBy))} ${chartsShortTitles[_sortBy]}"),
|
2023-07-07 20:32:57 +00:00
|
|
|
trailing: Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
2024-01-13 18:49:36 +00:00
|
|
|
Text("${f2.format(allPlayers[index].rating)} TR", style: bigScreen ? const TextStyle(fontSize: 28) : null),
|
2023-07-09 16:50:17 +00:00
|
|
|
Image.asset("res/tetrio_tl_alpha_ranks/${allPlayers[index].rank}.png", height: bigScreen ? 48 : 16),
|
2023-07-07 20:32:57 +00:00
|
|
|
],
|
|
|
|
),
|
|
|
|
onTap: () {
|
2023-07-09 16:50:17 +00:00
|
|
|
Navigator.push(
|
|
|
|
context,
|
|
|
|
MaterialPageRoute(
|
|
|
|
builder: (context) => MainView(player: allPlayers[index].userId),
|
|
|
|
maintainState: false,
|
|
|
|
),
|
|
|
|
);
|
2023-07-07 20:32:57 +00:00
|
|
|
},
|
|
|
|
);
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
})),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|