Files
philip b239ae3e5f Initial commit: Flutter app + PHP/MySQL backend on Hostinger
Replaces Firebase with a self-hosted PHP/MySQL API served from
winded.prymsolutions.com. Includes full backend (schema, auth, events,
teams, brackets, suggestions, stats, media, file upload) and updated
Flutter repositories and domain models.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-14 20:13:57 -07:00

113 lines
2.8 KiB
Dart

import 'dart:async';
import 'package:riverpod_annotation/riverpod_annotation.dart';
import '../../../core/api/api_client.dart';
import '../domain/app_user.dart';
part 'auth_repository.g.dart';
/// Manages auth state backed by the PHP/MySQL API.
///
/// Because the backend has no push mechanism, auth state is held in memory and
/// exposed via a [StreamController]. Sign-in and registration update the stream
/// immediately; the token is persisted in [FlutterSecureStorage] via [ApiClient].
class AuthRepository {
AuthRepository(this._api) {
_init();
}
final ApiClient _api;
final _controller = StreamController<AppUser?>.broadcast();
Stream<AppUser?> authStateChanges() => _controller.stream;
AppUser? get currentUser => _currentUser;
AppUser? _currentUser;
Future<void> _init() async {
final token = await _api.token;
if (token == null) {
_emit(null);
return;
}
try {
final data = await _api.get('/auth/me.php');
final user = _mapUser(data);
_emit(user);
} catch (_) {
await _api.clearToken();
_emit(null);
}
}
Future<AppUser?> signInWithEmail({
required String email,
required String password,
}) async {
final data = await _api.post(
'/auth/login.php',
{'email': email.trim(), 'password': password},
auth: false,
);
await _api.saveToken(data['token'] as String);
final user = _mapUser(data['user'] as Map<String, dynamic>);
_emit(user);
return user;
}
Future<AppUser?> registerWithEmail({
required String email,
required String password,
String? displayName,
}) async {
final data = await _api.post(
'/auth/register.php',
{
'email': email.trim(),
'password': password,
'display_name': displayName?.trim() ?? '',
},
auth: false,
);
await _api.saveToken(data['token'] as String);
final user = _mapUser(data['user'] as Map<String, dynamic>);
_emit(user);
return user;
}
Future<void> signOut() async {
await _api.clearToken();
_emit(null);
}
void _emit(AppUser? user) {
_currentUser = user;
_controller.add(user);
}
AppUser? _mapUser(Map<String, dynamic> data) {
final id = data['id'] as String?;
if (id == null || id.isEmpty) return null;
return AppUser(
uid: id,
email: (data['email'] as String?) ?? '',
displayName: data['display_name'] as String?,
photoUrl: data['photo_url'] as String?,
);
}
void dispose() => _controller.close();
}
@Riverpod(keepAlive: true)
AuthRepository authRepository(AuthRepositoryRef ref) {
final repo = AuthRepository(ref.watch(apiClientProvider));
ref.onDispose(repo.dispose);
return repo;
}
@Riverpod(keepAlive: true)
Stream<AppUser?> authStateChanges(AuthStateChangesRef ref) {
return ref.watch(authRepositoryProvider).authStateChanges();
}