models.py 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  1. from django.db import models
  2. from erp.models import InboundBill, MaterialDetail, OutboundBill,OutMaterialDetail
  3. from django.db.models import Sum
  4. from django.db.models.signals import post_save, post_delete, pre_save
  5. from django.dispatch import receiver
  6. from decimal import Decimal
  7. class BoundListModel(models.Model):
  8. STATUS =(
  9. ("100", '入库申请'), ("101", '入库同意'), ("102", '组盘中'), ("103", '部分入库'), ("104", '已入库'),
  10. ("111", '质检合格'),
  11. ("200", '出库申请'), ("201", '出库同意'), ("202", '出库中'), ("203", '部分出库'), ("204", '已出库'),
  12. ("300", '完成')
  13. )
  14. audit_status = models.BooleanField(default=False, verbose_name='审核状态')
  15. bound_month = models.CharField(max_length=255, verbose_name="月份")
  16. bound_date = models.DateField(verbose_name="单据日期")
  17. bound_code = models.CharField(max_length=255, verbose_name="出入库编号",unique=True)
  18. bound_code_type = models.CharField(max_length=255, verbose_name="单据类型")
  19. bound_bs_type = models.CharField(max_length=255, verbose_name="业务类型")
  20. bound_type = models.CharField(max_length=255, verbose_name="出入库类型")
  21. bound_desc = models.CharField(default='', max_length=255, verbose_name="出入库描述")
  22. bound_department = models.CharField(max_length=255, verbose_name="部门")
  23. base_type = models.IntegerField(default=0, verbose_name="基准类型")
  24. bound_status = models.CharField(max_length=255, verbose_name="单据状态",choices=STATUS, default='100')
  25. note = models.CharField(default='', max_length=255, verbose_name="备注")
  26. creater = models.CharField(default='uesr', max_length=255, verbose_name="Who Created")
  27. openid = models.CharField(max_length=255, verbose_name="Openid")
  28. is_delete = models.BooleanField(default=False, verbose_name='Delete Label')
  29. create_time = models.DateTimeField(auto_now_add=True, verbose_name="Create Time")
  30. update_time = models.DateTimeField(auto_now=True, blank=True, null=True, verbose_name="Update Time")
  31. relate_bill = models.ForeignKey(InboundBill, on_delete=models.CASCADE, verbose_name="关联单据", related_name='bound_list',null=True, blank=True)
  32. relate_out_bill = models.ForeignKey(OutboundBill, on_delete=models.CASCADE, verbose_name="关联出库单据", related_name='bound_out_list',null=True, blank=True)
  33. class Meta:
  34. db_table = 'boundlist'
  35. verbose_name = 'Bound List'
  36. verbose_name_plural = "Bound List"
  37. ordering = ['-id']
  38. class BoundBatchModel(models.Model):
  39. CONTAINER_STATUS = (
  40. (0, '入库申请'),
  41. (1, '入库中'),
  42. (2, '部分入库'),
  43. (3, '在库'),
  44. (4, '出库中'),
  45. (5, '部分出库'),
  46. (6, '已出库'),
  47. )
  48. CHECK_STATUS=(
  49. (0, '未质检'),
  50. (1, '已质检'),
  51. (2, '质检不合格'),
  52. )
  53. bound_number = models.CharField(max_length=255, verbose_name="入库批次号",blank=False, null=False, unique=True)
  54. sourced_number = models.CharField(max_length=255, verbose_name="来源批次号",blank=True, null=True)
  55. bound_month = models.CharField(max_length=255, verbose_name="月份")
  56. bound_batch_order = models.IntegerField(default=0, verbose_name="批次顺序")
  57. warehouse_code = models.CharField(max_length=255, verbose_name="仓库编码")
  58. warehouse_name = models.CharField(max_length=255, verbose_name="仓库名称")
  59. goods_code = models.CharField(max_length=255, verbose_name="商品编码")
  60. goods_desc = models.CharField(max_length=255, verbose_name="商品描述")
  61. goods_std = models.CharField(default='待填写', max_length=255, verbose_name="商品标准",blank=True, null=True)
  62. goods_unit = models.CharField(default='待填写', max_length=255, verbose_name="商品单位")
  63. goods_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="商品数量")
  64. goods_package = models.CharField(default='待填写', max_length=255, verbose_name="商品包装")
  65. goods_in_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="组盘入库数量")
  66. goods_in_location_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="库位入库数量")
  67. goods_out_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="出库数量")
  68. goods_reserve_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="预定出库数量")
  69. status = models.IntegerField(choices=CONTAINER_STATUS, default=0, verbose_name='批次状态')
  70. container_number = models.IntegerField( default=0, verbose_name="托盘数目")
  71. goods_weight = models.FloatField(default=0, verbose_name="商品单重")
  72. goods_total_weight = models.FloatField(default=0, verbose_name="商品总重")
  73. note = models.CharField(default='', max_length=255, verbose_name="备注")
  74. creater = models.CharField(default='uesr', max_length=255, verbose_name="Who Created")
  75. openid = models.CharField(max_length=255, verbose_name="Openid")
  76. is_delete = models.BooleanField(default=False, verbose_name='Delete Label')
  77. create_time = models.DateTimeField(auto_now_add=True, verbose_name="Create Time")
  78. update_time = models.DateTimeField(auto_now=True, blank=True, null=True, verbose_name="Update Time")
  79. relate_material = models.ForeignKey(MaterialDetail, on_delete=models.CASCADE, verbose_name="关联物料", related_name='bound_batch',null=True, blank=True)
  80. check_status = models.IntegerField(choices=CHECK_STATUS, default=0, verbose_name='质检状态')
  81. check_user = models.CharField(default='', max_length=255, verbose_name="质检人")
  82. check_time = models.DateTimeField(auto_now=True, blank=True, null=True, verbose_name="质检时间")
  83. class Meta:
  84. db_table = 'boundbatch'
  85. verbose_name = 'Bound Batch'
  86. verbose_name_plural = "Bound Batch"
  87. ordering = ['-id']
  88. def __str__(self):
  89. return f"{self.bound_number} - {self.goods_code} - {self.goods_desc}"
  90. class OutBoundDemandModel(models.Model):
  91. OUT_TYPE = (
  92. (0, '发货出库'),
  93. (4, '其他出库'),
  94. )
  95. bound_list = models.ForeignKey(BoundListModel, on_delete=models.CASCADE, verbose_name="Bound List", related_name='out_demand')
  96. goods_code = models.CharField(max_length=255, verbose_name="商品编码")
  97. goods_desc = models.CharField(max_length=255, verbose_name="商品描述")
  98. goods_std = models.CharField(default='待填写', max_length=255, verbose_name="商品标准",blank=True, null=True)
  99. goods_unit = models.CharField(default='待填写', max_length=255, verbose_name="商品单位")
  100. goods_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="计划数量")
  101. working = models.BooleanField(default=False, verbose_name="是否在工作")
  102. creater = models.CharField(default='uesr', max_length=255, verbose_name="Who Created")
  103. create_time = models.DateTimeField(auto_now_add=True, verbose_name="Create Time")
  104. update_time = models.DateTimeField(auto_now=True, blank=True, null=True, verbose_name="Update Time")
  105. is_delete = models.BooleanField(default=False, verbose_name='Delete Label')
  106. out_type = models.IntegerField(choices=OUT_TYPE, default=4, verbose_name="出库类型")
  107. class Meta:
  108. db_table = 'outbounddemand'
  109. verbose_name = '出库需求'
  110. verbose_name_plural = "出库需求"
  111. ordering = ['-id']
  112. class MaterialStatistics(models.Model):
  113. goods_code = models.CharField(max_length=255, verbose_name="商品编码", unique=True)
  114. goods_desc = models.CharField(max_length=255, verbose_name="商品描述")
  115. goods_std = models.CharField(default='待填写', max_length=255, verbose_name="商品标准", blank=True, null=True)
  116. goods_unit = models.CharField(default='待填写', max_length=255, verbose_name="商品单位")
  117. total_quantity = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="在库数量")
  118. total_demanded_quantity = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="需求数量")
  119. def __str__(self):
  120. return f"{self.goods_code} - {self.goods_desc}"
  121. @property
  122. def bound_batches(self):
  123. return BoundBatchModel.objects.filter(goods_code=self.goods_code).exclude(goods_in_location_qty=0).order_by('bound_batch_order')
  124. class Meta:
  125. db_table = 'materialstatistics'
  126. verbose_name = '物料统计'
  127. verbose_name_plural = "物料统计"
  128. ordering = ['goods_code']
  129. @receiver(pre_save, sender=BoundBatchModel)
  130. def store_original_goods_code(sender, instance, **kwargs):
  131. """保存前记录原始物料编码(仅更新时生效)"""
  132. if not instance.pk: # 新建对象无原始值
  133. return
  134. try:
  135. original = BoundBatchModel.objects.get(pk=instance.pk)
  136. # 将原始值存储在实例属性中供post_save使用
  137. instance._original_goods_code = original.goods_code
  138. except BoundBatchModel.DoesNotExist:
  139. pass
  140. @receiver([post_save, post_delete], sender=BoundBatchModel)
  141. def update_material_statistics(sender, instance, **kwargs):
  142. # 处理删除/创建/更新操作的核心逻辑
  143. def refresh_stats(target_goods_code, force_desc=False):
  144. """刷新指定物料的统计数据"""
  145. stats, created = MaterialStatistics.objects.get_or_create(
  146. goods_code=target_goods_code,
  147. defaults={
  148. 'goods_desc': instance.goods_desc,
  149. 'goods_std': instance.goods_std or '待填写',
  150. 'goods_unit': instance.goods_unit or '待填写',
  151. }
  152. )
  153. # 仅强制更新描述(用于当前物料)或新建时初始化
  154. if force_desc or created:
  155. stats.goods_desc = instance.goods_desc
  156. if instance.goods_std and instance.goods_std != '待填写':
  157. stats.goods_std = instance.goods_std
  158. if instance.goods_unit and instance.goods_unit != '待填写':
  159. stats.goods_unit = instance.goods_unit
  160. # 重新聚合统计数据(始终执行)
  161. agg = BoundBatchModel.objects.filter(goods_code=target_goods_code).aggregate(
  162. total=Sum('goods_in_location_qty'),
  163. total_out=Sum('goods_out_qty')
  164. )
  165. stats.total_quantity = agg['total'] or 0
  166. # 如果有出库统计字段需更新,可在此添加
  167. stats.save()
  168. # 情况1:删除操作直接更新当前物料
  169. if kwargs.get('signal') == post_delete:
  170. refresh_stats(instance.goods_code)
  171. return
  172. # 情况2:创建操作直接更新新物料
  173. if kwargs.get('created'):
  174. refresh_stats(instance.goods_code, force_desc=True)
  175. return
  176. # 情况3:更新操作(检测物料编码变更)
  177. original_code = getattr(instance, '_original_goods_code', None)
  178. current_code = instance.goods_code
  179. # 物料编码未变化时仅更新当前物料
  180. if original_code == current_code:
  181. refresh_stats(current_code, force_desc=True)
  182. # 物料编码变化时更新新旧两个物料
  183. else:
  184. refresh_stats(original_code) # 更新旧物料(仅刷新数量)
  185. refresh_stats(current_code, force_desc=True) # 更新新物料(全量刷新)
  186. class OutBatchModel(models.Model):
  187. CONTAINER_STATUS = (
  188. (0, '申请'),
  189. (1, '在途'),
  190. (2, '已出库')
  191. )
  192. OUT_TYPE = (
  193. (0, '发货出库'),
  194. (4, '其他出库'),
  195. )
  196. bound_list = models.ForeignKey(BoundListModel, on_delete=models.CASCADE, verbose_name="Bound List", related_name='out_batch_list',blank=True, null=True)
  197. out_number = models.CharField(max_length=255, verbose_name="出库批次号",blank=False, null=False)
  198. batch_number = models.ForeignKey(BoundBatchModel, on_delete=models.CASCADE, verbose_name="批次号", related_name='out_batch')
  199. out_date = models.DateTimeField(verbose_name="出库日期")
  200. out_type = models.IntegerField(choices=OUT_TYPE, default=4, verbose_name="出库类型")
  201. out_note = models.CharField(default='', max_length=255, verbose_name="出库备注")
  202. warehouse_code = models.CharField(max_length=255, verbose_name="仓库编码")
  203. warehouse_name = models.CharField(max_length=255, verbose_name="仓库名称")
  204. goods_code = models.CharField(max_length=255, verbose_name="商品编码")
  205. goods_desc = models.CharField(max_length=255, verbose_name="商品描述")
  206. goods_std = models.CharField(default='待填写', max_length=255, verbose_name="商品标准",blank=True, null=True)
  207. goods_unit = models.CharField(default='待填写', max_length=255, verbose_name="商品单位")
  208. goods_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="商品数量")
  209. goods_out_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="出库数量")
  210. status = models.IntegerField(choices=CONTAINER_STATUS, default=0, verbose_name='批次状态')
  211. container_number = models.IntegerField( default=0, verbose_name="托盘数目")
  212. goods_weight = models.FloatField(default=0, verbose_name="商品单重")
  213. goods_total_weight = models.FloatField(default=0, verbose_name="商品总重")
  214. creater = models.CharField(default='uesr', max_length=255, verbose_name="Who Created")
  215. openid = models.CharField(max_length=255, verbose_name="Openid")
  216. is_delete = models.BooleanField(default=False, verbose_name='Delete Label')
  217. create_time = models.DateTimeField(auto_now_add=True, verbose_name="Create Time")
  218. update_time = models.DateTimeField(auto_now=True, blank=True, null=True, verbose_name="Update Time")
  219. relate_material = models.ForeignKey(OutMaterialDetail, on_delete=models.CASCADE, verbose_name="关联物料", related_name='out_batch',null=True, blank=True)
  220. class Meta:
  221. db_table = 'outbatch'
  222. verbose_name = 'Out Batch'
  223. verbose_name_plural = "Out Batch"
  224. ordering = ['-id']
  225. @receiver([post_save, post_delete], sender=OutBatchModel)
  226. def update_material_demanded(sender, instance, **kwargs):
  227. goods_code = instance.goods_code
  228. stats, created = MaterialStatistics.objects.get_or_create(
  229. goods_code=goods_code,
  230. defaults={
  231. 'goods_desc': instance.goods_desc,
  232. 'goods_std': instance.goods_std or '待填写',
  233. 'goods_unit': instance.goods_unit or '待填写',
  234. }
  235. )
  236. # 更新物料信息为最新批次的信息(可选)
  237. stats.goods_desc = instance.goods_desc
  238. if instance.goods_std and instance.goods_std != '待填写':
  239. stats.goods_std = instance.goods_std
  240. if instance.goods_unit and instance.goods_unit != '待填写':
  241. stats.goods_unit = instance.goods_unit
  242. stats.save()
  243. # 计算总数量
  244. total = OutBatchModel.objects.filter(goods_code=goods_code,is_delete=False).aggregate(
  245. total=Sum('goods_out_qty')
  246. )['total'] or 0
  247. stats.total_demanded_quantity = total
  248. stats.save()
  249. batch_number = instance.batch_number
  250. batch_out_qty = OutBatchModel.objects.filter(batch_number=batch_number,goods_code=goods_code,is_delete=False).aggregate(
  251. total=Sum('goods_out_qty')
  252. )['total'] or 0
  253. batch_number.goods_reserve_qty = batch_out_qty
  254. batch_number.save()
  255. # 利用创建好的批次来与申请单相对应
  256. class OutBoundDetailModel(models.Model):
  257. CONTAINER_STATUS = (
  258. (0, '出库申请'),
  259. (1, '出库中'),
  260. (2, '已出库'),
  261. (3, '入库中'),
  262. (4, '已入库')
  263. )
  264. bound_list = models.ForeignKey(BoundListModel, on_delete=models.CASCADE, verbose_name="Bound List", related_name='out_bound_detail')
  265. bound_batch = models.ForeignKey(OutBatchModel, on_delete=models.PROTECT, verbose_name="Bound Batch", related_name='out_bound_batch_detail')
  266. bound_batch_number = models.ForeignKey(BoundBatchModel, on_delete=models.CASCADE, verbose_name="Bound Batch Number", related_name='out_bound_batch_number_detail')
  267. status = models.IntegerField(choices=CONTAINER_STATUS, default=0, verbose_name='批次状态')
  268. detail_code = models.CharField(max_length=255, verbose_name="明细编号")
  269. creater = models.CharField(default='uesr', max_length=255, verbose_name="Who Created")
  270. openid = models.CharField(max_length=255, verbose_name="Openid")
  271. is_delete = models.BooleanField(default=False, verbose_name='Delete Label')
  272. create_time = models.DateTimeField(auto_now_add=True, verbose_name="Create Time")
  273. update_time = models.DateTimeField(auto_now=True, blank=True, null=True, verbose_name="Update Time")
  274. class Meta:
  275. db_table = 'Outbounddetail'
  276. verbose_name = 'OutBound Detail'
  277. verbose_name_plural = "OutBound Detail"
  278. ordering = ['-id']
  279. class BatchLogModel(models.Model):
  280. BATCH_LOG_TYPE = (
  281. (0, '入库'),
  282. (1, '出库'),
  283. (2, '移库'),
  284. (9, '其他'),
  285. )
  286. batch_id =models.ForeignKey(BoundBatchModel, on_delete=models.CASCADE, verbose_name="批次ID", related_name='batch_log')
  287. log_type = models.IntegerField(choices=BATCH_LOG_TYPE, default=9, verbose_name="日志类型")
  288. log_date = models.DateTimeField(verbose_name="日志日期")
  289. goods_code = models.CharField(max_length=255, verbose_name="商品编码")
  290. goods_desc = models.CharField(max_length=255, verbose_name="商品描述")
  291. goods_qty = models.DecimalField(max_digits=10, decimal_places=3, default=Decimal('0'), verbose_name="商品数量")
  292. log_content = models.CharField(max_length=255, verbose_name="日志内容")
  293. creater = models.CharField(default='uesr', max_length=255, verbose_name="Who Created")
  294. openid = models.CharField(max_length=255, verbose_name="Openid")
  295. is_delete = models.BooleanField(default=False, verbose_name='Delete Label')
  296. create_time = models.DateTimeField(auto_now_add=True, verbose_name="Create Time")
  297. update_time = models.DateTimeField(auto_now=True, blank=True, null=True, verbose_name="Update Time")
  298. class Meta:
  299. db_table = 'batchlog'
  300. verbose_name = 'Batch Log'
  301. verbose_name_plural = "Batch Log"
  302. # 利用创建好的批次来与申请单相对应
  303. class BoundDetailModel(models.Model):
  304. CONTAINER_STATUS = (
  305. (0, '入库申请'),
  306. (1, '入库中'),
  307. (2, '在库'),
  308. (3, '出库中'),
  309. (4, '已出库')
  310. )
  311. bound_list = models.ForeignKey(BoundListModel, on_delete=models.CASCADE, verbose_name="Bound List", related_name='bound_detail')
  312. bound_batch = models.ForeignKey(BoundBatchModel, on_delete=models.PROTECT, verbose_name="Bound Batch", related_name='bound_batch_detail')
  313. status = models.IntegerField(choices=CONTAINER_STATUS, default=0, verbose_name='批次状态')
  314. detail_code = models.CharField(max_length=255, verbose_name="明细编号",unique=True)
  315. creater = models.CharField(default='uesr', max_length=255, verbose_name="Who Created")
  316. openid = models.CharField(max_length=255, verbose_name="Openid")
  317. is_delete = models.BooleanField(default=False, verbose_name='Delete Label')
  318. create_time = models.DateTimeField(auto_now_add=True, verbose_name="Create Time")
  319. update_time = models.DateTimeField(auto_now=True, blank=True, null=True, verbose_name="Update Time")
  320. class Meta:
  321. db_table = 'bounddetail'
  322. verbose_name = 'Bound Detail'
  323. verbose_name_plural = "Bound Detail"
  324. ordering = ['-id']