sojorn/sojorn_app/lib/main.dart
Patrick Britton 38653f5854 Sojorn Backend Finalization & Cleanup - Complete Migration from Supabase
##  Phase 1: Critical Feature Completion (Beacon Voting)
- Add VouchBeacon, ReportBeacon, RemoveBeaconVote methods to PostRepository
- Implement beacon voting HTTP handlers with confidence score calculations
- Register new beacon routes: /beacons/:id/vouch, /beacons/:id/report, /beacons/:id/vouch (DELETE)
- Auto-flag beacons at 5+ reports, confidence scoring (0.5 base + 0.1 per vouch)

##  Phase 2: Feed Logic & Post Distribution Integrity
- Verify unified feed logic supports all content types (Standard, Quips, Beacons)
- Ensure proper distribution: Profile Feed + Main/Home Feed for followers
- Beacon Map integration for location-based content
- Video content filtering for Quips feed

##  Phase 3: The Notification System
- Create comprehensive NotificationService with FCM integration
- Add CreateNotification method to NotificationRepository
- Implement smart deep linking: beacon_map, quip_feed, main_feed
- Trigger notifications for beacon interactions and cross-post comments
- Push notification logic with proper content type detection

##  Phase 4: The Great Supabase Purge
- Delete function_proxy.go and remove /functions/:name route
- Remove SupabaseURL, SupabaseKey from config.go
- Remove SupabaseID field from User model
- Clean all Supabase imports and dependencies
- Sanitize codebase of legacy Supabase references

##  Phase 5: Flutter Frontend Integration
- Implement vouchBeacon(), reportBeacon(), removeBeaconVote() in ApiService
- Replace TODO delay in video_comments_sheet.dart with actual publishComment call
- Fix compilation errors (named parameters, orphaned child properties)
- Complete frontend integration with Go API endpoints

##  Additional Improvements
- Fix compilation errors in threaded_comment_widget.dart (orphaned child property)
- Update video_comments_sheet.dart to use proper named parameters
- Comprehensive error handling and validation
- Production-ready notification system with deep linking

##  Migration Status: 100% Complete
- Backend: Fully migrated from Supabase to custom Go/Gin API
- Frontend: Integrated with new Go endpoints
- Notifications: Complete FCM integration with smart routing
- Database: Clean of all Supabase dependencies
- Features: All functionality preserved and enhanced

Ready for VPS deployment and production testing!
2026-01-30 09:24:31 -06:00

171 lines
4.6 KiB
Dart

import 'dart:async';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:flutter_quill/flutter_quill.dart' as flutter_quill;
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:firebase_core/firebase_core.dart';
import 'package:firebase_messaging/firebase_messaging.dart';
import 'package:app_links/app_links.dart';
import 'config/firebase_web_config.dart';
import 'services/notification_service.dart';
import 'services/auth_service.dart';
import 'services/secure_chat_service.dart';
import 'services/simple_e2ee_service.dart';
import 'services/sync_manager.dart';
import 'theme/app_theme.dart';
import 'providers/theme_provider.dart' as theme_provider;
import 'providers/auth_provider.dart';
import 'routes/app_routes.dart';
Future<void> _firebaseMessagingBackgroundHandler(RemoteMessage message) async {
await Firebase.initializeApp();
}
void main() async {
WidgetsFlutterBinding.ensureInitialized();
if (kIsWeb) {
await Firebase.initializeApp(options: FirebaseWebConfig.options);
} else {
await Firebase.initializeApp();
}
FirebaseMessaging.onBackgroundMessage(_firebaseMessagingBackgroundHandler);
runApp(
const ProviderScope(
child: sojornApp(),
),
);
}
class sojornApp extends ConsumerStatefulWidget {
const sojornApp({super.key});
@override
ConsumerState<sojornApp> createState() => _sojornAppState();
}
class _sojornAppState extends ConsumerState<sojornApp> {
late AppLinks _appLinks;
StreamSubscription<Uri>? _linkSub;
StreamSubscription<AuthState>? _authSub;
late final AuthService _authService = AuthService();
SyncManager? _syncManager;
@override
void initState() {
super.initState();
_initDeepLinks();
_initE2ee();
_initNotifications();
_listenForAuth();
_initSyncManagerIfAuthenticated();
}
@override
void dispose() {
_linkSub?.cancel();
_authSub?.cancel();
_syncManager?.dispose();
super.dispose();
}
Future<void> _initDeepLinks() async {
_appLinks = AppLinks();
try {
final initialUri = await _appLinks.getInitialLink();
if (initialUri != null) {
_handleUri(initialUri);
}
} catch (_) {}
_linkSub = _appLinks.uriLinkStream.listen(
_handleUri,
onError: (_) {},
);
}
void _handleUri(Uri uri) {
if (uri.scheme != 'sojorn') return;
if (uri.host == 'beacon') {
final lat = double.tryParse(uri.queryParameters['lat'] ?? '');
final long = double.tryParse(uri.queryParameters['long'] ?? '');
if (lat != null && long != null) {
AppRoutes.router.go(
'/beacon?lat=${lat.toStringAsFixed(6)}&long=${long.toStringAsFixed(6)}',
);
}
} else if (uri.host == 'verified') {
ref.read(emailVerifiedEventProvider.notifier).state = true;
if (_authService.isAuthenticated) {
_authService.refreshSession();
}
}
}
void _initE2ee() {
if (_authService.isAuthenticated) {
SimpleE2EEService().initialize();
}
}
void _initNotifications() {
if (_authService.isAuthenticated) {
NotificationService.instance.init();
}
}
void _listenForAuth() {
_authSub = _authService.authStateChanges.listen((data) {
if (data.event == AuthChangeEvent.signedIn ||
data.event == AuthChangeEvent.tokenRefreshed) {
SimpleE2EEService().initialize();
NotificationService.instance.init();
_ensureSyncManager();
} else if (data.event == AuthChangeEvent.signedOut) {
_syncManager?.dispose();
_syncManager = null;
}
});
}
void _initSyncManagerIfAuthenticated() {
if (_authService.isAuthenticated) {
_ensureSyncManager();
}
}
void _ensureSyncManager() {
if (_syncManager != null) return;
_syncManager = SyncManager(
secureChatService: SecureChatService.instance,
authService: _authService,
);
_syncManager!.init();
}
@override
Widget build(BuildContext context) {
final themeMode = ref.watch(theme_provider.themeProvider);
AppTheme.setThemeType(themeMode == theme_provider.ThemeMode.pop
? AppThemeType.pop
: AppThemeType.basic);
return MaterialApp.router(
title: 'sojorn',
theme: AppTheme.lightTheme,
debugShowCheckedModeBanner: false,
localizationsDelegates: const [
GlobalWidgetsLocalizations.delegate,
GlobalMaterialLocalizations.delegate,
flutter_quill.FlutterQuillLocalizations.delegate,
],
routerConfig: AppRoutes.router,
);
}
}