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:
2026-05-14 20:13:57 -07:00
commit b239ae3e5f
208 changed files with 19187 additions and 0 deletions
@@ -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