interface wrapping

This commit is contained in:
Mikkeli Matlock
2026-01-26 15:47:59 +09:00
parent 8f22966eb0
commit 46ac9d3123
4 changed files with 342 additions and 87 deletions

View File

@@ -2,38 +2,80 @@ import 'package:flutter/material.dart';
import '../theme/app_theme.dart';
/// A labeled stat display box for the dashboard
/// A labeled stat display box for the dashboard bottom row.
class StatBox extends StatelessWidget {
final String label;
final String value;
final String? unit;
final String label;
final int flex;
const StatBox({super.key, required this.label, required this.value});
const StatBox({
super.key,
required this.value,
this.unit,
required this.label,
this.flex = 1,
});
@override
Widget build(BuildContext context) {
final theme = AppTheme.of(context);
return Expanded(
flex: 1,
child: Column(
children: [
Text(
value,
style: Theme.of(context).textTheme.headlineMedium?.copyWith(
fontSize: 100,
color: theme.foreground,
),
),
Text(
label,
style: Theme.of(context).textTheme.bodySmall?.copyWith(
fontSize: 80,
color: theme.subdued,
letterSpacing: 1,
),
),
],
)
flex: flex,
child: LayoutBuilder(
builder: (context, constraints) {
final baseSize = constraints.maxHeight * 0.4;
return Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
// Value + optional unit
Row(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.baseline,
textBaseline: TextBaseline.alphabetic,
children: [
Text(
value,
style: TextStyle(
fontSize: baseSize,
fontWeight: FontWeight.w400,
fontFeatures: const [FontFeature.tabularFigures()],
color: theme.foreground,
height: 1,
),
),
SizedBox(width: baseSize * 0.1),
if (unit != null) ...[
const SizedBox(width: 4),
Text(
unit!,
style: TextStyle(
fontSize: baseSize * 0.5,
fontWeight: FontWeight.w400,
color: theme.subdued,
height: 1,
),
),
],
],
),
SizedBox(height: baseSize * 0.1),
// Label
Text(
label,
style: TextStyle(
fontSize: baseSize * 0.6,
fontWeight: FontWeight.w400,
color: theme.subdued,
letterSpacing: 1,
),
),
],
);
},
),
);
}
}