pedido_view_model.dart 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  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> fetchLocalPedidosForScreen({int page = 1}) async {
  153. setIsLoading(true);
  154. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  155. _currentPage = page;
  156. var db = await RepoService().db;
  157. int? count = Sqflite.firstIntValue(
  158. await db!.rawQuery('SELECT COUNT(*) FROM Pedido'));
  159. _totalPedidos = count ?? 0;
  160. int offset = (_limit * (page - 1));
  161. List<Pedido> localPedidos =
  162. await repoPedido.obtenerPedidosPaginados(_limit, offset);
  163. _pedidos = localPedidos;
  164. setIsLoading(false);
  165. notifyListeners();
  166. }
  167. Future<List<Pedido>> fetchAllLocalPedidos() async {
  168. setIsLoading(true);
  169. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  170. List<Pedido> allPedidos = await repoPedido.obtenerTodos();
  171. setIsLoading(false);
  172. return allPedidos;
  173. }
  174. Future<List<Pedido>> fetchPedidosPorFechaSinLimit(
  175. DateTime startDate, DateTime endDate) async {
  176. setIsLoading(true);
  177. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  178. List<Pedido> pedidos = await repoPedido.buscarPorFecha(startDate, endDate);
  179. setIsLoading(false);
  180. return pedidos;
  181. }
  182. Future<Pedido?> fetchPedidoConProductos(int idPedido) async {
  183. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  184. Pedido? pedido = await repoPedido.obtenerPorId(idPedido);
  185. if (pedido != null) {
  186. RepoService<PedidoProducto> repoProducto = RepoService<PedidoProducto>();
  187. RepoService<Producto> repoProductoInfo = RepoService<Producto>();
  188. List<PedidoProducto> productos =
  189. await repoProducto.obtenerPorIdPedido(idPedido);
  190. for (var producto in productos) {
  191. Producto? prodInfo =
  192. await repoProductoInfo.obtenerProductoPorId(producto.idProducto!);
  193. if (prodInfo != null) {
  194. producto.producto = prodInfo;
  195. }
  196. RepoService<PedidoProductoTopping> repoTopping =
  197. RepoService<PedidoProductoTopping>();
  198. List<PedidoProductoTopping> toppings =
  199. await repoTopping.obtenerToppingsPorPedidoProducto(producto.id!);
  200. for (var topping in toppings) {
  201. Producto? toppingInfo =
  202. await repoProductoInfo.obtenerProductoPorId(topping.idTopping!);
  203. if (toppingInfo != null) {
  204. topping.topping = toppingInfo;
  205. }
  206. }
  207. producto.toppings = toppings;
  208. }
  209. pedido.productos = productos;
  210. }
  211. return pedido;
  212. }
  213. Future<void> buscarPedidosPorFolio(String folio) async {
  214. setIsLoading(true);
  215. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  216. List<Pedido> localPedidos = await repoPedido.buscarPorFolio(folio);
  217. _pedidos = localPedidos;
  218. setIsLoading(false);
  219. notifyListeners();
  220. }
  221. Future<void> buscarPedidosPorFecha(
  222. DateTime startDate, DateTime endDate) async {
  223. setIsLoading(true);
  224. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  225. List<Pedido> localPedidos =
  226. await repoPedido.buscarPorFecha(startDate, endDate);
  227. _pedidos = localPedidos;
  228. setIsLoading(false);
  229. notifyListeners();
  230. }
  231. Future<List<Pedido>> buscarPorFecha(
  232. DateTime startDate, DateTime endDate) async {
  233. setIsLoading(true);
  234. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  235. print('Consulta SQL de pedidos desde: $startDate hasta: $endDate');
  236. List<Pedido> pedidos = await repoPedido.buscarPorFecha(startDate, endDate);
  237. print('Pedidos obtenidos desde la base de datos: ${pedidos.length}');
  238. pedidos.forEach((pedido) => print(
  239. 'Pedido Folio: ${pedido.folio}, Estatus: ${pedido.estatus}, Total: ${pedido.totalPedido}'));
  240. setIsLoading(false);
  241. notifyListeners();
  242. return pedidos;
  243. }
  244. Future<void> cancelarPedido(int idPedido) async {
  245. var db = await RepoService().db;
  246. await db?.update(
  247. 'Pedido',
  248. {
  249. 'estatus': 'CANCELADO',
  250. 'sincronizado': null,
  251. },
  252. where: 'id = ?',
  253. whereArgs: [idPedido],
  254. );
  255. fetchLocalPedidosForScreen();
  256. }
  257. Future<bool> sincronizarPedidos() async {
  258. List<Pedido> pedidosNoSincronizados =
  259. await fetchAllLocalPedidosOrdenadosPorFecha();
  260. if (pedidosNoSincronizados.isNotEmpty) {
  261. Pedido pedidoNoSincronizado = pedidosNoSincronizados.first;
  262. if (pedidoNoSincronizado.productos.isEmpty) {
  263. pedidoNoSincronizado =
  264. await fetchPedidoConProductos(pedidoNoSincronizado.id) ??
  265. pedidoNoSincronizado;
  266. }
  267. Map<String, dynamic> pedidoJson =
  268. await prepararPedidoParaApi(pedidoNoSincronizado);
  269. // print('JSON enviado: $pedidoJson');
  270. var response = ApiResponse(await BaseService()
  271. .post('/pos/pedido/sincronizar', body: pedidoJson));
  272. if (response.isOk && response.detalle != null) {
  273. int idWeb = response.detalle!['id'];
  274. String sincronizado = response.detalle!['sincronizado'];
  275. await actualizarPedidoSincronizado(
  276. pedidoNoSincronizado.id!, idWeb, sincronizado);
  277. return true;
  278. } else {
  279. print('Error en la sincronización del pedido: ${response.mensaje}');
  280. return true;
  281. }
  282. } else {
  283. print('No se encontraron pedidos no sincronizados.');
  284. return false;
  285. }
  286. }
  287. Future<void> actualizarPedidoSincronizado(
  288. int idPedido, int idWeb, String sincronizado) async {
  289. var db = await RepoService().db;
  290. await db!.update(
  291. 'Pedido',
  292. {
  293. 'idWeb': idWeb,
  294. 'sincronizado': sincronizado,
  295. },
  296. where: 'id = ?',
  297. whereArgs: [idPedido],
  298. );
  299. }
  300. Future<List<Pedido>> fetchAllLocalPedidosOrdenadosPorFecha() async {
  301. setIsLoading(true);
  302. RepoService<Pedido> repoPedido = RepoService<Pedido>();
  303. List<Pedido> allPedidos =
  304. await repoPedido.obtenerPedidosOrdenadosPorFecha();
  305. setIsLoading(false);
  306. return allPedidos;
  307. }
  308. }
  309. Future<Map<String, dynamic>> prepararPedidoParaApi(Pedido pedido) async {
  310. String? claveSucursal =
  311. await RepoService().obtenerClaveSucursalSeleccionada();
  312. Map<String, dynamic> apiMap = pedido.toApi();
  313. apiMap['claveSucursal'] = claveSucursal;
  314. if (pedido.idWeb != null && pedido.idWeb! > 0) {
  315. apiMap['idWeb'] = pedido.idWeb;
  316. }
  317. return apiMap;
  318. }