feat(01-01): add core infrastructure, localization, and Wave 0 tests

- AppDatabase with schemaVersion 1, in-memory executor for testing
- Database provider with @Riverpod(keepAlive: true)
- AppTheme with sage green seed ColorScheme and warm surface overrides
- ThemeNotifier with SharedPreferences persistence, defaults to system
- Full German ARB localization (15 keys) with proper umlauts
- Minimal main.dart with ProviderScope placeholder
- Drift schema v1 captured via make-migrations
- All .g.dart files generated via build_runner
- Wave 0 tests: database (3), color scheme (6), theme (3), localization (2) -- 14 total, all passing

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-03-15 19:59:44 +01:00
parent 4b27aeadf6
commit 51738f78bc
17 changed files with 784 additions and 149 deletions

View File

@@ -1,4 +1,27 @@
{
"@@locale": "de",
"appTitle": "HouseHoldKeaper"
"appTitle": "HouseHoldKeaper",
"tabHome": "\u00dcbersicht",
"tabRooms": "R\u00e4ume",
"tabSettings": "Einstellungen",
"homeEmptyTitle": "Noch nichts zu tun!",
"homeEmptyMessage": "Lege zuerst einen Raum an, um Aufgaben zu planen.",
"homeEmptyAction": "Raum erstellen",
"roomsEmptyTitle": "Hier ist noch alles leer!",
"roomsEmptyMessage": "Erstelle deinen ersten Raum, um loszulegen.",
"roomsEmptyAction": "Raum erstellen",
"settingsSectionAppearance": "Darstellung",
"settingsThemeLabel": "Farbschema",
"themeSystem": "System",
"themeLight": "Hell",
"themeDark": "Dunkel",
"settingsSectionAbout": "\u00dcber",
"aboutAppName": "HouseHoldKeaper",
"aboutTagline": "Dein Haushalt, entspannt organisiert.",
"aboutVersion": "Version {version}",
"@aboutVersion": {
"placeholders": {
"version": { "type": "String" }
}
}
}

View File

