Vaibhav_Feature-#768 #59

Closed
vaibhav.surve wants to merge 74 commits from Vaibhav_Feature-#768 into Feature_Expense
4 changed files with 120 additions and 94 deletions
Showing only changes of commit 0150400092 - Show all commits

View File

@ -18,7 +18,6 @@ import 'package:marco/model/expense/expense_type_model.dart';
import 'package:marco/model/expense/payment_types_model.dart'; import 'package:marco/model/expense/payment_types_model.dart';
class AddExpenseController extends GetxController { class AddExpenseController extends GetxController {
// === Text Controllers ===
final amountController = TextEditingController(); final amountController = TextEditingController();
final descriptionController = TextEditingController(); final descriptionController = TextEditingController();
final supplierController = TextEditingController(); final supplierController = TextEditingController();
@ -28,21 +27,17 @@ class AddExpenseController extends GetxController {
final transactionDateController = TextEditingController(); final transactionDateController = TextEditingController();
final TextEditingController noOfPersonsController = TextEditingController(); final TextEditingController noOfPersonsController = TextEditingController();
// === State Controllers ===
final RxBool isLoading = false.obs; final RxBool isLoading = false.obs;
final RxBool isSubmitting = false.obs; final RxBool isSubmitting = false.obs;
final RxBool isFetchingLocation = false.obs; final RxBool isFetchingLocation = false.obs;
// === Selected Models ===
final Rx<PaymentModeModel?> selectedPaymentMode = Rx<PaymentModeModel?>(null); final Rx<PaymentModeModel?> selectedPaymentMode = Rx<PaymentModeModel?>(null);
final Rx<ExpenseTypeModel?> selectedExpenseType = Rx<ExpenseTypeModel?>(null); final Rx<ExpenseTypeModel?> selectedExpenseType = Rx<ExpenseTypeModel?>(null);
final Rx<ExpenseStatusModel?> selectedExpenseStatus = final Rx<ExpenseStatusModel?> selectedExpenseStatus = Rx<ExpenseStatusModel?>(null);
Rx<ExpenseStatusModel?>(null);
final Rx<EmployeeModel?> selectedPaidBy = Rx<EmployeeModel?>(null); final Rx<EmployeeModel?> selectedPaidBy = Rx<EmployeeModel?>(null);
final RxString selectedProject = ''.obs; final RxString selectedProject = ''.obs;
final Rx<DateTime?> selectedTransactionDate = Rx<DateTime?>(null); final Rx<DateTime?> selectedTransactionDate = Rx<DateTime?>(null);
// === Lists ===
final RxList<File> attachments = <File>[].obs; final RxList<File> attachments = <File>[].obs;
final RxList<String> globalProjects = <String>[].obs; final RxList<String> globalProjects = <String>[].obs;
final RxList<String> projects = <String>[].obs; final RxList<String> projects = <String>[].obs;
@ -51,7 +46,6 @@ class AddExpenseController extends GetxController {
final RxList<ExpenseStatusModel> expenseStatuses = <ExpenseStatusModel>[].obs; final RxList<ExpenseStatusModel> expenseStatuses = <ExpenseStatusModel>[].obs;
final RxList<EmployeeModel> allEmployees = <EmployeeModel>[].obs; final RxList<EmployeeModel> allEmployees = <EmployeeModel>[].obs;
// === Mappings ===
final RxMap<String, String> projectsMap = <String, String>{}.obs; final RxMap<String, String> projectsMap = <String, String>{}.obs;
final ExpenseController expenseController = Get.find<ExpenseController>(); final ExpenseController expenseController = Get.find<ExpenseController>();
@ -77,7 +71,6 @@ class AddExpenseController extends GetxController {
super.onClose(); super.onClose();
} }
// === Pick Attachments ===
Future<void> pickAttachments() async { Future<void> pickAttachments() async {
try { try {
final result = await FilePicker.platform.pickFiles( final result = await FilePicker.platform.pickFiles(
@ -86,12 +79,15 @@ class AddExpenseController extends GetxController {
allowMultiple: true, allowMultiple: true,
); );
if (result != null && result.paths.isNotEmpty) { if (result != null && result.paths.isNotEmpty) {
final files = final files = result.paths.whereType<String>().map((e) => File(e)).toList();
result.paths.whereType<String>().map((e) => File(e)).toList();
attachments.addAll(files); attachments.addAll(files);
} }
} catch (e) { } catch (e) {
Get.snackbar("Error", "Failed to pick attachments: $e"); showAppSnackbar(
title: "Error",
message: "Failed to pick attachments: $e",
type: SnackbarType.error,
);
} }
} }
@ -99,16 +95,14 @@ class AddExpenseController extends GetxController {
attachments.remove(file); attachments.remove(file);
} }
// === Date Picker ===
void pickTransactionDate(BuildContext context) async { void pickTransactionDate(BuildContext context) async {
final now = DateTime.now(); final now = DateTime.now();
final picked = await showDatePicker( final picked = await showDatePicker(
context: context, context: context,
initialDate: selectedTransactionDate.value ?? now, initialDate: selectedTransactionDate.value ?? now,
firstDate: DateTime(now.year - 5), firstDate: DateTime(now.year - 5),
lastDate: now, // Restrict future dates lastDate: now,
); );
if (picked != null) { if (picked != null) {
selectedTransactionDate.value = picked; selectedTransactionDate.value = picked;
transactionDateController.text = transactionDateController.text =
@ -116,31 +110,33 @@ class AddExpenseController extends GetxController {
} }
} }
// === Fetch Current Location ===
Future<void> fetchCurrentLocation() async { Future<void> fetchCurrentLocation() async {
isFetchingLocation.value = true; isFetchingLocation.value = true;
try { try {
LocationPermission permission = await Geolocator.checkPermission(); LocationPermission permission = await Geolocator.checkPermission();
if (permission == LocationPermission.denied || if (permission == LocationPermission.denied || permission == LocationPermission.deniedForever) {
permission == LocationPermission.deniedForever) {
permission = await Geolocator.requestPermission(); permission = await Geolocator.requestPermission();
if (permission == LocationPermission.denied || if (permission == LocationPermission.denied || permission == LocationPermission.deniedForever) {
permission == LocationPermission.deniedForever) { showAppSnackbar(
Get.snackbar( title: "Error",
"Error", "Location permission denied. Enable in settings."); message: "Location permission denied. Enable in settings.",
type: SnackbarType.error,
);
return; return;
} }
} }
if (!await Geolocator.isLocationServiceEnabled()) { if (!await Geolocator.isLocationServiceEnabled()) {
Get.snackbar("Error", "Location services are disabled. Enable them."); showAppSnackbar(
title: "Error",
message: "Location services are disabled. Enable them.",
type: SnackbarType.error,
);
return; return;
} }
final position = await Geolocator.getCurrentPosition( final position = await Geolocator.getCurrentPosition(desiredAccuracy: LocationAccuracy.high);
desiredAccuracy: LocationAccuracy.high); final placemarks = await placemarkFromCoordinates(position.latitude, position.longitude);
final placemarks =
await placemarkFromCoordinates(position.latitude, position.longitude);
if (placemarks.isNotEmpty) { if (placemarks.isNotEmpty) {
final place = placemarks.first; final place = placemarks.first;
@ -157,13 +153,16 @@ class AddExpenseController extends GetxController {
locationController.text = "${position.latitude}, ${position.longitude}"; locationController.text = "${position.latitude}, ${position.longitude}";
} }
} catch (e) { } catch (e) {
Get.snackbar("Error", "Error fetching location: $e"); showAppSnackbar(
title: "Error",
message: "Error fetching location: $e",
type: SnackbarType.error,
);
} finally { } finally {
isFetchingLocation.value = false; isFetchingLocation.value = false;
} }
} }
// === Submit Expense ===
Future<void> submitExpense() async { Future<void> submitExpense() async {
if (isSubmitting.value) return; if (isSubmitting.value) return;
isSubmitting.value = true; isSubmitting.value = true;
@ -239,8 +238,7 @@ class AddExpenseController extends GetxController {
expensesTypeId: selectedExpenseType.value!.id, expensesTypeId: selectedExpenseType.value!.id,
paymentModeId: selectedPaymentMode.value!.id, paymentModeId: selectedPaymentMode.value!.id,
paidById: selectedPaidBy.value!.id, paidById: selectedPaidBy.value!.id,
transactionDate: transactionDate: selectedTransactionDate.value?.toUtc() ?? DateTime.now().toUtc(),
(selectedTransactionDate.value ?? DateTime.now()).toUtc(),
transactionId: transactionIdController.text, transactionId: transactionIdController.text,
description: descriptionController.text, description: descriptionController.text,
location: locationController.text, location: locationController.text,
@ -278,7 +276,6 @@ class AddExpenseController extends GetxController {
} }
} }
// === Fetch Data Methods ===
Future<void> fetchMasterData() async { Future<void> fetchMasterData() async {
try { try {
final expenseTypesData = await ApiService.getMasterExpenseTypes(); final expenseTypesData = await ApiService.getMasterExpenseTypes();
@ -286,22 +283,22 @@ class AddExpenseController extends GetxController {
final expenseStatusData = await ApiService.getMasterExpenseStatus(); final expenseStatusData = await ApiService.getMasterExpenseStatus();
if (expenseTypesData is List) { if (expenseTypesData is List) {
expenseTypes.value = expenseTypes.value = expenseTypesData.map((e) => ExpenseTypeModel.fromJson(e)).toList();
expenseTypesData.map((e) => ExpenseTypeModel.fromJson(e)).toList();
} }
if (paymentModesData is List) { if (paymentModesData is List) {
paymentModes.value = paymentModes.value = paymentModesData.map((e) => PaymentModeModel.fromJson(e)).toList();
paymentModesData.map((e) => PaymentModeModel.fromJson(e)).toList();
} }
if (expenseStatusData is List) { if (expenseStatusData is List) {
expenseStatuses.value = expenseStatusData expenseStatuses.value = expenseStatusData.map((e) => ExpenseStatusModel.fromJson(e)).toList();
.map((e) => ExpenseStatusModel.fromJson(e))
.toList();
} }
} catch (e) { } catch (e) {
Get.snackbar("Error", "Failed to fetch master data: $e"); showAppSnackbar(
title: "Error",
message: "Failed to fetch master data: $e",
type: SnackbarType.error,
);
} }
} }
@ -332,8 +329,7 @@ class AddExpenseController extends GetxController {
final response = await ApiService.getAllEmployees(); final response = await ApiService.getAllEmployees();
if (response != null && response.isNotEmpty) { if (response != null && response.isNotEmpty) {
allEmployees.assignAll(response.map((e) => EmployeeModel.fromJson(e))); allEmployees.assignAll(response.map((e) => EmployeeModel.fromJson(e)));
logSafe("All Employees fetched: ${allEmployees.length}", logSafe("All Employees fetched: ${allEmployees.length}", level: LogLevel.info);
level: LogLevel.info);
} else { } else {
allEmployees.clear(); allEmployees.clear();
logSafe("No employees found.", level: LogLevel.warning); logSafe("No employees found.", level: LogLevel.warning);

View File

@ -7,6 +7,8 @@ import 'package:marco/model/expense/payment_types_model.dart';
import 'package:marco/model/expense/expense_type_model.dart'; import 'package:marco/model/expense/expense_type_model.dart';
import 'package:marco/model/expense/expense_status_model.dart'; import 'package:marco/model/expense/expense_status_model.dart';
import 'package:marco/model/employee_model.dart'; import 'package:marco/model/employee_model.dart';
import 'package:marco/helpers/widgets/my_snackbar.dart';
class ExpenseController extends GetxController { class ExpenseController extends GetxController {
final RxList<ExpenseModel> expenses = <ExpenseModel>[].obs; final RxList<ExpenseModel> expenses = <ExpenseModel>[].obs;
@ -68,15 +70,27 @@ class ExpenseController extends GetxController {
if (success) { if (success) {
expenses.removeWhere((e) => e.id == expenseId); expenses.removeWhere((e) => e.id == expenseId);
logSafe("Expense deleted successfully."); logSafe("Expense deleted successfully.");
Get.snackbar("Deleted", "Expense has been deleted successfully."); showAppSnackbar(
title: "Deleted",
message: "Expense has been deleted successfully.",
type: SnackbarType.success,
);
} else { } else {
logSafe("Failed to delete expense: $expenseId", level: LogLevel.error); logSafe("Failed to delete expense: $expenseId", level: LogLevel.error);
Get.snackbar("Failed", "Failed to delete expense."); showAppSnackbar(
title: "Failed",
message: "Failed to delete expense.",
type: SnackbarType.error,
);
} }
} catch (e, stack) { } catch (e, stack) {
logSafe("Exception in deleteExpense: $e", level: LogLevel.error); logSafe("Exception in deleteExpense: $e", level: LogLevel.error);
logSafe("StackTrace: $stack", level: LogLevel.debug); logSafe("StackTrace: $stack", level: LogLevel.debug);
Get.snackbar("Error", "Something went wrong while deleting."); showAppSnackbar(
title: "Error",
message: "Something went wrong while deleting.",
type: SnackbarType.error,
);
} }
} }
@ -174,14 +188,17 @@ class ExpenseController extends GetxController {
final expenseStatusData = await ApiService.getMasterExpenseStatus(); final expenseStatusData = await ApiService.getMasterExpenseStatus();
if (expenseStatusData is List) { if (expenseStatusData is List) {
expenseStatuses.value = expenseStatusData expenseStatuses.value =
.map((e) => ExpenseStatusModel.fromJson(e)) expenseStatusData.map((e) => ExpenseStatusModel.fromJson(e)).toList();
.toList();
} }
} catch (e) { } catch (e) {
Get.snackbar("Error", "Failed to fetch master data: $e"); showAppSnackbar(
} title: "Error",
message: "Failed to fetch master data: $e",
type: SnackbarType.error,
);
} }
}
/// Fetch global projects /// Fetch global projects
Future<void> fetchGlobalProjects() async { Future<void> fetchGlobalProjects() async {

View File

@ -7,6 +7,8 @@ import 'package:marco/helpers/widgets/my_spacing.dart';
import 'package:marco/helpers/widgets/my_text.dart'; import 'package:marco/helpers/widgets/my_text.dart';
import 'package:marco/helpers/widgets/my_text_style.dart'; import 'package:marco/helpers/widgets/my_text_style.dart';
import 'package:marco/helpers/utils/base_bottom_sheet.dart'; import 'package:marco/helpers/utils/base_bottom_sheet.dart';
import 'package:marco/helpers/widgets/my_snackbar.dart';
class ReimbursementBottomSheet extends StatefulWidget { class ReimbursementBottomSheet extends StatefulWidget {
final String expenseId; final String expenseId;
@ -34,7 +36,8 @@ class ReimbursementBottomSheet extends StatefulWidget {
} }
class _ReimbursementBottomSheetState extends State<ReimbursementBottomSheet> { class _ReimbursementBottomSheetState extends State<ReimbursementBottomSheet> {
final ExpenseDetailController controller = Get.find<ExpenseDetailController>(); final ExpenseDetailController controller =
Get.find<ExpenseDetailController>();
final TextEditingController commentCtrl = TextEditingController(); final TextEditingController commentCtrl = TextEditingController();
final TextEditingController txnCtrl = TextEditingController(); final TextEditingController txnCtrl = TextEditingController();
@ -119,7 +122,11 @@ class _ReimbursementBottomSheetState extends State<ReimbursementBottomSheet> {
txnCtrl.text.trim().isEmpty || txnCtrl.text.trim().isEmpty ||
dateStr.value.isEmpty || dateStr.value.isEmpty ||
controller.selectedReimbursedBy.value == null) { controller.selectedReimbursedBy.value == null) {
Get.snackbar("Incomplete", "Please fill all fields"); showAppSnackbar(
title: "Incomplete",
message: "Please fill all fields",
type: SnackbarType.warning,
);
return; return;
} }
@ -133,9 +140,17 @@ class _ReimbursementBottomSheetState extends State<ReimbursementBottomSheet> {
if (success) { if (success) {
Get.back(); Get.back();
Get.snackbar('Success', 'Reimbursement submitted'); showAppSnackbar(
title: "Success",
message: "Reimbursement submitted",
type: SnackbarType.success,
);
} else { } else {
Get.snackbar('Error', controller.errorMessage.value); showAppSnackbar(
title: "Error",
message: controller.errorMessage.value,
type: SnackbarType.error,
);
} }
}, },
child: Column( child: Column(
@ -148,7 +163,6 @@ class _ReimbursementBottomSheetState extends State<ReimbursementBottomSheet> {
decoration: _inputDecoration("Enter comment"), decoration: _inputDecoration("Enter comment"),
), ),
MySpacing.height(16), MySpacing.height(16),
MyText.labelMedium("Transaction ID"), MyText.labelMedium("Transaction ID"),
MySpacing.height(8), MySpacing.height(8),
TextField( TextField(
@ -156,7 +170,6 @@ class _ReimbursementBottomSheetState extends State<ReimbursementBottomSheet> {
decoration: _inputDecoration("Enter transaction ID"), decoration: _inputDecoration("Enter transaction ID"),
), ),
MySpacing.height(16), MySpacing.height(16),
MyText.labelMedium("Reimbursement Date"), MyText.labelMedium("Reimbursement Date"),
MySpacing.height(8), MySpacing.height(8),
GestureDetector( GestureDetector(
@ -183,7 +196,6 @@ class _ReimbursementBottomSheetState extends State<ReimbursementBottomSheet> {
), ),
), ),
MySpacing.height(16), MySpacing.height(16),
MyText.labelMedium("Reimbursed By"), MyText.labelMedium("Reimbursed By"),
MySpacing.height(8), MySpacing.height(8),
GestureDetector( GestureDetector(

View File

@ -12,6 +12,7 @@ import 'package:marco/model/expense/expense_detail_model.dart';
import 'package:marco/helpers/widgets/image_viewer_dialog.dart'; import 'package:marco/helpers/widgets/image_viewer_dialog.dart';
import 'package:url_launcher/url_launcher.dart'; import 'package:url_launcher/url_launcher.dart';
import 'package:marco/model/expense/reimbursement_bottom_sheet.dart'; import 'package:marco/model/expense/reimbursement_bottom_sheet.dart';
import 'package:marco/helpers/widgets/my_snackbar.dart';
class ExpenseDetailScreen extends StatelessWidget { class ExpenseDetailScreen extends StatelessWidget {
final String expenseId; final String expenseId;
@ -224,21 +225,18 @@ class ExpenseDetailScreen extends StatelessWidget {
); );
if (success) { if (success) {
Get.snackbar( showAppSnackbar(
'Success', title: 'Success',
'Expense reimbursed successfully.', message: 'Expense reimbursed successfully.',
backgroundColor: Colors.green.withOpacity(0.8), type: SnackbarType.success,
colorText: Colors.white,
); );
await controller.fetchExpenseDetails(); await controller.fetchExpenseDetails();
return true; return true;
} else { } else {
Get.snackbar( showAppSnackbar(
'Error', title: 'Error',
'Failed to reimburse expense.', message: 'Failed to reimburse expense.',
backgroundColor: Colors.red.withOpacity(0.8), type: SnackbarType.error,
colorText: Colors.white,
); );
return false; return false;
} }
@ -250,19 +248,18 @@ class ExpenseDetailScreen extends StatelessWidget {
await controller.updateExpenseStatus(next.id); await controller.updateExpenseStatus(next.id);
if (success) { if (success) {
Get.snackbar( showAppSnackbar(
'Success', title: 'Success',
message:
'Expense moved to ${next.displayName.isNotEmpty ? next.displayName : next.name}', 'Expense moved to ${next.displayName.isNotEmpty ? next.displayName : next.name}',
backgroundColor: Colors.green.withOpacity(0.8), type: SnackbarType.success,
colorText: Colors.white,
); );
await controller.fetchExpenseDetails(); await controller.fetchExpenseDetails();
} else { } else {
Get.snackbar( showAppSnackbar(
'Error', title: 'Error',
'Failed to update status.', message: 'Failed to update status.',
backgroundColor: Colors.red.withOpacity(0.8), type: SnackbarType.error,
colorText: Colors.white,
); );
} }
} }
@ -491,7 +488,11 @@ class _InvoiceDocuments extends StatelessWidget {
if (await canLaunchUrl(url)) { if (await canLaunchUrl(url)) {
await launchUrl(url, mode: LaunchMode.externalApplication); await launchUrl(url, mode: LaunchMode.externalApplication);
} else { } else {
Get.snackbar("Error", "Could not open the document."); showAppSnackbar(
title: 'Error',
message: 'Could not open the document.',
type: SnackbarType.error,
);
} }
} }
}, },