2026-01-11 08:17:29 +00:00
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
|
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|
|
|
|
|
import 'package:flutter_localizations/flutter_localizations.dart'; // Localization
|
|
|
|
|
|
import 'package:hive_flutter/hive_flutter.dart';
|
|
|
|
|
|
import 'models/sake_item.dart';
|
|
|
|
|
|
import 'models/user_profile.dart';
|
|
|
|
|
|
import 'models/menu_settings.dart';
|
|
|
|
|
|
import 'providers/theme_provider.dart';
|
|
|
|
|
|
import 'screens/main_screen.dart';
|
|
|
|
|
|
import 'services/migration_service.dart';
|
|
|
|
|
|
|
2026-01-30 08:34:39 +00:00
|
|
|
|
/// Pro版かLite版かを判定するビルド時フラグ
|
|
|
|
|
|
///
|
|
|
|
|
|
/// ビルドコマンド:
|
|
|
|
|
|
/// - Pro版: flutter build apk --release --dart-define=IS_PRO_VERSION=true
|
|
|
|
|
|
/// - Lite版: flutter build apk --release --dart-define=IS_PRO_VERSION=false
|
|
|
|
|
|
///
|
|
|
|
|
|
/// デフォルトはtrue(Pro版)
|
|
|
|
|
|
const bool isProVersion = bool.fromEnvironment('IS_PRO_VERSION', defaultValue: true);
|
|
|
|
|
|
|
2026-01-11 08:17:29 +00:00
|
|
|
|
void main() async {
|
|
|
|
|
|
WidgetsFlutterBinding.ensureInitialized();
|
2026-01-30 16:00:37 +00:00
|
|
|
|
|
|
|
|
|
|
// 🔍 DEBUG: Check IS_PRO_VERSION flag
|
|
|
|
|
|
debugPrint('🔍 IS_PRO_VERSION = $isProVersion');
|
|
|
|
|
|
|
2026-01-11 08:17:29 +00:00
|
|
|
|
// Initialize Hive
|
|
|
|
|
|
await Hive.initFlutter();
|
|
|
|
|
|
|
|
|
|
|
|
// Register Adapters
|
|
|
|
|
|
Hive.registerAdapter(SakeItemAdapter());
|
|
|
|
|
|
Hive.registerAdapter(UserProfileAdapter());
|
|
|
|
|
|
Hive.registerAdapter(MenuSettingsAdapter());
|
|
|
|
|
|
// Phase 0 New Adapters
|
|
|
|
|
|
Hive.registerAdapter(DisplayDataAdapter());
|
|
|
|
|
|
Hive.registerAdapter(HiddenSpecsAdapter());
|
|
|
|
|
|
Hive.registerAdapter(UserDataAdapter());
|
|
|
|
|
|
Hive.registerAdapter(GamificationAdapter());
|
|
|
|
|
|
Hive.registerAdapter(MetadataAdapter());
|
2026-01-29 15:54:22 +00:00
|
|
|
|
Hive.registerAdapter(ItemTypeAdapter()); // Restored missing adapter
|
|
|
|
|
|
// Open all boxes first (faster to open them together)
|
|
|
|
|
|
// Reverted to synchronous wait to ensure Providers have data immediately
|
|
|
|
|
|
// Open all boxes in parallel (Much faster than sequential)
|
|
|
|
|
|
await Future.wait([
|
|
|
|
|
|
Hive.openBox('settings'),
|
|
|
|
|
|
Hive.openBox<UserProfile>('user_profile'),
|
|
|
|
|
|
Hive.openBox<SakeItem>('sake_items'),
|
|
|
|
|
|
Hive.openBox<MenuSettings>('menu_settings'),
|
|
|
|
|
|
]);
|
2026-01-11 08:17:29 +00:00
|
|
|
|
|
2026-01-29 15:54:22 +00:00
|
|
|
|
// Run Phase 0 Migration (Only once)
|
|
|
|
|
|
final box = Hive.box('settings');
|
|
|
|
|
|
final migrationCompleted = box.get('migration_completed', defaultValue: false);
|
|
|
|
|
|
if (!migrationCompleted) {
|
|
|
|
|
|
debugPrint('🚀 Running MigrationService...');
|
|
|
|
|
|
await MigrationService.runMigration();
|
|
|
|
|
|
await box.put('migration_completed', true);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
debugPrint('✅ Migration already completed. Skipping.');
|
|
|
|
|
|
}
|
2026-01-11 08:17:29 +00:00
|
|
|
|
|
2026-01-29 15:54:22 +00:00
|
|
|
|
// ✅ AI解析キャッシュは使うときに初期化する(Lazy initialization)
|
|
|
|
|
|
// AnalysisCacheService.init()はサービス内でLazy実装されているため、
|
|
|
|
|
|
// ここで呼び出すと起動が遅くなる。必要なときに自動初期化される。
|
2026-01-11 08:17:29 +00:00
|
|
|
|
|
|
|
|
|
|
runApp(
|
2026-01-29 15:54:22 +00:00
|
|
|
|
const ProviderScope(
|
|
|
|
|
|
child: MyApp(),
|
2026-01-11 08:17:29 +00:00
|
|
|
|
),
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
class MyApp extends ConsumerWidget {
|
|
|
|
|
|
const MyApp({super.key});
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
|
|
|
|
final lightTheme = ref.watch(lightThemeProvider);
|
|
|
|
|
|
final darkTheme = ref.watch(darkThemeProvider);
|
|
|
|
|
|
final themeMode = ref.watch(themeModeProvider);
|
2026-01-29 15:54:22 +00:00
|
|
|
|
final locale = ref.watch(localeProvider); // NEW: User-selected locale
|
2026-01-11 08:17:29 +00:00
|
|
|
|
|
|
|
|
|
|
return MaterialApp(
|
|
|
|
|
|
debugShowCheckedModeBanner: false,
|
|
|
|
|
|
title: 'Ponshu Room Lite',
|
|
|
|
|
|
theme: lightTheme,
|
|
|
|
|
|
darkTheme: darkTheme,
|
|
|
|
|
|
themeMode: themeMode,
|
2026-01-29 15:54:22 +00:00
|
|
|
|
locale: locale, // NEW: Apply user's locale choice
|
|
|
|
|
|
|
|
|
|
|
|
// Localization (UPDATED)
|
2026-01-11 08:17:29 +00:00
|
|
|
|
localizationsDelegates: const [
|
|
|
|
|
|
GlobalMaterialLocalizations.delegate,
|
|
|
|
|
|
GlobalWidgetsLocalizations.delegate,
|
|
|
|
|
|
GlobalCupertinoLocalizations.delegate,
|
|
|
|
|
|
],
|
|
|
|
|
|
supportedLocales: const [
|
2026-01-29 15:54:22 +00:00
|
|
|
|
Locale('ja'), // 日本語
|
|
|
|
|
|
Locale('en'), // English
|
|
|
|
|
|
// Phase 2: フランス語・ドイツ語を追加予定
|
|
|
|
|
|
// Locale('fr'), // Français
|
|
|
|
|
|
// Locale('de'), // Deutsch
|
2026-01-11 08:17:29 +00:00
|
|
|
|
],
|
|
|
|
|
|
|
2026-01-29 15:54:22 +00:00
|
|
|
|
navigatorObservers: [routeObserver],
|
2026-01-11 08:17:29 +00:00
|
|
|
|
home: const MainScreen(),
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-01-29 15:54:22 +00:00
|
|
|
|
|
|
|
|
|
|
// Global RouteObserver
|
|
|
|
|
|
final RouteObserver<ModalRoute<void>> routeObserver = RouteObserver<ModalRoute<void>>();
|