first commit
This commit is contained in:
272
lib/Screens/Sales/batch_select_popup_sales.dart
Normal file
272
lib/Screens/Sales/batch_select_popup_sales.dart
Normal file
@@ -0,0 +1,272 @@
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||
import 'package:mobile_pos/Screens/Sales/provider/sales_cart_provider.dart';
|
||||
import 'package:mobile_pos/Screens/Products/Model/product_model.dart';
|
||||
import 'package:mobile_pos/constant.dart';
|
||||
import 'package:mobile_pos/generated/l10n.dart' as lang;
|
||||
import '../../currency.dart';
|
||||
import '../../model/add_to_cart_model.dart';
|
||||
import '../../service/check_user_role_permission_provider.dart';
|
||||
|
||||
Future<void> showAddItemPopup({
|
||||
required BuildContext mainContext,
|
||||
required Product productModel,
|
||||
required WidgetRef ref,
|
||||
required String? customerType,
|
||||
required bool fromPOSSales,
|
||||
}) async {
|
||||
TextEditingController _searchController = TextEditingController();
|
||||
final product = productModel;
|
||||
final permissionService = PermissionService(ref);
|
||||
List<SaleCartModel> tempCartItemList = [];
|
||||
List<TextEditingController> controllers = [];
|
||||
if (product.stocks?.isNotEmpty ?? false) {
|
||||
final cartList = ref.read(cartNotifier).cartItemList;
|
||||
|
||||
for (var element in product.stocks!) {
|
||||
num sentProductPrice;
|
||||
|
||||
if (customerType != null) {
|
||||
if (customerType.contains('Dealer')) {
|
||||
sentProductPrice = element.productDealerPrice ?? 0;
|
||||
} else if (customerType.contains('Wholesaler')) {
|
||||
sentProductPrice = element.productWholeSalePrice ?? 0;
|
||||
} else if (customerType.contains('Supplier')) {
|
||||
sentProductPrice = element.productPurchasePrice ?? 0;
|
||||
} else {
|
||||
sentProductPrice = element.productSalePrice ?? 0;
|
||||
}
|
||||
} else {
|
||||
sentProductPrice = element.productSalePrice ?? 0;
|
||||
}
|
||||
|
||||
final existingCartItem = cartList.firstWhere(
|
||||
(cartItem) => cartItem.productId == product.id && cartItem.stockId == element.id,
|
||||
orElse: () => SaleCartModel(productId: -1, batchName: '', stockId: 0), // default not-found case
|
||||
);
|
||||
|
||||
final existingQuantity = existingCartItem.productId != -1 ? existingCartItem.quantity : 0;
|
||||
|
||||
controllers.add(TextEditingController(text: existingQuantity.toString()));
|
||||
|
||||
tempCartItemList.add(SaleCartModel(
|
||||
batchName: element.batchNo ?? 'N/A',
|
||||
productName: product.productName,
|
||||
stockId: element.id ?? 0,
|
||||
unitPrice: sentProductPrice,
|
||||
productType: product.productType,
|
||||
productCode: product.productCode,
|
||||
productPurchasePrice: element.productPurchasePrice,
|
||||
stock: element.productStock,
|
||||
productId: product.id ?? 0,
|
||||
quantity: existingQuantity,
|
||||
));
|
||||
}
|
||||
}
|
||||
|
||||
showDialog(
|
||||
context: mainContext,
|
||||
barrierDismissible: false,
|
||||
builder: (BuildContext context) {
|
||||
return StatefulBuilder(builder: (context, setState) {
|
||||
void updateQuantity(int change, int index) {
|
||||
int currentQty = int.tryParse(controllers[index].text) ?? 0;
|
||||
int updatedQty = currentQty + change;
|
||||
|
||||
if (updatedQty > (tempCartItemList[index].stock ?? 0)) return;
|
||||
if (updatedQty < 0) return;
|
||||
setState(() {
|
||||
controllers[index].text = updatedQty.toString();
|
||||
});
|
||||
}
|
||||
|
||||
_searchController.addListener(
|
||||
() {
|
||||
setState(() {});
|
||||
},
|
||||
);
|
||||
return Dialog(
|
||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(16)),
|
||||
insetPadding: EdgeInsets.all(20),
|
||||
child: SingleChildScrollView(
|
||||
child: Padding(
|
||||
padding: const EdgeInsets.all(16),
|
||||
child: Column(
|
||||
mainAxisSize: MainAxisSize.min,
|
||||
children: [
|
||||
/// Title Row
|
||||
Row(
|
||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||
children: [
|
||||
Text(product.productName ?? '',
|
||||
maxLines: 1,
|
||||
overflow: TextOverflow.ellipsis,
|
||||
style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)),
|
||||
IconButton(
|
||||
icon: Icon(Icons.close),
|
||||
onPressed: () => Navigator.pop(context),
|
||||
)
|
||||
],
|
||||
),
|
||||
|
||||
SizedBox(height: 8),
|
||||
|
||||
/// Search Field
|
||||
TextField(
|
||||
controller: _searchController,
|
||||
decoration: InputDecoration(
|
||||
hintText: lang.S.of(context).searchBatchNo,
|
||||
prefixIcon: Icon(Icons.search),
|
||||
contentPadding: EdgeInsets.symmetric(vertical: 0, horizontal: 12),
|
||||
border: OutlineInputBorder(borderRadius: BorderRadius.circular(8)),
|
||||
),
|
||||
),
|
||||
|
||||
SizedBox(height: 16),
|
||||
|
||||
SizedBox(
|
||||
height: 250,
|
||||
child: SingleChildScrollView(
|
||||
child: Column(
|
||||
children: [
|
||||
/// Batch List
|
||||
...productModel.stocks!.map((item) => Visibility(
|
||||
visible: _searchController.text.isEmpty ||
|
||||
(item.batchNo?.toLowerCase().contains(_searchController.text.toLowerCase()) ??
|
||||
true),
|
||||
child: Column(
|
||||
children: [
|
||||
Row(
|
||||
children: [
|
||||
/// Batch Info
|
||||
Expanded(
|
||||
child: Column(
|
||||
crossAxisAlignment: CrossAxisAlignment.start,
|
||||
children: [
|
||||
Text('${lang.S.of(context).batch}: ${item.batchNo ?? 'N/A'}',
|
||||
style: TextStyle(fontWeight: FontWeight.w600)),
|
||||
Text('${lang.S.of(context).stock}: ${item.productStock}',
|
||||
style: TextStyle(color: Colors.green)),
|
||||
],
|
||||
),
|
||||
),
|
||||
|
||||
/// Price
|
||||
// if (permissionService.hasPermission(Permit.salesPriceView.value))
|
||||
Text('$currency${item.productSalePrice}',
|
||||
style: TextStyle(fontWeight: FontWeight.w600)),
|
||||
|
||||
SizedBox(width: 12),
|
||||
|
||||
/// Quantity Controller with Round Buttons
|
||||
Row(
|
||||
children: [
|
||||
/// - Button
|
||||
InkWell(
|
||||
onTap: () => updateQuantity(-1, productModel.stocks?.indexOf(item) ?? 0),
|
||||
borderRadius: BorderRadius.circular(20),
|
||||
child: Container(
|
||||
padding: EdgeInsets.all(2),
|
||||
decoration: BoxDecoration(
|
||||
shape: BoxShape.circle,
|
||||
color: Colors.grey.shade200,
|
||||
),
|
||||
child: Icon(Icons.remove, size: 16),
|
||||
),
|
||||
),
|
||||
|
||||
SizedBox(width: 8),
|
||||
|
||||
/// Quantity TextField
|
||||
Container(
|
||||
width: 60,
|
||||
height: 32,
|
||||
alignment: Alignment.center,
|
||||
child: TextFormField(
|
||||
controller: controllers[productModel.stocks?.indexOf(item) ?? 0],
|
||||
textAlign: TextAlign.center,
|
||||
keyboardType: TextInputType.number,
|
||||
style: TextStyle(fontSize: 14),
|
||||
decoration: InputDecoration(
|
||||
contentPadding: EdgeInsets.zero,
|
||||
isDense: true,
|
||||
border: OutlineInputBorder(
|
||||
borderRadius: BorderRadius.circular(6),
|
||||
),
|
||||
),
|
||||
onChanged: (val) {
|
||||
final parsed = int.tryParse(val);
|
||||
if (parsed == null ||
|
||||
parsed < 0 ||
|
||||
parsed > (item.productStock ?? 0)) {
|
||||
controllers[productModel.stocks?.indexOf(item) ?? 0].text = '';
|
||||
}
|
||||
},
|
||||
),
|
||||
),
|
||||
|
||||
SizedBox(width: 8),
|
||||
|
||||
/// + Button
|
||||
InkWell(
|
||||
onTap: () => updateQuantity(1, productModel.stocks?.indexOf(item) ?? 0),
|
||||
borderRadius: BorderRadius.circular(20),
|
||||
child: Container(
|
||||
padding: EdgeInsets.all(2),
|
||||
decoration: BoxDecoration(
|
||||
shape: BoxShape.circle,
|
||||
color: Colors.grey.shade200,
|
||||
),
|
||||
child: Icon(Icons.add, size: 16),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
],
|
||||
),
|
||||
Divider(),
|
||||
],
|
||||
),
|
||||
)),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
|
||||
SizedBox(height: 16),
|
||||
|
||||
/// Add to Cart Button
|
||||
SizedBox(
|
||||
width: double.infinity,
|
||||
child: ElevatedButton(
|
||||
style: ElevatedButton.styleFrom(
|
||||
backgroundColor: kMainColor,
|
||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(8)),
|
||||
padding: EdgeInsets.symmetric(vertical: 14),
|
||||
),
|
||||
onPressed: () {
|
||||
for (var element in tempCartItemList) {
|
||||
element.quantity = num.tryParse(controllers[tempCartItemList.indexOf(element)].text) ?? 0;
|
||||
}
|
||||
|
||||
tempCartItemList.removeWhere((element) => element.quantity <= 0);
|
||||
for (var element in tempCartItemList) {
|
||||
ref
|
||||
.read(cartNotifier)
|
||||
.addToCartRiverPod(cartItem: element, fromEditSales: false, isVariant: true);
|
||||
}
|
||||
if (!fromPOSSales) Navigator.pop(context);
|
||||
Navigator.pop(context);
|
||||
},
|
||||
child: Text(lang.S.of(context).addedToCart, style: TextStyle(fontSize: 16, color: Colors.white)),
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
),
|
||||
);
|
||||
});
|
||||
},
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user