GoodERP
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

502 lines
19KB

  1. from .utils import create_name, safe_division
  2. from odoo.exceptions import UserError
  3. from odoo import models
  4. from odoo import fields
  5. from odoo import api
  6. from odoo.tools import float_compare, float_is_zero
  7. class WhInventory(models.Model):
  8. _name = 'wh.inventory'
  9. _description = '盘点单'
  10. _inherit = ['mail.thread']
  11. _order = 'date DESC, id DESC'
  12. INVENTORY_STATE = [
  13. ('draft', '草稿'),
  14. ('query', '查询中'),
  15. ('confirmed', '待确认盘盈盘亏'),
  16. ('done', '完成'),
  17. ]
  18. @api.model
  19. def _get_default_warehouse_impl(self):
  20. if self.env.context.get('warehouse_type', 'stock'):
  21. return self.env['warehouse'].get_warehouse_by_type(
  22. self.env.context.get('warehouse_type', 'stock'))
  23. @api.model
  24. def _get_default_warehouse(self):
  25. '''获取盘点仓库'''
  26. return self._get_default_warehouse_impl()
  27. date = fields.Date('日期', default=fields.Date.context_today,
  28. help='盘点单创建日期,默认为当前天')
  29. name = fields.Char('名称', copy=False, default='/',
  30. help='单据编号,创建时会自动生成')
  31. warehouse_id = fields.Many2one('warehouse', '仓库', required=True, default=_get_default_warehouse,
  32. help='盘点单盘点的仓库')
  33. goods = fields.Many2many('goods', string='商品',
  34. help='盘点单盘点的商品')
  35. out_id = fields.Many2one('wh.out', '盘亏单据', copy=False,
  36. help='盘亏生成的其他出库单单据')
  37. in_id = fields.Many2one('wh.in', '盘盈单据', copy=False,
  38. help='盘盈生成的其他入库单单据')
  39. state = fields.Selection(
  40. INVENTORY_STATE, '状态', copy=False, default='draft',
  41. index=True,
  42. tracking=True,
  43. help='盘点单状态,新建时状态为草稿;'
  44. '点击查询后为确认后状态为查询中;'
  45. '有盘亏盘盈时生成的其他出入库单没有确认时状态为待确认盘盈盘亏;'
  46. '盘亏盘盈生成的其他出入库单确认后状态为完成')
  47. line_ids = fields.One2many(
  48. 'wh.inventory.line', 'inventory_id', '明细', copy=False,
  49. help='盘点单的明细行')
  50. note = fields.Text('备注',
  51. help='可以为该单据添加一些需要的标识信息')
  52. company_id = fields.Many2one(
  53. 'res.company',
  54. string='公司',
  55. change_default=True,
  56. default=lambda self: self.env.company)
  57. def requery_inventory(self):
  58. self.delete_confirmed_wh()
  59. self.state = 'query'
  60. @api.model
  61. @create_name
  62. def create(self, vals):
  63. return super(WhInventory, self).create(vals)
  64. def unlink(self):
  65. for inventory in self:
  66. inventory.delete_confirmed_wh()
  67. return super(WhInventory, self).unlink()
  68. def delete_confirmed_wh(self):
  69. for inventory in self:
  70. if inventory.state == 'confirmed':
  71. if (inventory.out_id and inventory.out_id.state == 'done') \
  72. or (inventory.in_id and inventory.in_id.state == 'done'):
  73. raise UserError('请先撤销掉相关的盘盈盘亏单据')
  74. else:
  75. inventory.out_id.unlink()
  76. inventory.in_id.unlink()
  77. return True
  78. def check_done(self):
  79. for inventory in self:
  80. if inventory.state == 'confirmed' and \
  81. (not inventory.out_id or inventory.out_id.state == 'done') and \
  82. (not inventory.in_id or inventory.in_id.state == 'done'):
  83. self.state = 'done'
  84. return True
  85. if inventory.state == 'done' and \
  86. (not inventory.out_id or inventory.out_id.state != 'done') and \
  87. (not inventory.in_id or inventory.in_id.state != 'done'):
  88. self.state = 'confirmed'
  89. return True
  90. return False
  91. def open_out(self):
  92. for inventory in self:
  93. return {
  94. 'type': 'ir.actions.act_window',
  95. 'res_model': 'wh.out',
  96. 'view_mode': 'form',
  97. 'res_id': inventory.out_id.id,
  98. }
  99. def open_in(self):
  100. for inventory in self:
  101. return {
  102. 'type': 'ir.actions.act_window',
  103. 'res_model': 'wh.in',
  104. 'view_mode': 'form',
  105. 'res_id': inventory.in_id.id,
  106. }
  107. def delete_line(self):
  108. self.line_ids.unlink()
  109. def create_losses_out(self, inventory, out_line):
  110. inventory_warehouse = self.env.ref('warehouse.warehouse_inventory')
  111. out_vals = {
  112. 'warehouse_dest_id': inventory_warehouse.id,
  113. 'warehouse_id': inventory.warehouse_id.id,
  114. 'type': 'inventory',
  115. 'line_out_ids': [],
  116. }
  117. for line in out_line:
  118. out_vals['line_out_ids'].append(
  119. [0, False, line.get_move_line(wh_type='out')])
  120. out_id = self.env['wh.out'].create(out_vals)
  121. inventory.out_id = out_id
  122. def create_overage_in(self, inventory, in_line):
  123. inventory_warehouse = self.env.ref('warehouse.warehouse_inventory')
  124. in_vals = {
  125. 'warehouse_dest_id': inventory.warehouse_id.id,
  126. 'warehouse_id': inventory_warehouse.id,
  127. 'type': 'inventory',
  128. 'line_in_ids': [],
  129. }
  130. for line in in_line:
  131. in_vals['line_in_ids'].append(
  132. [0, False, line.get_move_line(wh_type='in')])
  133. in_id = self.env['wh.in'].create(in_vals)
  134. inventory.in_id = in_id
  135. def generate_inventory(self):
  136. for inventory in self:
  137. if self.state in ['done', 'confirmed']:
  138. raise UserError('请不要重复点击生成盘点单据按钮')
  139. out_line, in_line = [], []
  140. for line in inventory.line_ids:
  141. if line.difference_qty < 0:
  142. out_line.append(line)
  143. elif line.difference_qty > 0:
  144. in_line.append(line)
  145. if out_line:
  146. self.create_losses_out(inventory, out_line)
  147. if in_line:
  148. self.create_overage_in(inventory, in_line)
  149. if len(out_line) + len(in_line) == 0:
  150. inventory.state = 'done'
  151. if out_line or in_line:
  152. inventory.state = 'confirmed'
  153. return True
  154. def get_line_detail(self):
  155. for inventory in self:
  156. sql_text = '''
  157. SELECT wh.id as warehouse_id,
  158. goods.id as goods_id,
  159. line.attribute_id as attribute_id,
  160. line.lot as lot,
  161. uom.id as uom_id,
  162. uos.id as uos_id,
  163. sum(line.qty_remaining) as qty,
  164. sum(line.uos_qty_remaining) as uos_qty
  165. FROM wh_move_line line
  166. LEFT JOIN goods goods ON line.goods_id = goods.id
  167. LEFT JOIN uom uom ON goods.uom_id = uom.id
  168. LEFT JOIN uom uos ON goods.uos_id = uos.id
  169. LEFT JOIN warehouse wh ON line.warehouse_dest_id = wh.id
  170. WHERE line.qty_remaining != 0
  171. AND wh.type = 'stock'
  172. AND line.state = 'done'
  173. %s
  174. GROUP BY
  175. wh.id, line.lot, line.attribute_id, goods.id, uom.id, uos.id
  176. ORDER BY
  177. goods.id, line.lot
  178. '''
  179. extra_text = ' AND wh.id = %s' % inventory.warehouse_id.id
  180. if inventory.goods:
  181. goods_ids = inventory.goods.ids
  182. goods_ids.append(0)
  183. extra_text += " AND goods.id IN {ids}".format(
  184. ids=tuple(set(goods_ids)))
  185. inventory.env.cr.execute(sql_text % extra_text)
  186. res = inventory.env.cr.dictfetchall()
  187. for line in res:
  188. # 盘点单查询的盘点数量不应该包含移库在途的 #1358
  189. for int_line in self.env['wh.move.line'].search(
  190. [('goods_id', '=', line['goods_id']),
  191. ('attribute_id', '=', line['attribute_id']),
  192. ('lot_id', '=', line['lot']),
  193. ('warehouse_id', '=', line['warehouse_id']),
  194. ('type', '=', 'internal'),
  195. ('state', '=', 'draft')]):
  196. line['qty'] -= int_line.goods_qty
  197. line['uos_qty'] -= int_line.goods_uos_qty
  198. if not line['qty']:
  199. res.remove(line)
  200. return res
  201. def query_inventory(self):
  202. line_obj = self.env['wh.inventory.line']
  203. for inventory in self:
  204. inventory.delete_line()
  205. line_ids = inventory.get_line_detail()
  206. for line in line_ids:
  207. line_obj.create_wh_inventory_line_by_data(inventory.id, line)
  208. if line_ids:
  209. inventory.state = 'query'
  210. return True
  211. class WhInventoryLine(models.Model):
  212. _name = 'wh.inventory.line'
  213. _description = '盘点单明细'
  214. LOT_TYPE = [
  215. ('out', '出库'),
  216. ('in', '入库'),
  217. ('nothing', '不做处理'),
  218. ]
  219. @api.depends('inventory_qty', 'real_qty', 'inventory_uos_qty', 'real_uos_qty')
  220. def _get_difference_qty(self):
  221. for line in self:
  222. line.difference_qty = line.inventory_qty - line.real_qty
  223. line.difference_uos_qty = line.inventory_uos_qty - line.real_uos_qty
  224. if float_is_zero(line.difference_qty, 2) and not float_is_zero(line.difference_uos_qty, 2):
  225. line.difference_qty = line.difference_uos_qty * line.goods_id.conversion
  226. if not float_is_zero(line.difference_qty, 2) and line.difference_uos_qty == 0:
  227. line.difference_uos_qty = line.difference_qty / line.goods_id.conversion
  228. @api.depends('inventory_uos_qty', 'real_uos_qty')
  229. def _get_difference_uos_qty(self):
  230. for line in self:
  231. line.difference_uos_qty = line.inventory_uos_qty - line.real_uos_qty
  232. inventory_id = fields.Many2one('wh.inventory', '盘点', ondelete='cascade',
  233. help='盘点单行对应的盘点单')
  234. warehouse_id = fields.Many2one('warehouse', '仓库', required=True, ondelete='restrict',
  235. help='盘点单行对应的仓库')
  236. goods_id = fields.Many2one('goods', '商品', required=True, ondelete='restrict',
  237. help='盘点单行对应的商品')
  238. attribute_id = fields.Many2one('attribute', '属性', ondelete='restrict',
  239. help='盘点单行对应的商品的属性')
  240. using_batch = fields.Boolean(
  241. related='goods_id.using_batch', string='批号管理',
  242. help='盘点单行对应的商品是否使用批号管理,是True否则False')
  243. force_batch_one = fields.Boolean(
  244. related='goods_id.force_batch_one', string='每批号数量为1',
  245. help='盘点单行对应的商品是否使用每批号数量为1,是True否则False')
  246. lot = fields.Char('批号',
  247. help='盘点单行对应的商品批号')
  248. new_lot = fields.Char('盘盈批号',
  249. help='盘点单行对应的商品盘盈批号')
  250. new_lot_id = fields.Many2one('wh.move.line', '盘亏批号',
  251. ondelete='restrict',
  252. help='盘点单行对应的商品盘亏批号')
  253. lot_type = fields.Selection(LOT_TYPE, '批号类型', default='nothing',
  254. help='批号类型: 出库、入库、不做处理')
  255. uom_id = fields.Many2one('uom', '单位', ondelete='restrict',
  256. help='盘点单行对应的商品的计量单位')
  257. uos_id = fields.Many2one('uom', '辅助单位', ondelete='restrict',
  258. help='盘点单行对应的商品的辅助单位')
  259. real_qty = fields.Float(
  260. '账面数量', digits='Quantity',
  261. help='盘点单行对应的商品的账面数量')
  262. real_uos_qty = fields.Float(
  263. '账面辅助数量', digits='Quantity',
  264. help='盘点单行对应的商品的账面辅助数量')
  265. inventory_qty = fields.Float(
  266. '实际数量', digits='Quantity',
  267. required=True,
  268. help='盘点单行对应的商品的实际数量')
  269. inventory_uos_qty = fields.Float(
  270. '实际辅助数量', digits='Quantity',
  271. required=True,
  272. help='盘点单行对应的商品的实际辅助数量')
  273. difference_qty = fields.Float(
  274. '差异数量', digits='Quantity',
  275. compute='_get_difference_qty',
  276. help='盘点单行对应的商品的差异数量')
  277. difference_uos_qty = fields.Float(
  278. '差异辅助数量', digits='Quantity',
  279. compute='_get_difference_uos_qty',
  280. help='盘点单行对应的商品的差异辅助数量')
  281. company_id = fields.Many2one(
  282. 'res.company',
  283. string='公司',
  284. change_default=True,
  285. default=lambda self: self.env.company)
  286. def check_difference_identical(self):
  287. if self.difference_qty * self.difference_uos_qty < 0:
  288. self.inventory_qty = self.real_qty
  289. self.inventory_uos_qty = self.real_uos_qty
  290. return {'warning': {
  291. 'title': '错误',
  292. 'message': '盘盈盘亏数量应该与辅助单位的盘盈盘亏数量盈亏方向一致',
  293. }}
  294. def create_wh_inventory_line_by_data(self, inventory_id, line_data):
  295. self.create({
  296. 'inventory_id': inventory_id,
  297. 'warehouse_id': line_data.get('warehouse_id'),
  298. 'goods_id': line_data.get('goods_id'),
  299. 'attribute_id': line_data.get('attribute_id'),
  300. 'lot': line_data.get('lot'),
  301. 'uom_id': line_data.get('uom_id'),
  302. 'uos_id': line_data.get('uos_id'),
  303. 'real_qty': line_data.get('qty'),
  304. 'real_uos_qty': line_data.get('uos_qty'),
  305. 'inventory_qty': line_data.get('qty'),
  306. 'inventory_uos_qty': line_data.get('uos_qty'),
  307. })
  308. def line_role_back(self):
  309. self.inventory_qty = self.real_qty
  310. self.inventory_uos_qty = self.real_uos_qty
  311. self.difference_qty = 0
  312. self.difference_uos_qty = 0
  313. self.new_lot = ''
  314. self.new_lot_id = False
  315. self.lot_type = 'nothing'
  316. @api.onchange('inventory_qty')
  317. def onchange_qty(self):
  318. self.ensure_one()
  319. if self.goods_id and self.goods_id.using_batch:
  320. if self.goods_id.force_batch_one and self.difference_qty:
  321. flag = self.difference_qty > 0 and 1 or -1
  322. if abs(self.difference_qty) != 1:
  323. self.line_role_back()
  324. return {'warning': {
  325. 'title': '警告',
  326. 'message': '商品上设置了序号为1,此时一次只能盘亏或盘盈一个商品数量',
  327. }}
  328. if self.difference_qty > 0:
  329. self.lot_type = 'in'
  330. self.new_lot = self.new_lot or self.lot
  331. self.new_lot_id = False
  332. elif self.difference_qty < 0:
  333. self.lot_type = 'out'
  334. self.new_lot = ''
  335. else:
  336. self.lot_type = 'nothing'
  337. self.new_lot_id = False
  338. self.new_lot = ''
  339. return self.check_difference_identical()
  340. @api.onchange('inventory_uos_qty')
  341. def onchange_uos_qty(self):
  342. self.inventory_qty = self.goods_id.conversion_unit(
  343. self.inventory_uos_qty)
  344. def get_move_line(self, wh_type='in', context=None):
  345. inventory_warehouse = self.env['warehouse'] \
  346. .get_warehouse_by_type('inventory')
  347. for inventory in self:
  348. cost, cost_unit = inventory.goods_id. \
  349. get_suggested_cost_by_warehouse(
  350. inventory.warehouse_id, abs(inventory.difference_qty),
  351. lot_id=inventory.new_lot_id,
  352. attribute=inventory.attribute_id)
  353. if wh_type == 'out':
  354. lot_id = self.env['wh.move.line'].search([('warehouse_dest_id', '=', inventory.warehouse_id.id),
  355. ('goods_id', '=', inventory.goods_id.id),
  356. ('lot', '=', inventory.lot),
  357. ('state', '=', 'done')
  358. ], limit=1)
  359. res = {
  360. 'type': wh_type,
  361. 'lot': inventory.lot,
  362. 'lot_id': lot_id.id,
  363. 'goods_id': inventory.goods_id.id,
  364. 'attribute_id': inventory.attribute_id.id,
  365. 'uom_id': inventory.uom_id.id,
  366. 'uos_id': inventory.uos_id.id,
  367. 'cost_unit': cost_unit,
  368. 'cost': cost,
  369. }
  370. elif wh_type == 'in':
  371. res = {
  372. 'type': wh_type,
  373. 'lot': inventory.new_lot,
  374. 'lot_id': inventory.new_lot_id.id,
  375. 'goods_id': inventory.goods_id.id,
  376. 'attribute_id': inventory.attribute_id.id,
  377. 'uom_id': inventory.uom_id.id,
  378. 'uos_id': inventory.uos_id.id,
  379. 'cost_unit': cost_unit,
  380. 'cost': cost,
  381. }
  382. difference_qty, difference_uos_qty = abs(
  383. inventory.difference_qty), abs(inventory.difference_uos_qty)
  384. res.update({'goods_qty': difference_qty})
  385. return res
  386. class WhOut(models.Model):
  387. _inherit = 'wh.out'
  388. inventory_ids = fields.One2many('wh.inventory', 'out_id', '盘点单',
  389. help='其他出库单行对应的盘点单行,盘亏的情况')
  390. def approve_order(self):
  391. res = super(WhOut, self).approve_order()
  392. for order in self:
  393. order.inventory_ids.check_done()
  394. return res
  395. def cancel_approved_order(self):
  396. res = super(WhOut, self).cancel_approved_order()
  397. for order in self:
  398. order.inventory_ids.check_done()
  399. return res
  400. class WhIn(models.Model):
  401. _inherit = 'wh.in'
  402. inventory_ids = fields.One2many('wh.inventory', 'in_id', '盘点单',
  403. help='其他入库单行对应的盘点单行,盘盈的情况')
  404. def approve_order(self):
  405. res = super(WhIn, self).approve_order()
  406. for order in self:
  407. order.inventory_ids.check_done()
  408. return res
  409. def cancel_approved_order(self):
  410. res = super(WhIn, self).cancel_approved_order()
  411. for order in self:
  412. order.inventory_ids.check_done()
  413. return res
上海开阖软件有限公司 沪ICP备12045867号-1