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>
This commit is contained in:
@@ -0,0 +1,80 @@
|
||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||
|
||||
import '../../../core/api/api_client.dart';
|
||||
import '../domain/bracket.dart';
|
||||
|
||||
part 'brackets_repository.g.dart';
|
||||
|
||||
class BracketsRepository {
|
||||
BracketsRepository(this._api);
|
||||
|
||||
final ApiClient _api;
|
||||
|
||||
Future<List<Bracket>> fetchBrackets() async {
|
||||
final data = await _api.get('/brackets/index.php');
|
||||
final list = (data['brackets'] as List?) ?? [];
|
||||
return list.whereType<Map<String, dynamic>>().map(Bracket.fromJson).toList();
|
||||
}
|
||||
|
||||
Future<Bracket?> getBracket(String id) async {
|
||||
try {
|
||||
final data = await _api.get('/brackets/detail.php', params: {'id': id});
|
||||
return Bracket.fromJson(data);
|
||||
} on ApiException catch (e) {
|
||||
if (e.statusCode == 404) return null;
|
||||
rethrow;
|
||||
}
|
||||
}
|
||||
|
||||
Future<String> createBracket(Bracket bracket) async {
|
||||
final data = await _api.post('/brackets/index.php', bracket.toJson());
|
||||
return data['id'] as String;
|
||||
}
|
||||
|
||||
Future<Bracket> updateBracket(Bracket bracket) async {
|
||||
final data = await _api.put(
|
||||
'/brackets/detail.php',
|
||||
bracket.toJson(),
|
||||
params: {'id': bracket.id},
|
||||
);
|
||||
return Bracket.fromJson(data);
|
||||
}
|
||||
|
||||
Future<void> deleteBracket(String id) async {
|
||||
await _api.delete('/brackets/detail.php', params: {'id': id});
|
||||
}
|
||||
|
||||
Future<void> updateMatch(
|
||||
String bracketId,
|
||||
String roundLabel,
|
||||
BracketMatch match,
|
||||
) async {
|
||||
final bracket = await getBracket(bracketId);
|
||||
if (bracket == null) return;
|
||||
final rounds = bracket.rounds.map((round) {
|
||||
if (round.label != roundLabel) return round;
|
||||
final updatedMatches = round.matches
|
||||
.map((m) => m.id == match.id ? match : m)
|
||||
.toList(growable: false);
|
||||
return round.copyWith(matches: updatedMatches);
|
||||
}).toList(growable: false);
|
||||
await updateBracket(bracket.copyWith(rounds: rounds));
|
||||
}
|
||||
|
||||
Stream<List<Bracket>> watchBrackets() async* {
|
||||
yield await fetchBrackets();
|
||||
await for (final _ in Stream<void>.periodic(const Duration(seconds: 30))) {
|
||||
yield await fetchBrackets();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Riverpod(keepAlive: true)
|
||||
BracketsRepository bracketsRepository(BracketsRepositoryRef ref) {
|
||||
return BracketsRepository(ref.watch(apiClientProvider));
|
||||
}
|
||||
|
||||
@riverpod
|
||||
Stream<List<Bracket>> bracketsStream(BracketsStreamRef ref) {
|
||||
return ref.watch(bracketsRepositoryProvider).watchBrackets();
|
||||
}
|
||||
@@ -0,0 +1,50 @@
|
||||
// GENERATED CODE - DO NOT MODIFY BY HAND
|
||||
|
||||
part of 'brackets_repository.dart';
|
||||
|
||||
// **************************************************************************
|
||||
// RiverpodGenerator
|
||||
// **************************************************************************
|
||||
|
||||
String _$bracketsRepositoryHash() =>
|
||||
r'942ebdb136bee1840c05c7d263e6a4e530cc2d4d';
|
||||
|
||||
/// See also [bracketsRepository].
|
||||
@ProviderFor(bracketsRepository)
|
||||
final bracketsRepositoryProvider = Provider<BracketsRepository>.internal(
|
||||
bracketsRepository,
|
||||
name: r'bracketsRepositoryProvider',
|
||||
debugGetCreateSourceHash: const bool.fromEnvironment('dart.vm.product')
|
||||
? null
|
||||
: _$bracketsRepositoryHash,
|
||||
dependencies: null,
|
||||
allTransitiveDependencies: null,
|
||||
);
|
||||
|
||||
@Deprecated('Will be removed in 3.0. Use Ref instead')
|
||||
// ignore: unused_element
|
||||
typedef BracketsRepositoryRef = ProviderRef<BracketsRepository>;
|
||||
String _$bracketsStreamHash() => r'72d5d17ad76cbfcf900c81d6bcf44f6678e52dfa';
|
||||
|
||||
/// Stream of brackets surfaced to the UI. Currently emits the mock list as a
|
||||
/// single tick — swap to `ref.watch(bracketsRepositoryProvider).watchBrackets()`
|
||||
/// once Firestore is seeded.
|
||||
///
|
||||
/// Copied from [bracketsStream].
|
||||
@ProviderFor(bracketsStream)
|
||||
final bracketsStreamProvider =
|
||||
AutoDisposeStreamProvider<List<Bracket>>.internal(
|
||||
bracketsStream,
|
||||
name: r'bracketsStreamProvider',
|
||||
debugGetCreateSourceHash: const bool.fromEnvironment('dart.vm.product')
|
||||
? null
|
||||
: _$bracketsStreamHash,
|
||||
dependencies: null,
|
||||
allTransitiveDependencies: null,
|
||||
);
|
||||
|
||||
@Deprecated('Will be removed in 3.0. Use Ref instead')
|
||||
// ignore: unused_element
|
||||
typedef BracketsStreamRef = AutoDisposeStreamProviderRef<List<Bracket>>;
|
||||
// ignore_for_file: type=lint
|
||||
// ignore_for_file: subtype_of_sealed_class, invalid_use_of_internal_member, invalid_use_of_visible_for_testing_member, deprecated_member_use_from_same_package
|
||||
Reference in New Issue
Block a user