Implemented new nav bar with selected animation
All checks were successful
Pull Request Pipeline / test (pull_request) Successful in 2m3s
Pull Request Pipeline / lint (pull_request) Successful in 2m7s

This commit is contained in:
2026-01-09 21:12:09 +01:00
parent c2394c3733
commit 799b7d8403
4 changed files with 110 additions and 52 deletions

View File

@@ -11,6 +11,8 @@ class CustomTheme {
static Color onBoxColor = const Color(0xFF181818); static Color onBoxColor = const Color(0xFF181818);
static Color boxBorder = const Color(0xFF272727); static Color boxBorder = const Color(0xFF272727);
static const Color textColor = Colors.white; static const Color textColor = Colors.white;
static Color navBarItemSelectedColor = primaryColor.withValues(green: 0.4);
static Color navBarItemUnselectedColor = Colors.white.withValues(alpha: 0.6);
// ==================== Border Radius ==================== // ==================== Border Radius ====================
static const double standardBorderRadius = 12.0; static const double standardBorderRadius = 12.0;

View File

@@ -1,3 +1,5 @@
import 'dart:ui';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:game_tracker/core/custom_theme.dart'; import 'package:game_tracker/core/custom_theme.dart';
import 'package:game_tracker/l10n/generated/app_localizations.dart'; import 'package:game_tracker/l10n/generated/app_localizations.dart';
@@ -70,50 +72,49 @@ class _CustomNavigationBarState extends State<CustomNavigationBar>
backgroundColor: CustomTheme.backgroundColor, backgroundColor: CustomTheme.backgroundColor,
body: tabs[currentIndex], body: tabs[currentIndex],
extendBody: true, extendBody: true,
bottomNavigationBar: SafeArea( bottomNavigationBar: ClipRRect(
minimum: const EdgeInsets.only(bottom: 30), child: BackdropFilter(
child: Container( filter: ImageFilter.blur(sigmaX: 10, sigmaY: 10),
margin: const EdgeInsets.symmetric(horizontal: 12.0, vertical: 10), child: Container(
decoration: BoxDecoration( decoration: BoxDecoration(
borderRadius: BorderRadius.circular(24), color: CustomTheme.boxColor.withValues(alpha: 0.8),
color: CustomTheme.primaryColor, ),
), child: SafeArea(
child: ClipRRect( child: SizedBox(
borderRadius: BorderRadius.circular(24), height: 70,
child: SizedBox( child: Row(
height: 60, mainAxisAlignment: MainAxisAlignment.spaceAround,
child: Row( children: <Widget>[
mainAxisAlignment: MainAxisAlignment.spaceAround, NavbarItem(
children: <Widget>[ index: 0,
NavbarItem( isSelected: currentIndex == 0,
index: 0, icon: Icons.home_rounded,
isSelected: currentIndex == 0, label: loc.home,
icon: Icons.home_rounded, onTabTapped: onTabTapped,
label: loc.home, ),
onTabTapped: onTabTapped, NavbarItem(
), index: 1,
NavbarItem( isSelected: currentIndex == 1,
index: 1, icon: Icons.gamepad_rounded,
isSelected: currentIndex == 1, label: loc.matches,
icon: Icons.gamepad_rounded, onTabTapped: onTabTapped,
label: loc.matches, ),
onTabTapped: onTabTapped, NavbarItem(
), index: 2,
NavbarItem( isSelected: currentIndex == 2,
index: 2, icon: Icons.group_rounded,
isSelected: currentIndex == 2, label: loc.groups,
icon: Icons.group_rounded, onTabTapped: onTabTapped,
label: loc.groups, ),
onTabTapped: onTabTapped, NavbarItem(
), index: 3,
NavbarItem( isSelected: currentIndex == 3,
index: 3, icon: Icons.bar_chart_rounded,
isSelected: currentIndex == 3, label: loc.statistics,
icon: Icons.bar_chart_rounded, onTabTapped: onTabTapped,
label: loc.statistics, ),
onTabTapped: onTabTapped, ],
), ),
],
), ),
), ),
), ),

View File

@@ -1,4 +1,5 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:game_tracker/core/custom_theme.dart';
/// A navigation bar item widget that represents a single tab in a navigation bar. /// A navigation bar item widget that represents a single tab in a navigation bar.
/// - [index]: The index of the tab. /// - [index]: The index of the tab.
@@ -35,7 +36,45 @@ class NavbarItem extends StatefulWidget {
State<NavbarItem> createState() => _NavbarItemState(); State<NavbarItem> createState() => _NavbarItemState();
} }
class _NavbarItemState extends State<NavbarItem> { class _NavbarItemState extends State<NavbarItem>
with SingleTickerProviderStateMixin {
/// Animation controller for the scale animation
late AnimationController _animationController;
/// Scale animation for the icon when selected
late Animation<double> _scaleAnimation;
@override
void initState() {
super.initState();
_animationController = AnimationController(
duration: const Duration(milliseconds: 300),
vsync: this,
);
_scaleAnimation = Tween<double>(begin: 1.0, end: 1.2).animate(
CurvedAnimation(
parent: _animationController,
curve: Curves.easeInOutBack,
),
);
if (widget.isSelected) {
_animationController.forward();
}
}
// Retrigger animation on selection change
@override
void didUpdateWidget(NavbarItem oldWidget) {
super.didUpdateWidget(oldWidget);
if (widget.isSelected && !oldWidget.isSelected) {
_animationController.forward();
} else if (!widget.isSelected && oldWidget.isSelected) {
_animationController.reverse();
}
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Expanded( return Expanded(
@@ -48,19 +87,29 @@ class _NavbarItemState extends State<NavbarItem> {
mainAxisSize: MainAxisSize.min, mainAxisSize: MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
Icon( ScaleTransition(
widget.icon, scale: widget.isSelected
color: widget.isSelected ? Colors.white : Colors.black, ? _scaleAnimation
: const AlwaysStoppedAnimation(1.0),
child: Icon(
widget.icon,
color: widget.isSelected
? CustomTheme.navBarItemSelectedColor
: CustomTheme.navBarItemUnselectedColor,
size: 26,
),
), ),
const SizedBox(height: 4), const SizedBox(height: 4),
Text( Text(
widget.label, widget.label,
style: TextStyle( style: TextStyle(
color: widget.isSelected ? Colors.white : Colors.black, color: widget.isSelected
fontSize: 12, ? CustomTheme.navBarItemSelectedColor
: CustomTheme.navBarItemUnselectedColor,
fontSize: widget.isSelected ? 12 : 11,
fontWeight: widget.isSelected fontWeight: widget.isSelected
? FontWeight.bold ? FontWeight.bold
: FontWeight.normal, : FontWeight.w500,
), ),
), ),
], ],
@@ -69,4 +118,10 @@ class _NavbarItemState extends State<NavbarItem> {
), ),
); );
} }
@override
void dispose() {
_animationController.dispose();
super.dispose();
}
} }

View File

@@ -1,7 +1,7 @@
name: game_tracker name: game_tracker
description: "Game Tracking App for Card Games" description: "Game Tracking App for Card Games"
publish_to: 'none' publish_to: 'none'
version: 0.0.1+21 version: 0.0.1+69
environment: environment:
sdk: ^3.8.1 sdk: ^3.8.1