2023-10-09 18:48:50 +00:00
|
|
|
import 'dart:io';
|
2024-01-22 18:00:24 +00:00
|
|
|
import 'dart:math';
|
2023-08-14 21:26:20 +00:00
|
|
|
import 'package:fl_chart/fl_chart.dart';
|
2023-10-10 20:20:27 +00:00
|
|
|
import 'package:flutter/foundation.dart';
|
2024-01-22 18:00:24 +00:00
|
|
|
import 'package:flutter/gestures.dart';
|
2023-07-29 18:01:49 +00:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:intl/intl.dart';
|
2023-08-19 23:20:06 +00:00
|
|
|
import 'package:tetra_stats/data_objects/tetrio.dart';
|
2023-07-29 18:01:49 +00:00
|
|
|
import 'package:tetra_stats/gen/strings.g.dart';
|
2024-01-26 20:56:24 +00:00
|
|
|
import 'package:tetra_stats/views/main_view.dart' show MainView, textShadow;
|
2023-10-09 18:48:50 +00:00
|
|
|
import 'package:window_manager/window_manager.dart';
|
2023-07-29 18:01:49 +00:00
|
|
|
|
2023-10-26 22:38:03 +00:00
|
|
|
var _chartsShortTitlesDropdowns = <DropdownMenuItem>[for (MapEntry e in chartsShortTitles.entries) DropdownMenuItem(value: e.key, child: Text(e.value),)];
|
|
|
|
Stats _chartsX = Stats.tr;
|
|
|
|
Stats _chartsY = Stats.apm;
|
|
|
|
List<DropdownMenuItem> _itemStats = [for (MapEntry e in chartsShortTitles.entries) DropdownMenuItem(value: e.key, child: Text(e.value))];
|
|
|
|
Stats _sortBy = Stats.tr;
|
|
|
|
bool _reversed = false;
|
|
|
|
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 _f2 = NumberFormat.decimalPatternDigits(locale: LocaleSettings.currentLocale.languageCode, decimalDigits: 2);
|
|
|
|
final NumberFormat _f4 = NumberFormat.decimalPatternDigits(locale: LocaleSettings.currentLocale.languageCode, decimalDigits: 4);
|
2023-07-29 18:01:49 +00:00
|
|
|
|
|
|
|
class RankView extends StatefulWidget {
|
|
|
|
final List rank;
|
2024-01-13 18:49:36 +00:00
|
|
|
const RankView({super.key, required this.rank});
|
2023-07-29 18:01:49 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => RankState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class RankState extends State<RankView> with SingleTickerProviderStateMixin {
|
|
|
|
late ScrollController _scrollController;
|
|
|
|
late TabController _tabController;
|
2024-01-22 18:00:24 +00:00
|
|
|
late String previousAxisTitles;
|
|
|
|
late double minX;
|
|
|
|
late double actualMinX;
|
|
|
|
late double maxX;
|
|
|
|
late double actualMaxX;
|
|
|
|
late double minY;
|
|
|
|
late double actualMinY;
|
|
|
|
late double maxY;
|
|
|
|
late double actualMaxY;
|
|
|
|
late double xScale;
|
|
|
|
late double yScale;
|
|
|
|
String headerTooltip = t.pseudoTooltipHeaderInit;
|
|
|
|
String footerTooltip = t.pseudoTooltipFooterInit;
|
|
|
|
int hoveredPointId = -1;
|
|
|
|
double scaleFactor = 5e2;
|
|
|
|
double dragFactor = 7e2;
|
2023-07-29 18:01:49 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
_scrollController = ScrollController();
|
|
|
|
_tabController = TabController(length: 6, vsync: this);
|
2023-10-10 20:20:27 +00:00
|
|
|
if (!kIsWeb && !Platform.isAndroid && !Platform.isIOS){
|
2023-10-26 22:38:03 +00:00
|
|
|
windowManager.getTitle().then((value) => _oldWindowTitle = value);
|
2023-10-09 18:48:50 +00:00
|
|
|
windowManager.setTitle("Tetra Stats: ${widget.rank[1]["everyone"] ? t.everyoneAverages : t.rankAverages(rank: widget.rank[0].rank.toUpperCase())}");
|
|
|
|
}
|
2023-07-29 18:01:49 +00:00
|
|
|
super.initState();
|
2024-01-22 18:00:24 +00:00
|
|
|
previousAxisTitles = _chartsX.toString()+_chartsY.toString();
|
|
|
|
recalculateBoundaries();
|
|
|
|
resetScale();
|
|
|
|
}
|
|
|
|
|
|
|
|
void recalculateBoundaries(){
|
|
|
|
actualMinX = (widget.rank[1]["entries"] as List<TetrioPlayerFromLeaderboard>).reduce((value, element) {
|
|
|
|
num n = min(value.getStatByEnum(_chartsX), element.getStatByEnum(_chartsX));
|
|
|
|
if (value.getStatByEnum(_chartsX) == n) {
|
|
|
|
return value;
|
|
|
|
} else {
|
|
|
|
return element;
|
|
|
|
}
|
|
|
|
}).getStatByEnum(_chartsX) as double;
|
|
|
|
actualMaxX = (widget.rank[1]["entries"] as List<TetrioPlayerFromLeaderboard>).reduce((value, element) {
|
|
|
|
num n = max(value.getStatByEnum(_chartsX), element.getStatByEnum(_chartsX));
|
|
|
|
if (value.getStatByEnum(_chartsX) == n) {
|
|
|
|
return value;
|
|
|
|
} else {
|
|
|
|
return element;
|
|
|
|
}
|
|
|
|
}).getStatByEnum(_chartsX) as double;
|
|
|
|
actualMinY = (widget.rank[1]["entries"] as List<TetrioPlayerFromLeaderboard>).reduce((value, element) {
|
|
|
|
num n = min(value.getStatByEnum(_chartsY), element.getStatByEnum(_chartsY));
|
|
|
|
if (value.getStatByEnum(_chartsY) == n) {
|
|
|
|
return value;
|
|
|
|
} else {
|
|
|
|
return element;
|
|
|
|
}
|
|
|
|
}).getStatByEnum(_chartsY) as double;
|
|
|
|
actualMaxY = (widget.rank[1]["entries"] as List<TetrioPlayerFromLeaderboard>).reduce((value, element) {
|
|
|
|
num n = max(value.getStatByEnum(_chartsY), element.getStatByEnum(_chartsY));
|
|
|
|
if (value.getStatByEnum(_chartsY) == n) {
|
|
|
|
return value;
|
|
|
|
} else {
|
|
|
|
return element;
|
|
|
|
}
|
|
|
|
}).getStatByEnum(_chartsY) as double;
|
|
|
|
}
|
|
|
|
|
|
|
|
void resetScale(){
|
|
|
|
maxX = actualMaxX;
|
|
|
|
minX = actualMinX;
|
|
|
|
maxY = actualMaxY;
|
|
|
|
minY = actualMinY;
|
|
|
|
recalculateScales();
|
|
|
|
}
|
|
|
|
|
|
|
|
void recalculateScales(){
|
|
|
|
xScale = maxX - minX;
|
|
|
|
yScale = maxY - minY;
|
2023-07-29 18:01:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
_tabController.dispose();
|
|
|
|
_scrollController.dispose();
|
2023-10-26 22:38:03 +00:00
|
|
|
if (!kIsWeb && !Platform.isAndroid && !Platform.isIOS) windowManager.setTitle(_oldWindowTitle);
|
2023-07-29 18:01:49 +00:00
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
2024-01-22 18:00:24 +00:00
|
|
|
void dragHandler(DragUpdateDetails dragUpdDet){
|
|
|
|
setState(() {
|
|
|
|
minX -= (xScale / dragFactor) * dragUpdDet.delta.dx;
|
|
|
|
maxX -= (xScale / dragFactor) * dragUpdDet.delta.dx;
|
|
|
|
minY += (yScale / dragFactor) * dragUpdDet.delta.dy;
|
|
|
|
maxY += (yScale / dragFactor) * dragUpdDet.delta.dy;
|
|
|
|
|
|
|
|
if (minX < actualMinX) {
|
|
|
|
minX = actualMinX;
|
|
|
|
maxX = actualMinX + xScale;
|
|
|
|
}
|
|
|
|
if (maxX > actualMaxX) {
|
|
|
|
maxX = actualMaxX;
|
|
|
|
minX = maxX - xScale;
|
|
|
|
}
|
|
|
|
if(minY < actualMinY){
|
|
|
|
minY = actualMinY;
|
|
|
|
maxY = actualMinY + yScale;
|
|
|
|
}
|
|
|
|
if(maxY > actualMaxY){
|
|
|
|
maxY = actualMaxY;
|
|
|
|
minY = actualMaxY - yScale;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-07-29 18:01:49 +00:00
|
|
|
void _justUpdate() {
|
|
|
|
setState(() {});
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2024-01-22 18:00:24 +00:00
|
|
|
GlobalKey graphKey = GlobalKey();
|
2023-07-29 18:01:49 +00:00
|
|
|
bool bigScreen = MediaQuery.of(context).size.width > 768;
|
2024-01-22 18:00:24 +00:00
|
|
|
EdgeInsets padding = bigScreen ? const EdgeInsets.fromLTRB(40, 40, 40, 48) : const EdgeInsets.fromLTRB(0, 40, 16, 48);
|
|
|
|
double graphStartX = padding.left;
|
|
|
|
double graphEndX = MediaQuery.sizeOf(context).width - padding.right;
|
|
|
|
if (previousAxisTitles != _chartsX.toString()+_chartsY.toString()){
|
|
|
|
recalculateBoundaries();
|
|
|
|
resetScale();
|
|
|
|
previousAxisTitles = _chartsX.toString()+_chartsY.toString();
|
2024-01-26 20:56:24 +00:00
|
|
|
}
|
2024-01-22 18:00:24 +00:00
|
|
|
final t = Translations.of(context);
|
2023-10-26 22:38:03 +00:00
|
|
|
List<TetrioPlayerFromLeaderboard> they = TetrioPlayersLeaderboard("lol", []).getStatRanking(widget.rank[1]["entries"]!, _sortBy, reversed: _reversed, country: _country);
|
2023-07-29 18:01:49 +00:00
|
|
|
return Scaffold(
|
|
|
|
appBar: AppBar(
|
2023-08-20 21:57:01 +00:00
|
|
|
title: Text(widget.rank[1]["everyone"] ? t.everyoneAverages : t.rankAverages(rank: widget.rank[0].rank.toUpperCase())),
|
2023-07-29 18:01:49 +00:00
|
|
|
),
|
|
|
|
backgroundColor: Colors.black,
|
|
|
|
body: SafeArea(
|
|
|
|
child: NestedScrollView(
|
|
|
|
controller: _scrollController,
|
|
|
|
headerSliverBuilder: (context, value) {
|
2024-01-22 18:00:24 +00:00
|
|
|
return [ SliverToBoxAdapter(
|
|
|
|
child: Column(
|
2023-07-29 18:01:49 +00:00
|
|
|
children: [
|
|
|
|
Flex(
|
|
|
|
direction: Axis.vertical,
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
|
|
children: [
|
|
|
|
Stack(
|
|
|
|
alignment: Alignment.topCenter,
|
2023-09-03 18:25:59 +00:00
|
|
|
children: [Image.asset("res/tetrio_tl_alpha_ranks/${widget.rank[0].rank}.png",fit: BoxFit.fitHeight,height: 128), ],
|
2023-07-29 18:01:49 +00:00
|
|
|
),
|
|
|
|
Flexible(
|
|
|
|
child: Column(
|
|
|
|
children: [
|
|
|
|
Text(
|
2023-08-20 21:57:01 +00:00
|
|
|
widget.rank[1]["everyone"] ? t.everyoneAverages : t.rankAverages(rank: widget.rank[0].rank.toUpperCase()),
|
2023-07-29 18:01:49 +00:00
|
|
|
style: TextStyle(
|
|
|
|
fontFamily: "Eurostile Round Extended",
|
|
|
|
fontSize: bigScreen ? 42 : 28)),
|
|
|
|
Text(
|
2023-08-20 21:57:01 +00:00
|
|
|
t.players(n: widget.rank[1]["entries"].length),
|
2023-07-29 18:01:49 +00:00
|
|
|
style: TextStyle(
|
|
|
|
fontFamily: "Eurostile Round Extended",
|
|
|
|
fontSize: bigScreen ? 42 : 28)),
|
|
|
|
],
|
|
|
|
)),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
)),
|
|
|
|
SliverToBoxAdapter(
|
|
|
|
child: TabBar(
|
|
|
|
controller: _tabController,
|
|
|
|
isScrollable: true,
|
2023-08-20 21:57:01 +00:00
|
|
|
tabs: [
|
|
|
|
Tab(text: t.chart),
|
|
|
|
Tab(text: t.entries),
|
|
|
|
Tab(text: t.minimums),
|
|
|
|
Tab(text: t.averages),
|
|
|
|
Tab(text: t.maximums),
|
|
|
|
Tab(text: t.other),
|
2023-07-29 18:01:49 +00:00
|
|
|
],
|
|
|
|
)),
|
|
|
|
];
|
|
|
|
},
|
|
|
|
body: TabBarView(
|
|
|
|
controller: _tabController,
|
|
|
|
children: [
|
|
|
|
Column(
|
|
|
|
children: [
|
2023-08-20 21:57:01 +00:00
|
|
|
Wrap(
|
|
|
|
direction: Axis.horizontal,
|
|
|
|
alignment: WrapAlignment.center,
|
|
|
|
spacing: 25,
|
2023-08-19 23:20:06 +00:00
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
Column(
|
2023-08-19 23:20:06 +00:00
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
2024-01-22 18:00:24 +00:00
|
|
|
const Padding(
|
|
|
|
padding: EdgeInsets.all(8.0),
|
|
|
|
child: Text("X:", style: TextStyle(fontSize: 22))),
|
2023-09-03 18:25:59 +00:00
|
|
|
DropdownButton(
|
2023-10-26 22:38:03 +00:00
|
|
|
items: _chartsShortTitlesDropdowns,
|
|
|
|
value: _chartsX,
|
2023-09-03 18:25:59 +00:00
|
|
|
onChanged: (value) {
|
2023-10-26 22:38:03 +00:00
|
|
|
_chartsX = value;
|
2023-09-03 18:25:59 +00:00
|
|
|
_justUpdate();
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
),
|
2023-08-19 23:20:06 +00:00
|
|
|
],
|
|
|
|
),
|
2023-09-03 18:25:59 +00:00
|
|
|
Column(
|
2023-08-19 23:20:06 +00:00
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
2024-01-22 18:00:24 +00:00
|
|
|
const Padding(
|
|
|
|
padding: EdgeInsets.all(8.0),
|
|
|
|
child: Text("Y:", style: TextStyle(fontSize: 22)),
|
2023-08-19 23:20:06 +00:00
|
|
|
),
|
2023-09-03 18:25:59 +00:00
|
|
|
DropdownButton(
|
2023-10-26 22:38:03 +00:00
|
|
|
items: _chartsShortTitlesDropdowns,
|
|
|
|
value: _chartsY,
|
2023-09-03 18:25:59 +00:00
|
|
|
onChanged: (value) {
|
2023-10-26 22:38:03 +00:00
|
|
|
_chartsY = value;
|
2023-09-03 18:25:59 +00:00
|
|
|
_justUpdate();
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
),
|
2023-08-19 23:20:06 +00:00
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
2023-09-03 18:25:59 +00:00
|
|
|
),
|
|
|
|
if (widget.rank[1]["entries"].length > 1)
|
|
|
|
SizedBox(
|
|
|
|
width: MediaQuery.of(context).size.width,
|
|
|
|
height: MediaQuery.of(context).size.height - 104,
|
2024-01-22 18:00:24 +00:00
|
|
|
child: Listener(
|
|
|
|
behavior: HitTestBehavior.translucent,
|
|
|
|
onPointerSignal: (signal) {
|
|
|
|
if (signal is PointerScrollEvent) {
|
|
|
|
RenderBox graphBox = graphKey.currentContext?.findRenderObject() as RenderBox;
|
|
|
|
Offset graphPosition = graphBox.localToGlobal(Offset.zero);
|
|
|
|
double scrollPosRelativeX = (signal.position.dx - graphStartX) / (graphEndX - graphStartX);
|
|
|
|
double scrollPosRelativeY = (signal.position.dy - graphPosition.dy) / (graphBox.size.height - 30); // size - bottom titles height
|
|
|
|
double newMinX, newMaxX, newMinY, newMaxY;
|
|
|
|
newMinX = minX - (xScale / scaleFactor) * signal.scrollDelta.dy * scrollPosRelativeX;
|
|
|
|
newMaxX = maxX + (xScale / scaleFactor) * signal.scrollDelta.dy * (1-scrollPosRelativeX);
|
|
|
|
newMinY = minY - (yScale / scaleFactor) * signal.scrollDelta.dy * (1-scrollPosRelativeY);
|
|
|
|
newMaxY = maxY + (yScale / scaleFactor) * signal.scrollDelta.dy * scrollPosRelativeY;
|
|
|
|
if ((newMaxX - newMinX).isNegative) return;
|
|
|
|
if ((newMaxY - newMinY).isNegative) return;
|
|
|
|
setState(() {
|
|
|
|
minX = max(newMinX, actualMinX);
|
|
|
|
maxX = min(newMaxX, actualMaxX);
|
|
|
|
minY = max(newMinY, actualMinY);
|
|
|
|
maxY = min(newMaxY, actualMaxY);
|
|
|
|
recalculateScales();
|
|
|
|
_scrollController.jumpTo(_scrollController.position.maxScrollExtent - signal.scrollDelta.dy);
|
|
|
|
});
|
|
|
|
}},
|
|
|
|
child: GestureDetector(
|
|
|
|
behavior: HitTestBehavior.opaque,
|
|
|
|
onDoubleTap: () {
|
|
|
|
setState(() {
|
|
|
|
minX = actualMinX;
|
|
|
|
maxX = actualMaxX;
|
|
|
|
minY = actualMinY;
|
|
|
|
maxY = actualMaxY;
|
|
|
|
recalculateScales();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
// TODO: Figure out wtf is going on with gestures
|
|
|
|
child: Padding(
|
|
|
|
padding: bigScreen ? const EdgeInsets.fromLTRB(40, 40, 40, 48) : const EdgeInsets.fromLTRB(0, 40, 16, 48),
|
|
|
|
child: Stack(
|
|
|
|
children: [
|
|
|
|
ScatterChart(
|
|
|
|
key: graphKey,
|
|
|
|
ScatterChartData(
|
|
|
|
minX: minX,
|
|
|
|
maxX: maxX,
|
|
|
|
minY: minY,
|
|
|
|
maxY: maxY,
|
|
|
|
clipData: const FlClipData.all(),
|
|
|
|
scatterSpots: [
|
|
|
|
for (TetrioPlayerFromLeaderboard entry in widget.rank[1]["entries"])
|
|
|
|
if (entry.apm != 0.0 && entry.vs != 0.0) // prevents from ScatterChart "Offset argument contained a NaN value." exception
|
|
|
|
_MyScatterSpot(
|
|
|
|
entry.getStatByEnum(_chartsX) as double,
|
|
|
|
entry.getStatByEnum(_chartsY) as double,
|
|
|
|
entry.userId,
|
|
|
|
entry.username,
|
|
|
|
dotPainter: FlDotCirclePainter(color: rankColors[entry.rank]??Colors.white, radius: 3))
|
|
|
|
],
|
|
|
|
scatterTouchData: ScatterTouchData(
|
|
|
|
handleBuiltInTouches: false,
|
|
|
|
touchCallback:(touchEvent, touchResponse) {
|
|
|
|
if (touchEvent is FlPanUpdateEvent){
|
|
|
|
dragHandler(touchEvent.details);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (touchEvent is FlPointerHoverEvent){
|
|
|
|
setState(() {
|
|
|
|
if (touchResponse?.touchedSpot == null) {
|
|
|
|
hoveredPointId = -1;
|
|
|
|
} else {
|
|
|
|
hoveredPointId = touchResponse!.touchedSpot!.spotIndex;
|
|
|
|
_MyScatterSpot castedPoint = touchResponse.touchedSpot!.spot as _MyScatterSpot;
|
|
|
|
headerTooltip = castedPoint.nickname;
|
|
|
|
footerTooltip = "${_f4.format(castedPoint.x)} ${chartsShortTitles[_chartsX]}; ${_f4.format(castedPoint.y)} ${chartsShortTitles[_chartsY]}";
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (touchEvent is FlPointerExitEvent){
|
|
|
|
setState(() {hoveredPointId = -1;});
|
|
|
|
}
|
|
|
|
if (touchEvent is FlTapUpEvent && touchResponse?.touchedSpot?.spot != null){
|
|
|
|
_MyScatterSpot spot = touchResponse!.touchedSpot!.spot as _MyScatterSpot;
|
|
|
|
Navigator.push(context, MaterialPageRoute(builder: (context) => MainView(player: spot.nickname), maintainState: false));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
swapAnimationDuration: const Duration(milliseconds: 150), // Optional
|
|
|
|
swapAnimationCurve: Curves.linear, // Optional
|
2023-09-03 18:25:59 +00:00
|
|
|
),
|
2024-01-22 18:00:24 +00:00
|
|
|
Padding(
|
|
|
|
padding: EdgeInsets.fromLTRB(graphStartX+8, padding.top/2+8, 0, 0),
|
|
|
|
child: Column(
|
|
|
|
children: [
|
|
|
|
AnimatedDefaultTextStyle(style: TextStyle(fontFamily: "Eurostile Round Extended", fontSize: 24, color: Color.fromARGB(hoveredPointId == -1 ? 100 : 255, 255, 255, 255), shadows: hoveredPointId != -1 ? textShadow : null), duration: Durations.medium1, curve: Curves.elasticInOut, child: Text(headerTooltip)),
|
|
|
|
AnimatedDefaultTextStyle(style: TextStyle(fontFamily: "Eurostile Round", color: Color.fromARGB(hoveredPointId == -1 ? 100 : 255, 255, 255, 255), shadows: hoveredPointId != -1 ? textShadow : null), duration: Durations.medium1, curve: Curves.elasticInOut, child: Text(footerTooltip)),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
)
|
|
|
|
],
|
2023-09-03 18:25:59 +00:00
|
|
|
),
|
|
|
|
),
|
2024-01-22 18:00:24 +00:00
|
|
|
),
|
2023-09-03 18:25:59 +00:00
|
|
|
))
|
2023-08-19 23:20:06 +00:00
|
|
|
else Center(child: Text(t.notEnoughData, style: const TextStyle(fontFamily: "Eurostile Round Extended", fontSize: 28)))
|
2023-07-29 18:01:49 +00:00
|
|
|
],
|
|
|
|
),
|
|
|
|
Column(
|
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
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)),
|
|
|
|
DropdownButton(
|
2023-10-26 22:38:03 +00:00
|
|
|
items: _itemStats,
|
|
|
|
value: _sortBy,
|
2023-09-03 18:25:59 +00:00
|
|
|
onChanged: ((value) {
|
2023-10-26 22:38:03 +00:00
|
|
|
_sortBy = value;
|
2023-09-03 18:25:59 +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(
|
2023-10-26 22:38:03 +00:00
|
|
|
value: _reversed,
|
2023-09-03 18:25:59 +00:00
|
|
|
checkColor: Colors.black,
|
|
|
|
onChanged: ((value) {
|
2023-10-26 22:38:03 +00:00
|
|
|
_reversed = value!;
|
2023-09-03 18:25:59 +00:00
|
|
|
setState(() {});
|
|
|
|
}),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.baseline,
|
|
|
|
textBaseline: TextBaseline.alphabetic,
|
|
|
|
children: [
|
|
|
|
Text("${t.country}: ", style: const TextStyle(color: Colors.white, fontSize: 25)),
|
|
|
|
DropdownButton(
|
2023-10-26 22:38:03 +00:00
|
|
|
items: _itemCountries,
|
|
|
|
value: _country,
|
2023-09-03 18:25:59 +00:00
|
|
|
onChanged: ((value) {
|
2023-10-26 22:38:03 +00:00
|
|
|
_country = value;
|
2023-09-03 18:25:59 +00:00
|
|
|
setState(() {});
|
|
|
|
}),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
2023-07-29 18:01:49 +00:00
|
|
|
Expanded(
|
|
|
|
child: ListView.builder(
|
2023-09-03 18:25:59 +00:00
|
|
|
itemCount: they.length,
|
2023-07-29 18:01:49 +00:00
|
|
|
itemBuilder: (context, index) {
|
2023-09-03 18:25:59 +00:00
|
|
|
bool bigScreen = MediaQuery.of(context).size.width > 768;
|
2023-07-29 18:01:49 +00:00
|
|
|
return ListTile(
|
2023-09-03 18:25:59 +00:00
|
|
|
title: Text(they[index].username, style: const TextStyle(fontFamily: "Eurostile Round Extended")),
|
2023-10-26 22:38:03 +00:00
|
|
|
subtitle: Text(_sortBy == Stats.tr ? "${_f2.format(they[index].apm)} APM, ${_f2.format(they[index].pps)} PPS, ${_f2.format(they[index].vs)} VS, ${_f2.format(they[index].nerdStats.app)} APP, ${_f2.format(they[index].nerdStats.vsapm)} VS/APM" : "${_f4.format(they[index].getStatByEnum(_sortBy))} ${chartsShortTitles[_sortBy]}"),
|
2023-07-29 18:01:49 +00:00
|
|
|
trailing: Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
2023-10-26 22:38:03 +00:00
|
|
|
Text("${_f2.format(they[index].rating)} TR", style: bigScreen ? const TextStyle(fontSize: 28) : null),
|
2023-09-03 18:25:59 +00:00
|
|
|
Image.asset("res/tetrio_tl_alpha_ranks/${they[index].rank}.png", height: bigScreen ? 48 : 16),
|
2023-07-29 18:01:49 +00:00
|
|
|
],
|
|
|
|
),
|
|
|
|
onTap: () {
|
2023-09-04 18:00:13 +00:00
|
|
|
Navigator.push(context, MaterialPageRoute(builder: (context) => MainView(player: they[index].username), maintainState: false));
|
2023-07-29 18:01:49 +00:00
|
|
|
},
|
|
|
|
);
|
|
|
|
}),
|
|
|
|
)
|
|
|
|
],
|
|
|
|
),
|
|
|
|
Column(
|
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
Text(t.lowestValues, style: TextStyle( fontFamily: "Eurostile Round Extended", fontSize: bigScreen ? 42 : 28)),
|
2023-07-29 18:01:49 +00:00
|
|
|
Expanded(
|
|
|
|
child: ListView(
|
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["lowestTR"], label: t.statCellNum.tr.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestTRid"], username: widget.rank[1]["lowestTRnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestGlicko"], label: "Glicko", id: widget.rank[1]["lowestGlickoID"], username: widget.rank[1]["lowestGlickoNick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestRD"], label: t.statCellNum.rd.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestRdID"], username: widget.rank[1]["lowestRdNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestGamesPlayed"], label: t.statCellNum.gamesPlayed.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestGamesPlayedID"], username: widget.rank[1]["lowestGamesPlayedNick"], approximate: false),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestGamesWon"], label: t.statCellNum.gamesWonTL.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestGamesWonID"], username: widget.rank[1]["lowestGamesWonNick"], approximate: false),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestWinrate"] * 100, label: t.statCellNum.winrate.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestWinrateID"], username: widget.rank[1]["lowestWinrateNick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestAPM"], label: t.statCellNum.apm.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestAPMid"], username: widget.rank[1]["lowestAPMnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestPPS"], label: t.statCellNum.pps.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestPPSid"], username: widget.rank[1]["lowestPPSnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestVS"], label: t.statCellNum.vs.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestVSid"], username: widget.rank[1]["lowestVSnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestAPP"], label: t.statCellNum.app.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestAPPid"], username: widget.rank[1]["lowestAPPnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestVSAPM"], label: "VS / APM", id: widget.rank[1]["lowestVSAPMid"], username: widget.rank[1]["lowestVSAPMnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestDSS"], label: t.statCellNum.dss.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestDSSid"], username: widget.rank[1]["lowestDSSnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestDSP"], label: t.statCellNum.dsp.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestDSPid"], username: widget.rank[1]["lowestDSPnick"], approximate: false, fractionDigits: 3),
|
2023-09-23 19:09:36 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["lowestAPPDSP"], label: t.statCellNum.appdsp.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestAPPDSPid"], username: widget.rank[1]["lowestAPPDSPnick"], approximate: false, fractionDigits: 3),
|
2023-09-03 18:25:59 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["lowestCheese"], label: t.statCellNum.cheese.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestCheeseID"], username: widget.rank[1]["lowestCheeseNick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestGBE"], label: t.statCellNum.gbe.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestGBEid"], username: widget.rank[1]["lowestGBEnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestNyaAPP"], label: t.statCellNum.nyaapp.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestNyaAPPid"], username: widget.rank[1]["lowestNyaAPPnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestArea"], label: t.statCellNum.area.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestAreaID"], username: widget.rank[1]["lowestAreaNick"], approximate: false, fractionDigits: 1),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestEstTR"], label: t.statCellNum.estOfTR.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestEstTRid"], username: widget.rank[1]["lowestEstTRnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestEstAcc"], label: t.statCellNum.accOfEst.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["lowestEstAccID"], username: widget.rank[1]["lowestEstAccNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestOpener"], label: "Opener", id: widget.rank[1]["lowestOpenerID"], username: widget.rank[1]["lowestOpenerNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestPlonk"], label: "Plonk", id: widget.rank[1]["lowestPlonkID"], username: widget.rank[1]["lowestPlonkNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestStride"], label: "Stride", id: widget.rank[1]["lowestStrideID"], username: widget.rank[1]["lowestStrideNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["lowestInfDS"], label: "Inf. DS", id: widget.rank[1]["lowestInfDSid"], username: widget.rank[1]["lowestInfDSnick"], approximate: false, fractionDigits: 3)
|
2023-07-29 18:01:49 +00:00
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
Column(
|
2023-08-08 19:30:24 +00:00
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
Text(t.averageValues, style: TextStyle( fontFamily: "Eurostile Round Extended", fontSize: bigScreen ? 42 : 28)),
|
2023-08-08 19:30:24 +00:00
|
|
|
Expanded(
|
2023-08-14 21:26:20 +00:00
|
|
|
child: ListView(children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
_ListEntry(value: widget.rank[0].rating, label: t.statCellNum.tr.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[0].glicko, label: "Glicko", id: "", username: "", approximate: true, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[0].rd, label: t.statCellNum.rd.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[0].gamesPlayed, label: t.statCellNum.gamesPlayed.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 0),
|
|
|
|
_ListEntry(value: widget.rank[0].gamesWon, label: t.statCellNum.gamesWonTL.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 0),
|
|
|
|
_ListEntry(value: widget.rank[0].winrate * 100, label: t.statCellNum.winrate.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[0].apm, label: t.statCellNum.apm.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[0].pps, label: t.statCellNum.pps.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[0].vs, label: t.statCellNum.vs.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgAPP"], label: t.statCellNum.app.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
2023-10-07 16:44:54 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["avgVSAPM"], label: "VS / APM", id: "", username: "", approximate: true, fractionDigits: 3),
|
2023-09-03 18:25:59 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["avgDSS"], label: t.statCellNum.dss.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgDSP"], label: t.statCellNum.dsp.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
2023-09-23 19:09:36 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["avgAPPDSP"], label: t.statCellNum.appdsp.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
2023-09-03 18:25:59 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["avgCheese"], label: t.statCellNum.cheese.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgGBE"], label: t.statCellNum.gbe.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgNyaAPP"], label: t.statCellNum.nyaapp.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgArea"], label: t.statCellNum.area.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 1),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgEstTR"], label: t.statCellNum.estOfTR.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgEstAcc"], label: t.statCellNum.accOfEst.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgOpener"], label: "Opener", id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgPlonk"], label: "Plonk", id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgStride"], label: "Stride", id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["avgInfDS"], label: "Inf. DS", id: "", username: "", approximate: true, fractionDigits: 3),
|
2023-08-14 21:26:20 +00:00
|
|
|
]))
|
|
|
|
],
|
2023-07-29 18:01:49 +00:00
|
|
|
),
|
|
|
|
Column(
|
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
Text(t.highestValues, style: TextStyle(fontFamily: "Eurostile Round Extended", fontSize: bigScreen ? 42 : 28)),
|
2023-07-29 18:01:49 +00:00
|
|
|
Expanded(
|
|
|
|
child: ListView(
|
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["highestTR"], label: t.statCellNum.tr.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestTRid"], username: widget.rank[1]["highestTRnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestGlicko"], label: "Glicko", id: widget.rank[1]["highestGlickoID"], username: widget.rank[1]["highestGlickoNick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestRD"], label: t.statCellNum.rd.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestRdID"], username: widget.rank[1]["highestRdNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestGamesPlayed"], label: t.statCellNum.gamesPlayed.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestGamesPlayedID"], username: widget.rank[1]["highestGamesPlayedNick"], approximate: false),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestGamesWon"], label: t.statCellNum.gamesWonTL.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestGamesWonID"], username: widget.rank[1]["highestGamesWonNick"], approximate: false),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestWinrate"] * 100, label: t.statCellNum.winrate.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestWinrateID"], username: widget.rank[1]["highestWinrateNick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestAPM"], label: t.statCellNum.apm.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestAPMid"], username: widget.rank[1]["highestAPMnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestPPS"], label: t.statCellNum.pps.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestPPSid"], username: widget.rank[1]["highestPPSnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestVS"], label: t.statCellNum.vs.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestVSid"], username: widget.rank[1]["highestVSnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestAPP"], label: t.statCellNum.app.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestAPPid"], username: widget.rank[1]["highestAPPnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestVSAPM"], label: "VS / APM", id: widget.rank[1]["highestVSAPMid"], username: widget.rank[1]["highestVSAPMnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestDSS"], label: t.statCellNum.dss.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestDSSid"], username: widget.rank[1]["highestDSSnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestDSP"], label: t.statCellNum.dsp.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestDSPid"], username: widget.rank[1]["highestDSPnick"], approximate: false, fractionDigits: 3),
|
2023-09-23 19:09:36 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["highestAPPDSP"], label: t.statCellNum.appdsp.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestAPPDSPid"], username: widget.rank[1]["highestAPPDSPnick"], approximate: false, fractionDigits: 3),
|
2023-09-03 18:25:59 +00:00
|
|
|
_ListEntry(value: widget.rank[1]["highestCheese"], label: t.statCellNum.cheese.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestCheeseID"], username: widget.rank[1]["highestCheeseNick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestGBE"], label: t.statCellNum.gbe.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestGBEid"], username: widget.rank[1]["highestGBEnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestNyaAPP"], label: t.statCellNum.nyaapp.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestNyaAPPid"], username: widget.rank[1]["highestNyaAPPnick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestArea"], label: t.statCellNum.area.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestAreaID"], username: widget.rank[1]["highestAreaNick"], approximate: false, fractionDigits: 1),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestEstTR"], label: t.statCellNum.estOfTR.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestEstTRid"], username: widget.rank[1]["highestEstTRnick"], approximate: false, fractionDigits: 2),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestEstAcc"], label: t.statCellNum.accOfEst.replaceAll(RegExp(r'\n'), " "), id: widget.rank[1]["highestEstAccID"], username: widget.rank[1]["highestEstAccNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestOpener"], label: "Opener", id: widget.rank[1]["highestOpenerID"], username: widget.rank[1]["highestOpenerNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestPlonk"], label: "Plonk", id: widget.rank[1]["highestPlonkID"], username: widget.rank[1]["highestPlonkNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestStride"], label: "Stride", id: widget.rank[1]["highestStrideID"], username: widget.rank[1]["highestStrideNick"], approximate: false, fractionDigits: 3),
|
|
|
|
_ListEntry(value: widget.rank[1]["highestInfDS"], label: "Inf. DS", id: widget.rank[1]["highestInfDSid"], username: widget.rank[1]["highestInfDSnick"], approximate: false, fractionDigits: 3),
|
2023-07-29 18:01:49 +00:00
|
|
|
],
|
|
|
|
),
|
|
|
|
)
|
|
|
|
],
|
|
|
|
),
|
|
|
|
Column(
|
2023-08-14 21:26:20 +00:00
|
|
|
children: [
|
2023-09-03 18:25:59 +00:00
|
|
|
Expanded(
|
|
|
|
child: ListView(children: [
|
|
|
|
_ListEntry(value: widget.rank[1]["totalGamesPlayed"], label: t.statCellNum.totalGames, id: "", username: "", approximate: true, fractionDigits: 0),
|
|
|
|
_ListEntry(value: widget.rank[1]["totalGamesWon"], label: t.statCellNum.totalWon, id: "", username: "", approximate: true, fractionDigits: 0),
|
|
|
|
_ListEntry(value: (widget.rank[1]["totalGamesWon"] / widget.rank[1]["totalGamesPlayed"]) * 100, label: t.statCellNum.winrate.replaceAll(RegExp(r'\n'), " "), id: "", username: "", approximate: true, fractionDigits: 3),
|
|
|
|
]))
|
2023-08-14 21:26:20 +00:00
|
|
|
],
|
2023-07-29 18:01:49 +00:00
|
|
|
),
|
|
|
|
],
|
|
|
|
))));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _ListEntry extends StatelessWidget {
|
|
|
|
final num value;
|
|
|
|
final String label;
|
|
|
|
final String id;
|
|
|
|
final String username;
|
|
|
|
final bool approximate;
|
|
|
|
final int? fractionDigits;
|
|
|
|
const _ListEntry(
|
|
|
|
{required this.value,
|
|
|
|
required this.label,
|
|
|
|
this.fractionDigits,
|
|
|
|
required this.id,
|
|
|
|
required this.username,
|
|
|
|
required this.approximate});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
NumberFormat f = NumberFormat.decimalPatternDigits(
|
|
|
|
locale: LocaleSettings.currentLocale.languageCode,
|
|
|
|
decimalDigits: fractionDigits ?? 0);
|
|
|
|
return ListTile(
|
|
|
|
title: Text(label),
|
|
|
|
trailing: Column(
|
2023-08-08 19:30:24 +00:00
|
|
|
crossAxisAlignment: CrossAxisAlignment.end,
|
2023-07-29 18:01:49 +00:00
|
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
|
|
children: [
|
|
|
|
Text(f.format(value),
|
|
|
|
style: const TextStyle(fontSize: 22, height: 0.9)),
|
2023-08-20 21:57:01 +00:00
|
|
|
if (id.isNotEmpty) Text(t.forPlayer(username: username))
|
2023-07-29 18:01:49 +00:00
|
|
|
],
|
|
|
|
),
|
|
|
|
onTap: id.isNotEmpty
|
|
|
|
? () {
|
|
|
|
Navigator.push(
|
|
|
|
context,
|
|
|
|
MaterialPageRoute(
|
|
|
|
builder: (context) => MainView(player: id),
|
|
|
|
maintainState: false,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
: null,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2023-09-03 18:25:59 +00:00
|
|
|
|
|
|
|
class _MyScatterSpot extends ScatterSpot {
|
2023-08-19 23:20:06 +00:00
|
|
|
String id;
|
|
|
|
String nickname;
|
2023-12-26 22:02:47 +00:00
|
|
|
//Color color;
|
|
|
|
//FlDotPainter painter = FlDotCirclePainter(color: color, radius: 2);
|
2024-01-13 18:49:36 +00:00
|
|
|
_MyScatterSpot(super.x, super.y, this.id, this.nickname, {super.dotPainter});
|
2023-09-03 18:25:59 +00:00
|
|
|
}
|