@@ -0,0 +1,242 @@
import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:flutter/widgets.dart';
import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:intl/intl.dart' as intl;
import 'app_localizations_de.dart';
// ignore_for_file: type=lint
/// Callers can lookup localized strings with an instance of AppLocalizations
/// returned by `AppLocalizations.of(context)`.
///
/// Applications need to include `AppLocalizations.delegate()` in their app's
/// `localizationDelegates` list, and the locales they support in the app's
/// `supportedLocales` list. For example:
///
/// ```dart
/// import 'l10n/app_localizations.dart';
///
/// return MaterialApp(
/// localizationsDelegates: AppLocalizations.localizationsDelegates,
/// supportedLocales: AppLocalizations.supportedLocales,
/// home: MyApplicationHome(),
/// );
/// ```
///
/// ## Update pubspec.yaml
///
/// Please make sure to update your pubspec.yaml to include the following
/// packages:
///
/// ```yaml
/// dependencies:
/// # Internationalization support.
/// flutter_localizations:
/// sdk: flutter
/// intl: any # Use the pinned version from flutter_localizations
///
/// # Rest of dependencies
/// ```
///
/// ## iOS Applications
///
/// iOS applications define key application metadata, including supported
/// locales, in an Info.plist file that is built into the application bundle.
/// To configure the locales supported by your app, youll need to edit this
/// file.
///
/// First, open your projects ios/Runner.xcworkspace Xcode workspace file.
/// Then, in the Project Navigator, open the Info.plist file under the Runner
/// projects Runner folder.
///
/// Next, select the Information Property List item, select Add Item from the
/// Editor menu, then select Localizations from the pop-up menu.
///
/// Select and expand the newly-created Localizations item then, for each
/// locale your application supports, add a new item and select the locale
/// you wish to add from the pop-up menu in the Value field. This list should
/// be consistent with the languages listed in the AppLocalizations.supportedLocales
/// property.
abstract class AppLocalizations {
AppLocalizations(String locale)
: localeName = intl.Intl.canonicalizedLocale(locale.toString());
final String localeName;
static AppLocalizations of(BuildContext context) {
return Localizations.of<AppLocalizations>(context, AppLocalizations)!;
}
static const LocalizationsDelegate<AppLocalizations> delegate =
_AppLocalizationsDelegate();
/// A list of this localizations delegate along with the default localizations
/// delegates.
///
/// Returns a list of localizations delegates containing this delegate along with
/// GlobalMaterialLocalizations.delegate, GlobalCupertinoLocalizations.delegate,
/// and GlobalWidgetsLocalizations.delegate.
///
/// Additional delegates can be added by appending to this list in
/// MaterialApp. This list does not have to be used at all if a custom list
/// of delegates is preferred or required.
static const List<LocalizationsDelegate<dynamic>> localizationsDelegates =
<LocalizationsDelegate<dynamic>>[
delegate,
GlobalMaterialLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
];
/// A list of this localizations delegate's supported locales.
static const List<Locale> supportedLocales = <Locale>[Locale('de')];
/// No description provided for @appTitle.
///
/// In de, this message translates to:
/// **'HouseHoldKeaper'**
String get appTitle;
/// No description provided for @tabHome.
///
/// In de, this message translates to:
/// **'Übersicht'**
String get tabHome;
/// No description provided for @tabRooms.
///
/// In de, this message translates to:
/// **'Räume'**
String get tabRooms;
/// No description provided for @tabSettings.
///
/// In de, this message translates to:
/// **'Einstellungen'**
String get tabSettings;
/// No description provided for @homeEmptyTitle.
///
/// In de, this message translates to:
/// **'Noch nichts zu tun!'**
String get homeEmptyTitle;
/// No description provided for @homeEmptyMessage.
///
/// In de, this message translates to:
/// **'Lege zuerst einen Raum an, um Aufgaben zu planen.'**
String get homeEmptyMessage;
/// No description provided for @homeEmptyAction.
///
/// In de, this message translates to:
/// **'Raum erstellen'**
String get homeEmptyAction;
/// No description provided for @roomsEmptyTitle.
///
/// In de, this message translates to:
/// **'Hier ist noch alles leer!'**
String get roomsEmptyTitle;
/// No description provided for @roomsEmptyMessage.
///
/// In de, this message translates to:
/// **'Erstelle deinen ersten Raum, um loszulegen.'**
String get roomsEmptyMessage;
/// No description provided for @roomsEmptyAction.
///
/// In de, this message translates to:
/// **'Raum erstellen'**
String get roomsEmptyAction;
/// No description provided for @settingsSectionAppearance.
///
/// In de, this message translates to:
/// **'Darstellung'**
String get settingsSectionAppearance;
/// No description provided for @settingsThemeLabel.
///
/// In de, this message translates to:
/// **'Farbschema'**
String get settingsThemeLabel;
/// No description provided for @themeSystem.
///
/// In de, this message translates to:
/// **'System'**
String get themeSystem;
/// No description provided for @themeLight.
///
/// In de, this message translates to:
/// **'Hell'**
String get themeLight;
/// No description provided for @themeDark.
///
/// In de, this message translates to:
/// **'Dunkel'**
String get themeDark;
/// No description provided for @settingsSectionAbout.
///
/// In de, this message translates to:
/// **'Über'**
String get settingsSectionAbout;
/// No description provided for @aboutAppName.
///
/// In de, this message translates to:
/// **'HouseHoldKeaper'**
String get aboutAppName;
/// No description provided for @aboutTagline.
///
/// In de, this message translates to:
/// **'Dein Haushalt, entspannt organisiert.'**
String get aboutTagline;
/// No description provided for @aboutVersion.
///
/// In de, this message translates to:
/// **'Version {version}'**
String aboutVersion(String version);
}
class _AppLocalizationsDelegate
extends LocalizationsDelegate<AppLocalizations> {
const _AppLocalizationsDelegate();
@override
Future<AppLocalizations> load(Locale locale) {
return SynchronousFuture<AppLocalizations>(lookupAppLocalizations(locale));
}
@override
bool isSupported(Locale locale) =>
<String>['de'].contains(locale.languageCode);
@override
bool shouldReload(_AppLocalizationsDelegate old) => false;
}
AppLocalizations lookupAppLocalizations(Locale locale) {
// Lookup logic when only language code is specified.
switch (locale.languageCode) {
case 'de':
return AppLocalizationsDe();
}
throw FlutterError(
'AppLocalizations.delegate failed to load unsupported locale "$locale". This is likely '
'an issue with the localizations generation tool. Please file an issue '
'on GitHub with a reproducible sample app and the gen-l10n configuration '
'that was used.',
);
}

View File

@@ -0,0 +1,70 @@
// ignore: unused_import
import 'package:intl/intl.dart' as intl;
import 'app_localizations.dart';
// ignore_for_file: type=lint
/// The translations for German (`de`).
class AppLocalizationsDe extends AppLocalizations {
AppLocalizationsDe([String locale = 'de']) : super(locale);
@override
String get appTitle => 'HouseHoldKeaper';
@override
String get tabHome => 'Übersicht';
@override
String get tabRooms => 'Räume';
@override
String get tabSettings => 'Einstellungen';
@override
String get homeEmptyTitle => 'Noch nichts zu tun!';
@override
String get homeEmptyMessage =>
'Lege zuerst einen Raum an, um Aufgaben zu planen.';
@override
String get homeEmptyAction => 'Raum erstellen';
@override
String get roomsEmptyTitle => 'Hier ist noch alles leer!';
@override
String get roomsEmptyMessage => 'Erstelle deinen ersten Raum, um loszulegen.';
@override
String get roomsEmptyAction => 'Raum erstellen';
@override
String get settingsSectionAppearance => 'Darstellung';
@override
String get settingsThemeLabel => 'Farbschema';
@override
String get themeSystem => 'System';
@override
String get themeLight => 'Hell';
@override
String get themeDark => 'Dunkel';
@override
String get settingsSectionAbout => 'Über';
@override
String get aboutAppName => 'HouseHoldKeaper';
@override
String get aboutTagline => 'Dein Haushalt, entspannt organisiert.';
@override
String aboutVersion(String version) {
return 'Version $version';
}
}