Refactor updateStaffProfile method to accept staff ID and data map, and update PersonalInfoBloc to handle form values

This commit is contained in:
Achintha Isuru
2026-01-27 00:36:24 -05:00
parent 098ae1d476
commit 3e7ddc446d
5 changed files with 82 additions and 58 deletions

View File

@@ -48,16 +48,28 @@ class PersonalInfoRepositoryImpl implements PersonalInfoRepositoryInterface {
}
@override
Future<Staff> updateStaffProfile(Staff staff) async {
// Update staff data through Firebase Data Connect
Future<Staff> updateStaffProfile({required String staffId, required Map<String, dynamic> data}) async {
// Start building the update mutation
var updateBuilder = _dataConnect.updateStaff(id: staffId);
// Apply updates from map if present
if (data.containsKey('name')) {
updateBuilder = updateBuilder.fullName(data['name'] as String);
}
if (data.containsKey('email')) {
updateBuilder = updateBuilder.email(data['email'] as String);
}
if (data.containsKey('phone')) {
updateBuilder = updateBuilder.phone(data['phone'] as String?);
}
if (data.containsKey('avatar')) {
updateBuilder = updateBuilder.photoUrl(data['avatar'] as String?);
}
// Add other fields as they become available in the schema and map
// Execute the update
final OperationResult<UpdateStaffData, UpdateStaffVariables> result =
await _dataConnect
.updateStaff(id: staff.id)
.fullName(staff.name)
.email(staff.email)
.phone(staff.phone)
.photoUrl(staff.avatar)
.execute();
await updateBuilder.execute();
if (result.data.staff_update == null) {
throw Exception('Failed to update staff profile');

View File

@@ -15,9 +15,9 @@ abstract interface class PersonalInfoRepositoryInterface {
/// Updates the staff profile information.
///
/// Takes a [Staff] entity with updated fields and persists changes
/// Takes a [Staff] entity ID and updated fields map and persists changes
/// through the data layer. Returns the updated [Staff] entity.
Future<Staff> updateStaffProfile(Staff staff);
Future<Staff> updateStaffProfile({required String staffId, required Map<String, dynamic> data});
/// Uploads a profile photo and returns the URL.
///

View File

@@ -2,12 +2,29 @@ import 'package:krow_core/core.dart';
import 'package:krow_domain/krow_domain.dart';
import '../repositories/personal_info_repository_interface.dart';
/// Arguments for updating staff profile information.
class UpdatePersonalInfoParams extends UseCaseArgument {
/// The staff member's ID.
final String staffId;
/// The fields to update.
final Map<String, dynamic> data;
const UpdatePersonalInfoParams({
required this.staffId,
required this.data,
});
@override
List<Object?> get props => [staffId, data];
}
/// Use case for updating staff profile information.
///
/// This use case updates the staff profile information
/// through the repository, which delegates to the data_connect layer.
class UpdatePersonalInfoUseCase
implements UseCase<Staff, Staff> {
implements UseCase<UpdatePersonalInfoParams, Staff> {
final PersonalInfoRepositoryInterface _repository;
/// Creates an [UpdatePersonalInfoUseCase].
@@ -16,7 +33,10 @@ class UpdatePersonalInfoUseCase
UpdatePersonalInfoUseCase(this._repository);
@override
Future<Staff> call(Staff arguments) {
return _repository.updateStaffProfile(arguments);
Future<Staff> call(UpdatePersonalInfoParams params) {
return _repository.updateStaffProfile(
staffId: params.staffId,
data: params.data,
);
}
}

View File

@@ -1,6 +1,5 @@
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:flutter_modular/flutter_modular.dart';
import 'package:krow_core/core.dart';
import 'package:krow_domain/krow_domain.dart';
import '../../domain/usecases/get_personal_info_usecase.dart';
@@ -41,9 +40,20 @@ class PersonalInfoBloc extends Bloc<PersonalInfoEvent, PersonalInfoState>
emit(state.copyWith(status: PersonalInfoStatus.loading));
try {
final Staff staff = await _getPersonalInfoUseCase();
// Initialize form values from staff entity
final Map<String, dynamic> initialValues = {
'name': staff.name,
'email': staff.email,
'phone': staff.phone,
'address': staff.address,
'avatar': staff.avatar,
};
emit(state.copyWith(
status: PersonalInfoStatus.loaded,
staff: staff,
formValues: initialValues,
));
} catch (e) {
emit(state.copyWith(
@@ -58,47 +68,9 @@ class PersonalInfoBloc extends Bloc<PersonalInfoEvent, PersonalInfoState>
PersonalInfoFieldUpdated event,
Emitter<PersonalInfoState> emit,
) {
if (state.staff == null) return;
final Staff updatedStaff = _updateField(state.staff!, event.field, event.value);
emit(state.copyWith(staff: updatedStaff));
}
/// Updates a specific field in the Staff entity.
///
/// Returns a new Staff instance with the updated field.
Staff _updateField(Staff staff, String field, String value) {
// Note: Staff entity doesn't have a copyWith method or bio/languages/locations fields
// These fields would need to be added to the Staff entity or handled differently
// For now, we're just returning the same staff
// TODO: Add support for bio, languages, preferred locations to Staff entity
switch (field) {
case 'phone':
// Since Staff is immutable and doesn't have copyWith, we'd need to create a new instance
return Staff(
id: staff.id,
authProviderId: staff.authProviderId,
name: staff.name,
email: staff.email,
phone: value,
status: staff.status,
address: staff.address,
avatar: staff.avatar,
);
case 'address':
return Staff(
id: staff.id,
authProviderId: staff.authProviderId,
name: staff.name,
email: staff.email,
phone: staff.phone,
status: staff.status,
address: value,
avatar: staff.avatar,
);
default:
return staff;
}
final Map<String, dynamic> updatedValues = Map.from(state.formValues);
updatedValues[event.field] = event.value;
emit(state.copyWith(formValues: updatedValues));
}
/// Handles saving staff profile information.
@@ -111,11 +83,25 @@ class PersonalInfoBloc extends Bloc<PersonalInfoEvent, PersonalInfoState>
emit(state.copyWith(status: PersonalInfoStatus.saving));
try {
final Staff updatedStaff = await _updatePersonalInfoUseCase(
state.staff!,
UpdatePersonalInfoParams(
staffId: state.staff!.id,
data: state.formValues,
),
);
// Update local state with the returned staff and keep form values in sync
final Map<String, dynamic> newValues = {
'name': updatedStaff.name,
'email': updatedStaff.email,
'phone': updatedStaff.phone,
'address': updatedStaff.address,
'avatar': updatedStaff.avatar,
};
emit(state.copyWith(
status: PersonalInfoStatus.saved,
staff: updatedStaff,
formValues: newValues,
));
} catch (e) {
emit(state.copyWith(

View File

@@ -35,6 +35,9 @@ class PersonalInfoState extends Equatable {
/// The staff profile information.
final Staff? staff;
/// The form values being edited.
final Map<String, dynamic> formValues;
/// Error message if an error occurred.
final String? errorMessage;
@@ -42,6 +45,7 @@ class PersonalInfoState extends Equatable {
const PersonalInfoState({
this.status = PersonalInfoStatus.initial,
this.staff,
this.formValues = const {},
this.errorMessage,
});
@@ -49,15 +53,17 @@ class PersonalInfoState extends Equatable {
PersonalInfoState copyWith({
PersonalInfoStatus? status,
Staff? staff,
Map<String, dynamic>? formValues,
String? errorMessage,
}) {
return PersonalInfoState(
status: status ?? this.status,
staff: staff ?? this.staff,
formValues: formValues ?? this.formValues,
errorMessage: errorMessage,
);
}
@override
List<Object?> get props => [status, staff, errorMessage];
List<Object?> get props => [status, staff, formValues, errorMessage];
}