pedido_view_model.dart 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  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. comentario: item.comentario,
  104. costoUnitario: item.producto.precio.toString(),
  105. sincronizado: null,
  106. );
  107. // Guardar o actualizar el producto
  108. int idPedidoProducto =
  109. await repoPedidoProducto.guardarPedidoProducto(pedidoProducto);
  110. // Manejar los toppings seleccionados
  111. for (var entry in item.selectedToppings.entries) {
  112. int categoriaId = entry.key;
  113. for (int toppingId in entry.value) {
  114. PedidoProductoTopping topping = PedidoProductoTopping(
  115. idPedidoProducto: idPedidoProducto,
  116. idTopping: toppingId,
  117. idCategoria: categoriaId,
  118. );
  119. await repoPedidoProductoTopping.guardarLocal(topping);
  120. }
  121. }
  122. }
  123. notifyListeners();
  124. return true;
  125. } catch (e) {
  126. print("Error al guardar el pedido con productos: $e");
  127. return false;
  128. }
  129. }
  130. Future<void> fetchLocalPedidos({int page = 1}) async {
  131. _isLoading = true;
  132. _currentPage = page;
  133. notifyListeners();
  134. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  135. _totalPedidos = await repoPedido.contarPedidos();
  136. int offset = (_limit * (page - 1));
  137. List<Pedido> paginatedPedidos =
  138. await repoPedido.obtenerPedidosPaginados(_limit, offset);
  139. _pedidos = paginatedPedidos;
  140. _isLoading = false;
  141. notifyListeners();
  142. }
  143. void nextPage() {
  144. if (_currentPage < totalPages) {
  145. fetchLocalPedidosForScreen(page: _currentPage + 1);
  146. }
  147. }
  148. void previousPage() {
  149. if (_currentPage > 1) {
  150. fetchLocalPedidosForScreen(page: _currentPage - 1);
  151. }
  152. }
  153. Future<void> eliminarProductoDelPedido(int idProducto, int idPedido) async {
  154. RepoService<PedidoProducto> repoPedidoProducto =
  155. RepoService<PedidoProducto>();
  156. PedidoProducto? producto =
  157. await repoPedidoProducto.obtenerPorIdPedidoYProducto(
  158. idPedido: idPedido,
  159. idProducto: idProducto,
  160. );
  161. if (producto != null) {
  162. producto.eliminado = DateTime.now().toUtc();
  163. await repoPedidoProducto.guardar(producto);
  164. notifyListeners();
  165. }
  166. }
  167. Future<void> fetchLocalPedidosForScreen({int page = 1}) async {
  168. setIsLoading(true);
  169. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  170. _currentPage = page;
  171. var db = await RepoService().db;
  172. int? count = Sqflite.firstIntValue(
  173. await db!.rawQuery('SELECT COUNT(*) FROM Pedido'));
  174. _totalPedidos = count ?? 0;
  175. int offset = (_limit * (page - 1));
  176. List<Pedido> localPedidos =
  177. await repoPedido.obtenerPedidosPaginados(_limit, offset);
  178. _pedidos = localPedidos;
  179. setIsLoading(false);
  180. notifyListeners();
  181. }
  182. Future<List<Pedido>> fetchAllLocalPedidos() async {
  183. setIsLoading(true);
  184. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  185. List<Pedido> allPedidos = await repoPedido.obtenerTodos();
  186. setIsLoading(false);
  187. return allPedidos;
  188. }
  189. Future<List<Pedido>> fetchPedidosPorFechaSinLimit(
  190. DateTime startDate, DateTime endDate) async {
  191. setIsLoading(true);
  192. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  193. List<Pedido> pedidos = await repoPedido.buscarPorFecha(startDate, endDate);
  194. setIsLoading(false);
  195. return pedidos;
  196. }
  197. Future<Pedido?> fetchPedidoConProductos(int idPedido) async {
  198. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  199. Pedido? pedido = await repoPedido.obtenerPorId(idPedido);
  200. if (pedido != null) {
  201. RepoService<PedidoProducto> repoProducto = RepoService<PedidoProducto>();
  202. RepoService<Producto> repoProductoInfo = RepoService<Producto>();
  203. RepoService<Propinas> repoPropina = RepoService<Propinas>();
  204. List<PedidoProducto> productos =
  205. await repoProducto.obtenerPorIdPedido(idPedido);
  206. productos = productos.where((p) => p.eliminado == null).toList();
  207. for (var producto in productos) {
  208. Producto? prodInfo =
  209. await repoProductoInfo.obtenerProductoPorId(producto.idProducto!);
  210. if (prodInfo != null) {
  211. producto.producto = prodInfo;
  212. }
  213. RepoService<PedidoProductoTopping> repoTopping =
  214. RepoService<PedidoProductoTopping>();
  215. List<PedidoProductoTopping> toppings =
  216. await repoTopping.obtenerToppingsPorPedidoProducto(producto.id!);
  217. for (var topping in toppings) {
  218. Producto? toppingInfo =
  219. await repoProductoInfo.obtenerProductoPorId(topping.idTopping!);
  220. if (toppingInfo != null) {
  221. topping.topping = toppingInfo;
  222. }
  223. }
  224. producto.toppings = toppings;
  225. List<Propinas> propinas = await repoPropina.obtenerTodos<Propinas>(
  226. where: 'idPedido = ?', whereArgs: [idPedido]);
  227. pedido.propinas = propinas;
  228. }
  229. pedido.productos = productos;
  230. }
  231. return pedido;
  232. }
  233. Future<void> buscarPedidosPorFolio(String folio) async {
  234. setIsLoading(true);
  235. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  236. List<Pedido> localPedidos = await repoPedido.buscarPorFolio(folio);
  237. _pedidos = localPedidos;
  238. setIsLoading(false);
  239. notifyListeners();
  240. }
  241. Future<void> buscarPedidosPorFecha(
  242. DateTime startDate, DateTime endDate) async {
  243. setIsLoading(true);
  244. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  245. List<Pedido> localPedidos =
  246. await repoPedido.buscarPorFecha(startDate, endDate);
  247. _pedidos = localPedidos;
  248. setIsLoading(false);
  249. notifyListeners();
  250. }
  251. Future<List<Pedido>> buscarPorFecha(
  252. DateTime startDate, DateTime endDate) async {
  253. setIsLoading(true);
  254. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  255. print('Consulta SQL de pedidos desde: $startDate hasta: $endDate');
  256. List<Pedido> pedidos = await repoPedido.buscarPorFecha(startDate, endDate);
  257. print('Pedidos obtenidos desde la base de datos: ${pedidos.length}');
  258. pedidos.forEach((pedido) => print(
  259. 'Pedido Folio: ${pedido.folio}, Estatus: ${pedido.estatus}, Total: ${pedido.totalPedido}'));
  260. setIsLoading(false);
  261. notifyListeners();
  262. return pedidos;
  263. }
  264. Future<void> cancelarPedido(int idPedido) async {
  265. var db = await RepoService().db;
  266. await db?.update(
  267. 'Pedido',
  268. {
  269. 'estatus': 'CANCELADO',
  270. 'sincronizado': null,
  271. },
  272. where: 'id = ?',
  273. whereArgs: [idPedido],
  274. );
  275. fetchLocalPedidosForScreen();
  276. }
  277. Future<bool> sincronizarPedidos() async {
  278. List<Pedido> pedidosNoSincronizados =
  279. await fetchAllLocalPedidosOrdenadosPorFecha();
  280. if (pedidosNoSincronizados.isNotEmpty) {
  281. Pedido pedidoNoSincronizado = pedidosNoSincronizados.first;
  282. if (pedidoNoSincronizado.productos.isEmpty) {
  283. pedidoNoSincronizado =
  284. await fetchPedidoConProductos(pedidoNoSincronizado.id) ??
  285. pedidoNoSincronizado;
  286. }
  287. Map<String, dynamic> pedidoJson =
  288. await prepararPedidoParaApi(pedidoNoSincronizado);
  289. print('JSON enviado: $pedidoJson');
  290. var response = ApiResponse(await BaseService()
  291. .post('/pos/pedido/sincronizar', body: pedidoJson));
  292. if (response.isOk && response.detalle != null) {
  293. int idWeb = response.detalle!['id'];
  294. String sincronizado = response.detalle!['sincronizado'];
  295. await actualizarPedidoSincronizado(
  296. pedidoNoSincronizado.id!, idWeb, sincronizado);
  297. return true;
  298. } else {
  299. print('Error en la sincronización del pedido: ${response.mensaje}');
  300. return true;
  301. }
  302. } else {
  303. print('No se encontraron pedidos no sincronizados.');
  304. return false;
  305. }
  306. }
  307. Future<void> actualizarPedidoSincronizado(
  308. int idPedido, int idWeb, String sincronizado) async {
  309. var db = await RepoService().db;
  310. await db!.update(
  311. 'Pedido',
  312. {
  313. 'idWeb': idWeb,
  314. 'sincronizado': sincronizado,
  315. },
  316. where: 'id = ?',
  317. whereArgs: [idPedido],
  318. );
  319. }
  320. Future<List<Pedido>> fetchAllLocalPedidosOrdenadosPorFecha() async {
  321. setIsLoading(true);
  322. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  323. List<Pedido> allPedidos =
  324. await repoPedido.obtenerPedidosOrdenadosPorFecha();
  325. setIsLoading(false);
  326. return allPedidos;
  327. }
  328. Future<List<Pedido>> fetchPedidosNuevosByCorteId(String idCorteCaja) async {
  329. var db = await RepoService().db;
  330. List<Map<String, dynamic>> maps = await db!.query(
  331. 'Pedido',
  332. where: 'estatus = ? AND idCorteCaja = ? AND eliminado IS NULL',
  333. whereArgs: ['NUEVO', idCorteCaja],
  334. );
  335. if (maps.isNotEmpty) {
  336. return maps.map((map) => Pedido.fromJson(map)).toList();
  337. }
  338. return [];
  339. }
  340. Future<void> actualizarCorteCajaEnPedidos(
  341. List<int> idsPedidos, String nuevoIdCorte) async {
  342. var db = await RepoService().db;
  343. for (int idPedido in idsPedidos) {
  344. await db!.update(
  345. 'Pedido',
  346. {'idCorteCaja': nuevoIdCorte, 'sincronizado': null},
  347. where: 'id = ?',
  348. whereArgs: [idPedido],
  349. );
  350. }
  351. }
  352. Future<Map<String, dynamic>> prepararPedidoParaApi(Pedido pedido) async {
  353. String? claveSucursal =
  354. await RepoService().obtenerClaveSucursalSeleccionada();
  355. Map<String, dynamic> apiMap = pedido.toApi();
  356. apiMap['claveSucursal'] = claveSucursal;
  357. if (pedido.idWeb != null && pedido.idWeb! > 0) {
  358. apiMap['idWeb'] = pedido.idWeb;
  359. }
  360. return apiMap;
  361. }
  362. }