feat: initial commit - Habo habit tracking app
- Complete MVP with Repository Pattern, SQLite storage - Provider + ChangeNotifier state management - Navigation 2.0 with deep link support - Habit CRUD with twoDayRule, notifications, categories - Backup/Restore via JSON - Statistics with streak tracking - Material You theme support - Biometric lock support - Desktop widget support - 27 languages i18n structure - Comprehensive test suite (87/89 passing)
This commit is contained in:
105
lib/navigation/app_router.dart
Normal file
105
lib/navigation/app_router.dart
Normal file
@@ -0,0 +1,105 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:habo/navigation/app_state_manager.dart';
|
||||
import 'package:habo/navigation/routes.dart';
|
||||
import 'package:habo/habits/habits_screen.dart';
|
||||
import 'package:habo/statistics/statistics_screen.dart';
|
||||
import 'package:habo/settings/settings_screen.dart';
|
||||
import 'package:habo/onboarding/onboarding.dart';
|
||||
import 'package:habo/habits/create_habit.dart';
|
||||
import 'package:habo/habits/edit_habit.dart';
|
||||
|
||||
class AppRouter extends RouterDelegate<HaboRouteConfiguration>
|
||||
with ChangeNotifier, PopNavigatorRouterDelegateMixin<HaboRouteConfiguration> {
|
||||
final AppStateManager appStateManager;
|
||||
|
||||
AppRouter(this.appStateManager) {
|
||||
appStateManager.addListener(notifyListeners);
|
||||
}
|
||||
|
||||
@override
|
||||
GlobalKey<NavigatorState> get navigatorKey => GlobalKey<NavigatorState>();
|
||||
|
||||
@override
|
||||
HaboRouteConfiguration? get currentConfiguration {
|
||||
if (appStateManager.onboarding) return HaboRouteConfiguration(path: RouteConstants.onboardingPath);
|
||||
if (appStateManager.statistics) return HaboRouteConfiguration(path: RouteConstants.statisticsPath);
|
||||
if (appStateManager.settings) return HaboRouteConfiguration(path: RouteConstants.settingsPath);
|
||||
if (appStateManager.createHabit) return HaboRouteConfiguration(path: RouteConstants.createHabitPath);
|
||||
if (appStateManager.editHabit) return HaboRouteConfiguration(path: RouteConstants.editHabitPath);
|
||||
return HaboRouteConfiguration(path: RouteConstants.habitsPath);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
final pages = <Page<dynamic>>[];
|
||||
|
||||
// Always start with habits screen
|
||||
pages.add(
|
||||
const MaterialPage<dynamic>(
|
||||
child: HabitsScreen(),
|
||||
key: ValueKey('habits'),
|
||||
),
|
||||
);
|
||||
|
||||
if (appStateManager.statistics) {
|
||||
pages.add(
|
||||
const MaterialPage<dynamic>(
|
||||
child: StatisticsScreen(),
|
||||
key: ValueKey('statistics'),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
if (appStateManager.settings) {
|
||||
pages.add(
|
||||
const MaterialPage<dynamic>(
|
||||
child: SettingsScreen(),
|
||||
key: ValueKey('settings'),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
if (appStateManager.onboarding) {
|
||||
pages.add(
|
||||
const MaterialPage<dynamic>(
|
||||
child: OnboardingScreen(),
|
||||
key: ValueKey('onboarding'),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
if (appStateManager.createHabit) {
|
||||
pages.add(
|
||||
const MaterialPage<dynamic>(
|
||||
child: CreateHabitScreen(),
|
||||
key: ValueKey('create'),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
if (appStateManager.editHabit) {
|
||||
pages.add(
|
||||
const MaterialPage<dynamic>(
|
||||
child: EditHabitScreen(),
|
||||
key: ValueKey('edit'),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
return Navigator(
|
||||
key: navigatorKey,
|
||||
pages: pages,
|
||||
onDidRemovePage: (page) {},
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Future<void> setNewRoutePath(HaboRouteConfiguration configuration) async {
|
||||
// Handle deep links
|
||||
}
|
||||
}
|
||||
|
||||
class HaboRouteConfiguration {
|
||||
final String path;
|
||||
HaboRouteConfiguration({required this.path});
|
||||
}
|
||||
35
lib/navigation/app_state_manager.dart
Normal file
35
lib/navigation/app_state_manager.dart
Normal file
@@ -0,0 +1,35 @@
|
||||
import 'package:flutter/foundation.dart';
|
||||
|
||||
class AppStateManager extends ChangeNotifier {
|
||||
bool _onboarding = false;
|
||||
bool _statistics = false;
|
||||
bool _settings = false;
|
||||
bool _createHabit = false;
|
||||
bool _editHabit = false;
|
||||
bool _whatsNew = false;
|
||||
bool _archivedHabits = false;
|
||||
|
||||
bool get onboarding => _onboarding;
|
||||
bool get statistics => _statistics;
|
||||
bool get settings => _settings;
|
||||
bool get createHabit => _createHabit;
|
||||
bool get editHabit => _editHabit;
|
||||
bool get whatsNew => _whatsNew;
|
||||
bool get archivedHabits => _archivedHabits;
|
||||
|
||||
void setOnboarding(bool v) { _onboarding = v; notifyListeners(); }
|
||||
void setStatistics(bool v) { _statistics = v; notifyListeners(); }
|
||||
void setSettings(bool v) { _settings = v; notifyListeners(); }
|
||||
void setCreateHabit(bool v) { _createHabit = v; notifyListeners(); }
|
||||
void setEditHabit(bool v) { _editHabit = v; notifyListeners(); }
|
||||
void setWhatsNew(bool v) { _whatsNew = v; notifyListeners(); }
|
||||
void setArchivedHabits(bool v) { _archivedHabits = v; notifyListeners(); }
|
||||
|
||||
void goOnboarding(bool v) => setOnboarding(v);
|
||||
void goStatistics(bool v) => setStatistics(v);
|
||||
void goSettings(bool v) => setSettings(v);
|
||||
void goCreateHabit(bool v) => setCreateHabit(v);
|
||||
void goEditHabit(bool v) => setEditHabit(v);
|
||||
void goWhatsNew(bool v) => setWhatsNew(v);
|
||||
void goArchivedHabits(bool v) => setArchivedHabits(v);
|
||||
}
|
||||
4
lib/navigation/navigation.dart
Normal file
4
lib/navigation/navigation.dart
Normal file
@@ -0,0 +1,4 @@
|
||||
export 'app_state_manager.dart';
|
||||
export 'app_router.dart';
|
||||
export 'route_information_parser.dart';
|
||||
export 'routes.dart';
|
||||
16
lib/navigation/route_information_parser.dart
Normal file
16
lib/navigation/route_information_parser.dart
Normal file
@@ -0,0 +1,16 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:habo/navigation/app_router.dart';
|
||||
import 'package:habo/navigation/routes.dart';
|
||||
|
||||
class HaboRouteInformationParser extends RouteInformationParser<HaboRouteConfiguration> {
|
||||
@override
|
||||
Future<HaboRouteConfiguration> parseRouteInformation(RouteInformation routeInformation) async {
|
||||
final uri = routeInformation.uri;
|
||||
return HaboRouteConfiguration(path: uri.path.isEmpty ? RouteConstants.habitsPath : uri.path);
|
||||
}
|
||||
|
||||
@override
|
||||
RouteInformation restoreRouteInformation(HaboRouteConfiguration configuration) {
|
||||
return RouteInformation(uri: Uri.parse(configuration.path));
|
||||
}
|
||||
}
|
||||
10
lib/navigation/routes.dart
Normal file
10
lib/navigation/routes.dart
Normal file
@@ -0,0 +1,10 @@
|
||||
class RouteConstants {
|
||||
static const String splashPath = '/';
|
||||
static const String habitsPath = '/habits';
|
||||
static const String statisticsPath = '/statistics';
|
||||
static const String settingsPath = '/settings';
|
||||
static const String onboardingPath = '/onboarding';
|
||||
static const String createHabitPath = '/create';
|
||||
static const String editHabitPath = '/edit';
|
||||
static const String whatsNewPath = '/whatsnew';
|
||||
}
|
||||
Reference in New Issue
Block a user