pedido_view_model.dart 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414
  1. import 'package:collection/collection.dart';
  2. import 'package:flutter/material.dart';
  3. import 'package:intl/intl.dart';
  4. import 'package:sqflite/sqflite.dart';
  5. import '../data/api_response.dart';
  6. import '../models/models.dart';
  7. import '../services/services.dart';
  8. class PedidoViewModel extends ChangeNotifier {
  9. String _busqueda = "";
  10. String get busqueda => _busqueda;
  11. List<Pedido> _pedidos = [];
  12. Pedido? _selectedPedido;
  13. bool _isLoading = false;
  14. int _currentPage = 1;
  15. int _totalPedidos = 0;
  16. int _limit = 10;
  17. int get currentPage => _currentPage;
  18. int get totalPedidos => _totalPedidos;
  19. int get totalPages => (_totalPedidos / _limit).ceil();
  20. List<Pedido> get pedidos => _pedidos;
  21. Pedido? get selectedPedido => _selectedPedido;
  22. bool get isLoading => _isLoading;
  23. void setIsLoading(bool loading) {
  24. _isLoading = loading;
  25. notifyListeners();
  26. }
  27. Future<bool> guardarPedidoLocal({required Pedido pedido}) async {
  28. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  29. if (pedido.id == null || pedido.id == 0) {
  30. int nextFolio = await repoPedido.obtenerProximoFolio();
  31. pedido.folio = nextFolio;
  32. }
  33. try {
  34. // Verifica si el pedido ya existe en la base de datos
  35. Pedido? pedidoExistente = await repoPedido.obtenerPorId(pedido.id!);
  36. if (pedidoExistente != null) {
  37. // Si existe, actualiza
  38. await repoPedido.guardar(pedido);
  39. } else {
  40. // Si no existe, inserta como nuevo
  41. int idPedido = await repoPedido.guardarLocal(pedido);
  42. pedido.id = idPedido;
  43. }
  44. RepoService<PedidoProducto> repoPedidoProducto =
  45. RepoService<PedidoProducto>();
  46. RepoService<PedidoProductoTopping> repoPedidoProductoTopping =
  47. RepoService<PedidoProductoTopping>();
  48. // Manejo de productos asociados al pedido
  49. for (var producto in pedido.productos) {
  50. PedidoProducto pedidoProducto = PedidoProducto(
  51. idPedido: pedido.id,
  52. idProducto: producto.idProducto,
  53. cantidad: producto.cantidad,
  54. costoUnitario: producto.costoUnitario,
  55. comentario: producto.comentario,
  56. );
  57. if (producto.id != null && producto.id! > 0) {
  58. await repoPedidoProducto.guardarLocal(pedidoProducto);
  59. } else {
  60. int idPedidoProducto =
  61. await repoPedidoProducto.guardarLocal(pedidoProducto);
  62. for (var topping in producto.toppings) {
  63. PedidoProductoTopping pedidoProductoTopping = PedidoProductoTopping(
  64. idPedidoProducto: idPedidoProducto,
  65. idTopping: topping.idTopping,
  66. idCategoria: topping.idCategoria,
  67. );
  68. await repoPedidoProductoTopping.guardarLocal(pedidoProductoTopping);
  69. }
  70. }
  71. }
  72. notifyListeners();
  73. return true;
  74. } catch (e) {
  75. print('Error al guardar el pedido: $e');
  76. return false;
  77. }
  78. }
  79. Future<bool> guardarPedidoConProductos({
  80. required Pedido pedido,
  81. required List<ItemCarrito> carrito,
  82. }) async {
  83. try {
  84. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  85. RepoService<PedidoProducto> repoPedidoProducto =
  86. RepoService<PedidoProducto>();
  87. RepoService<PedidoProductoTopping> repoPedidoProductoTopping =
  88. RepoService<PedidoProductoTopping>();
  89. // Guardar el pedido en la base de datos
  90. if (pedido.id == null || pedido.id == 0) {
  91. int nextFolio = await repoPedido.obtenerProximoFolio();
  92. pedido.folio = nextFolio;
  93. pedido.id = await repoPedido.guardarLocal(pedido);
  94. } else {
  95. await repoPedido.guardar(pedido);
  96. }
  97. // Manejar los productos del carrito
  98. for (var item in carrito) {
  99. PedidoProducto pedidoProducto = PedidoProducto(
  100. idPedido: pedido.id,
  101. idProducto: item.producto.id,
  102. cantidad: item.cantidad,
  103. costoUnitario: item.producto.precio.toString(),
  104. sincronizado: null,
  105. );
  106. // Guardar o actualizar el producto
  107. int idPedidoProducto =
  108. await repoPedidoProducto.guardarPedidoProducto(pedidoProducto);
  109. // Manejar los toppings seleccionados
  110. for (var entry in item.selectedToppings.entries) {
  111. int categoriaId = entry.key;
  112. for (int toppingId in entry.value) {
  113. PedidoProductoTopping topping = PedidoProductoTopping(
  114. idPedidoProducto: idPedidoProducto,
  115. idTopping: toppingId,
  116. idCategoria: categoriaId,
  117. );
  118. await repoPedidoProductoTopping.guardarLocal(topping);
  119. }
  120. }
  121. }
  122. notifyListeners();
  123. return true;
  124. } catch (e) {
  125. print("Error al guardar el pedido con productos: $e");
  126. return false;
  127. }
  128. }
  129. Future<void> fetchLocalPedidos({int page = 1}) async {
  130. _isLoading = true;
  131. _currentPage = page;
  132. notifyListeners();
  133. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  134. _totalPedidos = await repoPedido.contarPedidos();
  135. int offset = (_limit * (page - 1));
  136. List<Pedido> paginatedPedidos =
  137. await repoPedido.obtenerPedidosPaginados(_limit, offset);
  138. _pedidos = paginatedPedidos;
  139. _isLoading = false;
  140. notifyListeners();
  141. }
  142. void nextPage() {
  143. if (_currentPage < totalPages) {
  144. fetchLocalPedidosForScreen(page: _currentPage + 1);
  145. }
  146. }
  147. void previousPage() {
  148. if (_currentPage > 1) {
  149. fetchLocalPedidosForScreen(page: _currentPage - 1);
  150. }
  151. }
  152. Future<void> eliminarProductoDelPedido(int idProducto, int idPedido) async {
  153. RepoService<PedidoProducto> repoPedidoProducto =
  154. RepoService<PedidoProducto>();
  155. PedidoProducto? producto =
  156. await repoPedidoProducto.obtenerPorIdPedidoYProducto(
  157. idPedido: idPedido,
  158. idProducto: idProducto,
  159. );
  160. if (producto != null) {
  161. producto.eliminado = DateTime.now().toUtc();
  162. await repoPedidoProducto.guardar(producto);
  163. notifyListeners();
  164. }
  165. }
  166. Future<void> fetchLocalPedidosForScreen({int page = 1}) async {
  167. setIsLoading(true);
  168. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  169. _currentPage = page;
  170. var db = await RepoService().db;
  171. int? count = Sqflite.firstIntValue(
  172. await db!.rawQuery('SELECT COUNT(*) FROM Pedido'));
  173. _totalPedidos = count ?? 0;
  174. int offset = (_limit * (page - 1));
  175. List<Pedido> localPedidos =
  176. await repoPedido.obtenerPedidosPaginados(_limit, offset);
  177. _pedidos = localPedidos;
  178. setIsLoading(false);
  179. notifyListeners();
  180. }
  181. Future<List<Pedido>> fetchAllLocalPedidos() async {
  182. setIsLoading(true);
  183. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  184. List<Pedido> allPedidos = await repoPedido.obtenerTodos();
  185. setIsLoading(false);
  186. return allPedidos;
  187. }
  188. Future<List<Pedido>> fetchPedidosPorFechaSinLimit(
  189. DateTime startDate, DateTime endDate) async {
  190. setIsLoading(true);
  191. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  192. List<Pedido> pedidos = await repoPedido.buscarPorFecha(startDate, endDate);
  193. setIsLoading(false);
  194. return pedidos;
  195. }
  196. Future<Pedido?> fetchPedidoConProductos(int idPedido) async {
  197. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  198. Pedido? pedido = await repoPedido.obtenerPorId(idPedido);
  199. if (pedido != null) {
  200. RepoService<PedidoProducto> repoProducto = RepoService<PedidoProducto>();
  201. RepoService<Producto> repoProductoInfo = RepoService<Producto>();
  202. RepoService<Propinas> repoPropina = RepoService<Propinas>();
  203. List<PedidoProducto> productos =
  204. await repoProducto.obtenerPorIdPedido(idPedido);
  205. productos = productos.where((p) => p.eliminado == null).toList();
  206. for (var producto in productos) {
  207. Producto? prodInfo =
  208. await repoProductoInfo.obtenerProductoPorId(producto.idProducto!);
  209. if (prodInfo != null) {
  210. producto.producto = prodInfo;
  211. }
  212. RepoService<PedidoProductoTopping> repoTopping =
  213. RepoService<PedidoProductoTopping>();
  214. List<PedidoProductoTopping> toppings =
  215. await repoTopping.obtenerToppingsPorPedidoProducto(producto.id!);
  216. for (var topping in toppings) {
  217. Producto? toppingInfo =
  218. await repoProductoInfo.obtenerProductoPorId(topping.idTopping!);
  219. if (toppingInfo != null) {
  220. topping.topping = toppingInfo;
  221. }
  222. }
  223. producto.toppings = toppings;
  224. // List<Propinas> propinas = await repoPropina.obtenerTodos<Propinas>(
  225. // where: 'idPedido = ?', whereArgs: [idPedido]);
  226. // pedido.propinas = propinas;
  227. }
  228. pedido.productos = productos;
  229. }
  230. return pedido;
  231. }
  232. Future<void> buscarPedidosPorFolio(String folio) async {
  233. setIsLoading(true);
  234. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  235. List<Pedido> localPedidos = await repoPedido.buscarPorFolio(folio);
  236. _pedidos = localPedidos;
  237. setIsLoading(false);
  238. notifyListeners();
  239. }
  240. Future<void> buscarPedidosPorFecha(
  241. DateTime startDate, DateTime endDate) async {
  242. setIsLoading(true);
  243. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  244. List<Pedido> localPedidos =
  245. await repoPedido.buscarPorFecha(startDate, endDate);
  246. _pedidos = localPedidos;
  247. setIsLoading(false);
  248. notifyListeners();
  249. }
  250. Future<List<Pedido>> buscarPorFecha(
  251. DateTime startDate, DateTime endDate) async {
  252. setIsLoading(true);
  253. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  254. print('Consulta SQL de pedidos desde: $startDate hasta: $endDate');
  255. List<Pedido> pedidos = await repoPedido.buscarPorFecha(startDate, endDate);
  256. print('Pedidos obtenidos desde la base de datos: ${pedidos.length}');
  257. pedidos.forEach((pedido) => print(
  258. 'Pedido Folio: ${pedido.folio}, Estatus: ${pedido.estatus}, Total: ${pedido.totalPedido}'));
  259. setIsLoading(false);
  260. notifyListeners();
  261. return pedidos;
  262. }
  263. Future<void> cancelarPedido(int idPedido) async {
  264. var db = await RepoService().db;
  265. await db?.update(
  266. 'Pedido',
  267. {
  268. 'estatus': 'CANCELADO',
  269. 'sincronizado': null,
  270. },
  271. where: 'id = ?',
  272. whereArgs: [idPedido],
  273. );
  274. fetchLocalPedidosForScreen();
  275. }
  276. Future<bool> sincronizarPedidos() async {
  277. List<Pedido> pedidosNoSincronizados =
  278. await fetchAllLocalPedidosOrdenadosPorFecha();
  279. if (pedidosNoSincronizados.isNotEmpty) {
  280. Pedido pedidoNoSincronizado = pedidosNoSincronizados.first;
  281. if (pedidoNoSincronizado.productos.isEmpty) {
  282. pedidoNoSincronizado =
  283. await fetchPedidoConProductos(pedidoNoSincronizado.id) ??
  284. pedidoNoSincronizado;
  285. }
  286. Map<String, dynamic> pedidoJson =
  287. await prepararPedidoParaApi(pedidoNoSincronizado);
  288. //print('JSON enviado: $pedidoJson');
  289. var response = ApiResponse(await BaseService()
  290. .post('/pos/pedido/sincronizar', body: pedidoJson));
  291. if (response.isOk && response.detalle != null) {
  292. int idWeb = response.detalle!['id'];
  293. String sincronizado = response.detalle!['sincronizado'];
  294. await actualizarPedidoSincronizado(
  295. pedidoNoSincronizado.id!, idWeb, sincronizado);
  296. return true;
  297. } else {
  298. print('Error en la sincronización del pedido: ${response.mensaje}');
  299. return true;
  300. }
  301. } else {
  302. print('No se encontraron pedidos no sincronizados.');
  303. return false;
  304. }
  305. }
  306. Future<void> actualizarPedidoSincronizado(
  307. int idPedido, int idWeb, String sincronizado) async {
  308. var db = await RepoService().db;
  309. await db!.update(
  310. 'Pedido',
  311. {
  312. 'idWeb': idWeb,
  313. 'sincronizado': sincronizado,
  314. },
  315. where: 'id = ?',
  316. whereArgs: [idPedido],
  317. );
  318. }
  319. Future<List<Pedido>> fetchAllLocalPedidosOrdenadosPorFecha() async {
  320. setIsLoading(true);
  321. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  322. List<Pedido> allPedidos =
  323. await repoPedido.obtenerPedidosOrdenadosPorFecha();
  324. setIsLoading(false);
  325. return allPedidos;
  326. }
  327. }
  328. Future<Map<String, dynamic>> prepararPedidoParaApi(Pedido pedido) async {
  329. String? claveSucursal =
  330. await RepoService().obtenerClaveSucursalSeleccionada();
  331. Map<String, dynamic> apiMap = pedido.toApi();
  332. apiMap['claveSucursal'] = claveSucursal;
  333. if (pedido.idWeb != null && pedido.idWeb! > 0) {
  334. apiMap['idWeb'] = pedido.idWeb;
  335. }
  336. return apiMap;
  337. }