create submission

This commit is contained in:
Phuoc Nguyen
2025-11-27 16:56:01 +07:00
parent ba04576750
commit b6cb9e865a
18 changed files with 1445 additions and 138 deletions

View File

@@ -5,9 +5,12 @@ library;
import 'package:riverpod_annotation/riverpod_annotation.dart';
import 'package:worker/core/network/dio_client.dart';
import 'package:worker/features/projects/data/datasources/project_progress_local_datasource.dart';
import 'package:worker/features/projects/data/datasources/project_status_local_datasource.dart';
import 'package:worker/features/projects/data/datasources/submissions_remote_datasource.dart';
import 'package:worker/features/projects/data/models/project_submission_request.dart';
import 'package:worker/features/projects/data/repositories/submissions_repository_impl.dart';
import 'package:worker/features/projects/domain/entities/project_progress.dart';
import 'package:worker/features/projects/domain/entities/project_status.dart';
import 'package:worker/features/projects/domain/entities/project_submission.dart';
import 'package:worker/features/projects/domain/repositories/submissions_repository.dart';
@@ -20,6 +23,12 @@ ProjectStatusLocalDataSource projectStatusLocalDataSource(Ref ref) {
return ProjectStatusLocalDataSource();
}
/// Project Progress Local Data Source Provider
@riverpod
ProjectProgressLocalDataSource projectProgressLocalDataSource(Ref ref) {
return ProjectProgressLocalDataSource();
}
/// Submissions Remote Data Source Provider
@riverpod
Future<SubmissionsRemoteDataSource> submissionsRemoteDataSource(Ref ref) async {
@@ -32,7 +41,12 @@ Future<SubmissionsRemoteDataSource> submissionsRemoteDataSource(Ref ref) async {
Future<SubmissionsRepository> submissionsRepository(Ref ref) async {
final remoteDataSource = await ref.watch(submissionsRemoteDataSourceProvider.future);
final statusLocalDataSource = ref.watch(projectStatusLocalDataSourceProvider);
return SubmissionsRepositoryImpl(remoteDataSource, statusLocalDataSource);
final progressLocalDataSource = ref.watch(projectProgressLocalDataSourceProvider);
return SubmissionsRepositoryImpl(
remoteDataSource,
statusLocalDataSource,
progressLocalDataSource,
);
}
/// Project Status List Provider
@@ -57,16 +71,39 @@ class ProjectStatusList extends _$ProjectStatusList {
}
}
/// Project Progress List Provider
///
/// Fetches construction progress stages from API with cache-first pattern.
/// Used for dropdown selection when creating/updating project submissions.
@riverpod
class ProjectProgressList extends _$ProjectProgressList {
@override
Future<List<ProjectProgress>> build() async {
final repository = await ref.watch(submissionsRepositoryProvider.future);
return repository.getProjectProgressList();
}
/// Refresh progress list from remote (force refresh)
Future<void> refresh() async {
state = const AsyncValue.loading();
state = await AsyncValue.guard(() async {
final repository = await ref.read(submissionsRepositoryProvider.future);
return repository.getProjectProgressList(forceRefresh: true);
});
}
}
/// All Submissions Provider
///
/// Fetches and manages submissions data from remote.
/// Waits for project status list to be loaded first.
/// Waits for project status list and progress list to be loaded first.
@riverpod
class AllSubmissions extends _$AllSubmissions {
@override
Future<List<ProjectSubmission>> build() async {
// Ensure status list is loaded first (for filter options)
// Ensure status list and progress list are loaded first (for filter options)
await ref.watch(projectStatusListProvider.future);
await ref.watch(projectProgressListProvider.future);
// Then fetch submissions
final repository = await ref.watch(submissionsRepositoryProvider.future);
@@ -77,8 +114,9 @@ class AllSubmissions extends _$AllSubmissions {
Future<void> refresh() async {
state = const AsyncValue.loading();
state = await AsyncValue.guard(() async {
// Also refresh status list
// Also refresh status list and progress list
await ref.read(projectStatusListProvider.notifier).refresh();
await ref.read(projectProgressListProvider.notifier).refresh();
final repository = await ref.read(submissionsRepositoryProvider.future);
return repository.getSubmissions();
@@ -154,3 +192,152 @@ AsyncValue<List<ProjectSubmission>> filteredSubmissions(Ref ref) {
return filtered;
});
}
/// Save Submission Provider
///
/// Handles creating new project submissions via API.
@riverpod
class SaveSubmission extends _$SaveSubmission {
@override
AsyncValue<void> build() {
return const AsyncValue.data(null);
}
/// Save a new project submission
///
/// Returns the project name (ID) if successful, throws exception on failure.
Future<String> save(ProjectSubmissionRequest request) async {
state = const AsyncValue.loading();
try {
final repository = await ref.read(submissionsRepositoryProvider.future);
if (!ref.mounted) throw Exception('Provider disposed');
final projectName = await repository.saveSubmission(request);
if (!ref.mounted) return projectName;
state = const AsyncValue.data(null);
// Refresh submissions list after successful save
ref.invalidate(allSubmissionsProvider);
return projectName;
} catch (e, st) {
if (ref.mounted) {
state = AsyncValue.error(e, st);
}
rethrow;
}
}
}
/// Upload state for tracking individual file uploads
class FileUploadState {
final String filePath;
final bool isUploading;
final bool isUploaded;
final String? fileUrl;
final String? error;
const FileUploadState({
required this.filePath,
this.isUploading = false,
this.isUploaded = false,
this.fileUrl,
this.error,
});
FileUploadState copyWith({
bool? isUploading,
bool? isUploaded,
String? fileUrl,
String? error,
}) {
return FileUploadState(
filePath: filePath,
isUploading: isUploading ?? this.isUploading,
isUploaded: isUploaded ?? this.isUploaded,
fileUrl: fileUrl ?? this.fileUrl,
error: error,
);
}
}
/// Upload Project Files Provider
///
/// Handles uploading multiple files for a project submission.
/// Tracks upload state for each file individually.
@riverpod
class UploadProjectFiles extends _$UploadProjectFiles {
@override
List<FileUploadState> build() {
return [];
}
/// Initialize with file paths
void initFiles(List<String> filePaths) {
state = filePaths
.map((path) => FileUploadState(filePath: path))
.toList();
}
/// Upload all files for a project
/// Returns list of uploaded file URLs
Future<List<String>> uploadAll(String projectName) async {
final uploadedUrls = <String>[];
for (var i = 0; i < state.length; i++) {
if (!ref.mounted) break;
// Mark as uploading
state = [
...state.sublist(0, i),
state[i].copyWith(isUploading: true),
...state.sublist(i + 1),
];
try {
final repository = await ref.read(submissionsRepositoryProvider.future);
if (!ref.mounted) break;
final fileUrl = await repository.uploadProjectFile(
projectName: projectName,
filePath: state[i].filePath,
);
if (!ref.mounted) break;
// Mark as uploaded
state = [
...state.sublist(0, i),
state[i].copyWith(
isUploading: false,
isUploaded: true,
fileUrl: fileUrl,
),
...state.sublist(i + 1),
];
uploadedUrls.add(fileUrl);
} catch (e) {
if (!ref.mounted) break;
// Mark as failed
state = [
...state.sublist(0, i),
state[i].copyWith(
isUploading: false,
error: e.toString(),
),
...state.sublist(i + 1),
];
}
}
return uploadedUrls;
}
/// Clear all files
void clear() {
state = [];
}
}

View File

@@ -62,6 +62,62 @@ final class ProjectStatusLocalDataSourceProvider
String _$projectStatusLocalDataSourceHash() =>
r'c57291e51bd390f9524369860c241d7a0a90fdbf';
/// Project Progress Local Data Source Provider
@ProviderFor(projectProgressLocalDataSource)
const projectProgressLocalDataSourceProvider =
ProjectProgressLocalDataSourceProvider._();
/// Project Progress Local Data Source Provider
final class ProjectProgressLocalDataSourceProvider
extends
$FunctionalProvider<
ProjectProgressLocalDataSource,
ProjectProgressLocalDataSource,
ProjectProgressLocalDataSource
>
with $Provider<ProjectProgressLocalDataSource> {
/// Project Progress Local Data Source Provider
const ProjectProgressLocalDataSourceProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'projectProgressLocalDataSourceProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$projectProgressLocalDataSourceHash();
@$internal
@override
$ProviderElement<ProjectProgressLocalDataSource> $createElement(
$ProviderPointer pointer,
) => $ProviderElement(pointer);
@override
ProjectProgressLocalDataSource create(Ref ref) {
return projectProgressLocalDataSource(ref);
}
/// {@macro riverpod.override_with_value}
Override overrideWithValue(ProjectProgressLocalDataSource value) {
return $ProviderOverride(
origin: this,
providerOverride: $SyncValueProvider<ProjectProgressLocalDataSource>(
value,
),
);
}
}
String _$projectProgressLocalDataSourceHash() =>
r'653d03b47f5642f3391e7a312649a2603489b224';
/// Submissions Remote Data Source Provider
@ProviderFor(submissionsRemoteDataSource)
@@ -155,7 +211,7 @@ final class SubmissionsRepositoryProvider
}
String _$submissionsRepositoryHash() =>
r'd8261cc538c1fdaa47064e4945302b80f49098bb';
r'652208a4ef93cde9b40ae66164d44bba786dfed0';
/// Project Status List Provider
///
@@ -221,10 +277,80 @@ abstract class _$ProjectStatusList extends $AsyncNotifier<List<ProjectStatus>> {
}
}
/// Project Progress List Provider
///
/// Fetches construction progress stages from API with cache-first pattern.
/// Used for dropdown selection when creating/updating project submissions.
@ProviderFor(ProjectProgressList)
const projectProgressListProvider = ProjectProgressListProvider._();
/// Project Progress List Provider
///
/// Fetches construction progress stages from API with cache-first pattern.
/// Used for dropdown selection when creating/updating project submissions.
final class ProjectProgressListProvider
extends $AsyncNotifierProvider<ProjectProgressList, List<ProjectProgress>> {
/// Project Progress List Provider
///
/// Fetches construction progress stages from API with cache-first pattern.
/// Used for dropdown selection when creating/updating project submissions.
const ProjectProgressListProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'projectProgressListProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$projectProgressListHash();
@$internal
@override
ProjectProgressList create() => ProjectProgressList();
}
String _$projectProgressListHash() =>
r'5ee1c23f90bfa61237f38a6b72c353f0ecb7a2a9';
/// Project Progress List Provider
///
/// Fetches construction progress stages from API with cache-first pattern.
/// Used for dropdown selection when creating/updating project submissions.
abstract class _$ProjectProgressList
extends $AsyncNotifier<List<ProjectProgress>> {
FutureOr<List<ProjectProgress>> build();
@$mustCallSuper
@override
void runBuild() {
final created = build();
final ref =
this.ref
as $Ref<AsyncValue<List<ProjectProgress>>, List<ProjectProgress>>;
final element =
ref.element
as $ClassProviderElement<
AnyNotifier<
AsyncValue<List<ProjectProgress>>,
List<ProjectProgress>
>,
AsyncValue<List<ProjectProgress>>,
Object?,
Object?
>;
element.handleValue(ref, created);
}
}
/// All Submissions Provider
///
/// Fetches and manages submissions data from remote.
/// Waits for project status list to be loaded first.
/// Waits for project status list and progress list to be loaded first.
@ProviderFor(AllSubmissions)
const allSubmissionsProvider = AllSubmissionsProvider._();
@@ -232,13 +358,13 @@ const allSubmissionsProvider = AllSubmissionsProvider._();
/// All Submissions Provider
///
/// Fetches and manages submissions data from remote.
/// Waits for project status list to be loaded first.
/// Waits for project status list and progress list to be loaded first.
final class AllSubmissionsProvider
extends $AsyncNotifierProvider<AllSubmissions, List<ProjectSubmission>> {
/// All Submissions Provider
///
/// Fetches and manages submissions data from remote.
/// Waits for project status list to be loaded first.
/// Waits for project status list and progress list to be loaded first.
const AllSubmissionsProvider._()
: super(
from: null,
@@ -258,12 +384,12 @@ final class AllSubmissionsProvider
AllSubmissions create() => AllSubmissions();
}
String _$allSubmissionsHash() => r'a4a7fb0d2953efb21e2e6343429f7550c763ea85';
String _$allSubmissionsHash() => r'ab0f1ffdc5e6bdb62dbd56ff3e586ecc1ff05bea';
/// All Submissions Provider
///
/// Fetches and manages submissions data from remote.
/// Waits for project status list to be loaded first.
/// Waits for project status list and progress list to be loaded first.
abstract class _$AllSubmissions
extends $AsyncNotifier<List<ProjectSubmission>> {
@@ -442,3 +568,142 @@ final class FilteredSubmissionsProvider
String _$filteredSubmissionsHash() =>
r'5be22b3242426c6b0c2f9778eaee5c7cf23e4814';
/// Save Submission Provider
///
/// Handles creating new project submissions via API.
@ProviderFor(SaveSubmission)
const saveSubmissionProvider = SaveSubmissionProvider._();
/// Save Submission Provider
///
/// Handles creating new project submissions via API.
final class SaveSubmissionProvider
extends $NotifierProvider<SaveSubmission, AsyncValue<void>> {
/// Save Submission Provider
///
/// Handles creating new project submissions via API.
const SaveSubmissionProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'saveSubmissionProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$saveSubmissionHash();
@$internal
@override
SaveSubmission create() => SaveSubmission();
/// {@macro riverpod.override_with_value}
Override overrideWithValue(AsyncValue<void> value) {
return $ProviderOverride(
origin: this,
providerOverride: $SyncValueProvider<AsyncValue<void>>(value),
);
}
}
String _$saveSubmissionHash() => r'64afa1a9662c36431c143c46a8ca34a786cb0860';
/// Save Submission Provider
///
/// Handles creating new project submissions via API.
abstract class _$SaveSubmission extends $Notifier<AsyncValue<void>> {
AsyncValue<void> build();
@$mustCallSuper
@override
void runBuild() {
final created = build();
final ref = this.ref as $Ref<AsyncValue<void>, AsyncValue<void>>;
final element =
ref.element
as $ClassProviderElement<
AnyNotifier<AsyncValue<void>, AsyncValue<void>>,
AsyncValue<void>,
Object?,
Object?
>;
element.handleValue(ref, created);
}
}
/// Upload Project Files Provider
///
/// Handles uploading multiple files for a project submission.
/// Tracks upload state for each file individually.
@ProviderFor(UploadProjectFiles)
const uploadProjectFilesProvider = UploadProjectFilesProvider._();
/// Upload Project Files Provider
///
/// Handles uploading multiple files for a project submission.
/// Tracks upload state for each file individually.
final class UploadProjectFilesProvider
extends $NotifierProvider<UploadProjectFiles, List<FileUploadState>> {
/// Upload Project Files Provider
///
/// Handles uploading multiple files for a project submission.
/// Tracks upload state for each file individually.
const UploadProjectFilesProvider._()
: super(
from: null,
argument: null,
retry: null,
name: r'uploadProjectFilesProvider',
isAutoDispose: true,
dependencies: null,
$allTransitiveDependencies: null,
);
@override
String debugGetCreateSourceHash() => _$uploadProjectFilesHash();
@$internal
@override
UploadProjectFiles create() => UploadProjectFiles();
/// {@macro riverpod.override_with_value}
Override overrideWithValue(List<FileUploadState> value) {
return $ProviderOverride(
origin: this,
providerOverride: $SyncValueProvider<List<FileUploadState>>(value),
);
}
}
String _$uploadProjectFilesHash() =>
r'd6219bc1f0b0d6ac70b9e3cea731267c82a68e1f';
/// Upload Project Files Provider
///
/// Handles uploading multiple files for a project submission.
/// Tracks upload state for each file individually.
abstract class _$UploadProjectFiles extends $Notifier<List<FileUploadState>> {
List<FileUploadState> build();
@$mustCallSuper
@override
void runBuild() {
final created = build();
final ref = this.ref as $Ref<List<FileUploadState>, List<FileUploadState>>;
final element =
ref.element
as $ClassProviderElement<
AnyNotifier<List<FileUploadState>, List<FileUploadState>>,
List<FileUploadState>,
Object?,
Object?
>;
element.handleValue(ref, created);
}
}