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.

1748 lines
73KB

  1. from .utils import inherits, inherits_after, \
  2. create_name, safe_division, create_origin
  3. from itertools import islice
  4. from odoo import models, fields, api, _
  5. from odoo.exceptions import UserError
  6. class WhAssembly(models.Model):
  7. _name = 'wh.assembly'
  8. _description = '组装单'
  9. _inherit = ['mail.thread']
  10. _order = 'date DESC, id DESC'
  11. _inherits = {
  12. 'wh.move': 'move_id',
  13. }
  14. state = fields.Selection([('draft', '草稿'),
  15. ('feeding', '已发料'),
  16. ('done', '完成'),
  17. ('cancel', '已作废')],
  18. '状态', copy=False, default='draft',
  19. index=True,
  20. tracking=True,
  21. help='组装单状态标识,新建时状态为草稿;发料后状态为已发料,可以多次投料;成品入库后状态为完成。')
  22. move_id = fields.Many2one(
  23. 'wh.move', '移库单', required=True, index=True, ondelete='cascade',
  24. help='组装单对应的移库单')
  25. bom_id = fields.Many2one(
  26. 'wh.bom', '物料清单', domain=[('type', '=', 'assembly')],
  27. context={'type': 'assembly'}, ondelete='restrict',
  28. readonly=True,
  29. help='组装单对应的物料清单')
  30. fee = fields.Float(
  31. '组装费用', digits='Amount',
  32. readonly=True,
  33. help='组装单对应的组装费用,组装费用+组装行入库成本作为子件的出库成本')
  34. goods_id = fields.Many2one('goods', string='组合件商品',
  35. readonly=True)
  36. lot = fields.Char('批号')
  37. goods_qty = fields.Float('组合件数量', default=1, digits='Quantity',
  38. readonly=True,
  39. help=u"(选择使用物料清单后)当更改这个数量的时候后自动的改变相应的子件的数量")
  40. voucher_id = fields.Many2one(
  41. 'voucher', copy=False, ondelete='restrict', string='入库凭证号')
  42. out_voucher_id = fields.Many2one(
  43. 'voucher', copy=False, ondelete='restrict', string='出库凭证号')
  44. company_id = fields.Many2one(
  45. 'res.company',
  46. string='公司',
  47. change_default=True,
  48. default=lambda self: self.env.company)
  49. details = fields.Html('明细', compute='_compute_details')
  50. @api.depends('move_id.line_in_ids', 'move_id.line_out_ids')
  51. def _compute_details(self):
  52. for v in self:
  53. vl = {'col': [], 'val': []}
  54. vl['col'] = ['', '商品', '属性', '数量']
  55. for l in v.move_id.line_in_ids:
  56. vl['val'].append(['组合件', l.goods_id.name or '', l.attribute_id.name or '', l.goods_qty or ''])
  57. for l in v.move_id.line_out_ids:
  58. vl['val'].append(['子件', l.goods_id.name or '', l.attribute_id.name or '', l.goods_qty or ''])
  59. v.details = v.company_id._get_html_table(vl)
  60. def apportion_cost(self, cost):
  61. for assembly in self:
  62. if not assembly.line_in_ids:
  63. continue
  64. collects = []
  65. ignore_move = [line.id for line in assembly.line_in_ids]
  66. for parent in assembly.line_in_ids:
  67. collects.append((
  68. parent, parent.goods_id.get_suggested_cost_by_warehouse(
  69. parent.warehouse_dest_id, parent.goods_qty,
  70. lot_id=parent.lot_id,
  71. attribute=parent.attribute_id,
  72. ignore_move=ignore_move)[0]))
  73. amount_total, collect_parent_cost = sum(
  74. collect[1] for collect in collects), 0
  75. for parent, amount in islice(collects, 0, len(collects) - 1):
  76. parent_cost = safe_division(amount, amount_total) * cost
  77. collect_parent_cost += parent_cost
  78. parent.write({
  79. 'cost_unit': safe_division(
  80. parent_cost, parent.goods_qty),
  81. 'cost': parent_cost,
  82. })
  83. # 最后一行数据使用总金额减去已经消耗的金额来计算
  84. last_parent_cost = cost - collect_parent_cost
  85. collects[-1][0].write({
  86. 'cost_unit': safe_division(
  87. last_parent_cost, collects[-1][0].goods_qty),
  88. 'cost': last_parent_cost,
  89. })
  90. return True
  91. def update_parent_cost(self):
  92. for assembly in self:
  93. cost = sum(child.cost for child in assembly.line_out_ids) + \
  94. assembly.fee
  95. assembly.apportion_cost(cost)
  96. return True
  97. @api.onchange('goods_id')
  98. def onchange_goods_id(self):
  99. if self.goods_id and not self.bom_id:
  100. self.line_in_ids = [(0,0,{'goods_id': self.goods_id.id, 'goods_uos_qty': 1, 'goods_qty': 1,
  101. 'uom_id': self.goods_id.uom_id.id, 'uos_id': self.goods_id.uos_id.id,
  102. 'type': 'in'})]
  103. @api.onchange('goods_qty')
  104. def onchange_goods_qty(self):
  105. """
  106. 改变商品数量时(wh_assembly 中的goods_qty) 根据物料清单的 数量的比例及成本价的计算
  107. 算出新的组合件或者子件的 数量 (line.goods_qty / parent_line_goods_qty * self.goods_qty
  108. line.goods_qty 子件商品数量
  109. parent_line_goods_qty 物料清单组合件商品数量
  110. self.goods_qty 所要的组合件的商品数量
  111. line.goods_qty /parent_line_goods_qty 得出子件和组合件的比例
  112. line.goods_qty / parent_line_goods_qty * self.goods_qty 得出子件实际的数量的数量
  113. )
  114. :return:line_out_ids ,line_in_ids
  115. """
  116. line_out_ids, line_in_ids = [], []
  117. warehouse_id = self.env['warehouse'].search(
  118. [('type', '=', 'stock')], limit=1)
  119. if self.bom_id:
  120. line_in_ids = [(0,0,{'goods_id': line.goods_id.id,
  121. 'attribute_id': line.attribute_id.id,
  122. 'warehouse_id': self.env['warehouse'].get_warehouse_by_type(
  123. 'production').id,
  124. 'warehouse_dest_id': warehouse_id.id,
  125. 'uom_id': line.goods_id.uom_id.id,
  126. 'goods_qty': self.goods_qty,
  127. 'goods_uos_qty': self.goods_qty / line.goods_id.conversion,
  128. 'uos_id': line.goods_id.uos_id.id,
  129. 'type': 'in',
  130. }) for line in self.bom_id.line_parent_ids]
  131. parent_line_goods_qty = self.bom_id.line_parent_ids[0].goods_qty
  132. for line in self.bom_id.line_child_ids:
  133. cost, cost_unit = line.goods_id. \
  134. get_suggested_cost_by_warehouse(
  135. warehouse_id[0], line.goods_qty / parent_line_goods_qty * self.goods_qty)
  136. local_goods_qty = line.goods_qty / parent_line_goods_qty * self.goods_qty
  137. line_out_ids.append((0,0,{
  138. 'goods_id': line.goods_id.id,
  139. 'attribute_id': line.attribute_id.id,
  140. 'designator': line.designator,
  141. 'warehouse_id': warehouse_id.id,
  142. 'warehouse_dest_id': self.env[
  143. 'warehouse'].get_warehouse_by_type('production'),
  144. 'uom_id': line.goods_id.uom_id.id,
  145. 'goods_qty': local_goods_qty,
  146. 'cost_unit': cost_unit,
  147. 'cost': cost,
  148. 'goods_uos_qty': local_goods_qty / line.goods_id.conversion,
  149. 'uos_id': line.goods_id.uos_id.id,
  150. 'type': 'out',
  151. }))
  152. line_ids = []
  153. if line_out_ids:
  154. line_ids += line_out_ids
  155. if line_in_ids:
  156. line_ids += line_in_ids
  157. self.line_ids = False
  158. self.line_ids = line_ids
  159. def check_parent_length(self):
  160. for p in self:
  161. if not len(p.line_in_ids) or not len(p.line_out_ids):
  162. raise UserError('组合件和子件的商品都需要输入')
  163. def create_voucher_line(self, data):
  164. return [self.env['voucher.line'].create(data_line) for data_line in data]
  165. def create_vourcher_line_data(self, assembly, voucher_row):
  166. """
  167. 准备入库凭证行数据
  168. 借:库存商品(商品上)
  169. 贷:生产成本-基本生产成本(核算分类上)
  170. :param assembly: 组装单
  171. :param voucher_row: 入库凭证
  172. :return:
  173. """
  174. line_out_data, line_in_data = [], []
  175. line_out_credit = 0.0
  176. for line_out in assembly.line_in_ids:
  177. if line_out.cost:
  178. line_out_credit += line_out.cost
  179. if line_out_credit: # 贷方行
  180. account_id = self.finance_category_id.account_id.id
  181. line_out_data.append({'credit': line_out_credit - assembly.fee,
  182. 'goods_id': False,
  183. 'voucher_id': voucher_row.id,
  184. 'account_id': account_id,
  185. 'name': '%s 原料 %s' % (assembly.move_id.name, assembly.move_id.note or '')
  186. })
  187. for line_in in assembly.line_in_ids: # 借方行
  188. if line_in.cost:
  189. account_id = line_in.goods_id.category_id.account_id.id
  190. line_in_data.append({'debit': line_in.cost,
  191. 'goods_id': line_in.goods_id.id,
  192. 'goods_qty': line_in.goods_qty,
  193. 'voucher_id': voucher_row.id,
  194. 'account_id': account_id,
  195. 'name': '%s 成品 %s' % (assembly.move_id.name, assembly.move_id.note or '')})
  196. return line_out_data + line_in_data
  197. def wh_assembly_create_voucher_line(self, assembly, voucher_row):
  198. """
  199. 创建入库凭证行
  200. :param assembly: 组装单
  201. :param voucher_row: 入库凭证
  202. :return:
  203. """
  204. voucher_line_data = []
  205. # 贷方行
  206. if assembly.fee:
  207. account_row = assembly.create_uid.company_id.operating_cost_account_id
  208. voucher_line_data.append({'name': '组装费用', 'account_id': account_row.id,
  209. 'credit': assembly.fee, 'voucher_id': voucher_row.id})
  210. voucher_line_data += self.create_vourcher_line_data(
  211. assembly, voucher_row)
  212. self.create_voucher_line(voucher_line_data)
  213. def pre_out_vourcher_line_data(self, assembly, voucher):
  214. """
  215. 准备出库凭证行数据
  216. 借:生产成本-基本生产成本(核算分类上)
  217. 贷:库存商品(商品上)
  218. :param assembly: 组装单
  219. :param voucher: 出库凭证
  220. :return: 出库凭证行数据
  221. """
  222. line_out_data, line_in_data = [], []
  223. line_out_debit = 0.0
  224. for line_out in assembly.line_out_ids:
  225. if line_out.cost:
  226. line_out_debit += line_out.cost
  227. if line_out_debit: # 借方行
  228. account_id = self.finance_category_id.account_id.id
  229. line_in_data.append({'debit': line_out_debit,
  230. 'goods_id': False,
  231. 'voucher_id': voucher.id,
  232. 'account_id': account_id,
  233. 'name': '%s 成品 %s' % (assembly.move_id.name, assembly.move_id.note or '')
  234. })
  235. for line_out in assembly.line_out_ids: # 贷方行
  236. if line_out.cost:
  237. account_id = line_out.goods_id.category_id.account_id.id
  238. line_out_data.append({'credit': line_out.cost,
  239. 'goods_id': line_out.goods_id.id,
  240. 'goods_qty': line_out.goods_qty,
  241. 'voucher_id': voucher.id,
  242. 'account_id': account_id,
  243. 'name': '%s 原料 %s' % (assembly.move_id.name, assembly.move_id.note or '')})
  244. return line_out_data + line_in_data
  245. def create_out_voucher_line(self, assembly, voucher):
  246. """
  247. 创建出库凭证行
  248. :param assembly: 组装单
  249. :param voucher: 出库凭证
  250. :return:
  251. """
  252. voucher_line_data = self.pre_out_vourcher_line_data(assembly, voucher)
  253. self.create_voucher_line(voucher_line_data)
  254. def wh_assembly_create_voucher(self):
  255. """
  256. 生成入库凭证并审核
  257. :return:
  258. """
  259. for assembly in self:
  260. voucher_row = self.env['voucher'].create({
  261. 'date': self.date,
  262. })
  263. self.wh_assembly_create_voucher_line(assembly, voucher_row) # 入库凭证
  264. if not voucher_row.line_ids:
  265. voucher_row.unlink()
  266. return
  267. assembly.voucher_id = voucher_row.id
  268. voucher_row.voucher_done()
  269. def create_out_voucher(self):
  270. """
  271. 生成出库凭证并审核
  272. :return:
  273. """
  274. for assembly in self:
  275. out_voucher = self.env['voucher'].create({
  276. 'date': self.date,
  277. })
  278. self.create_out_voucher_line(assembly, out_voucher) # 出库凭证
  279. if not out_voucher.line_ids:
  280. out_voucher.unlink()
  281. return
  282. old_voucher = assembly.out_voucher_id
  283. assembly.out_voucher_id = out_voucher.id
  284. out_voucher.voucher_done()
  285. if old_voucher:
  286. old_voucher.voucher_draft()
  287. old_voucher.unlink()
  288. def check_is_child_enable(self):
  289. for child_line in self.line_out_ids:
  290. for parent_line in self.line_in_ids:
  291. if child_line.goods_id.id == parent_line.goods_id.id and child_line.attribute_id.id == parent_line.attribute_id.id:
  292. raise UserError('子件中不能包含与组合件中相同的 产品+属性,%s' % parent_line.goods_id.name)
  293. def approve_feeding(self):
  294. ''' 发料 '''
  295. for order in self:
  296. if order.state == 'feeding':
  297. raise UserError('请不要重复发料')
  298. order.check_parent_length()
  299. order.check_is_child_enable()
  300. for line_out in order.line_out_ids:
  301. if line_out.state != 'done':
  302. line_out.action_done()
  303. order.create_out_voucher() # 生成出库凭证并审核
  304. order.state = 'feeding'
  305. return
  306. def cancel_feeding(self):
  307. ''' 退料 '''
  308. for order in self:
  309. if order.state == 'done':
  310. raise UserError('已入库不可退料')
  311. for line_out in order.line_out_ids:
  312. if line_out.state != 'draft':
  313. line_out.action_draft()
  314. # 删除出库凭证
  315. voucher, order.out_voucher_id = order.out_voucher_id, False
  316. if voucher.state == 'done':
  317. voucher.voucher_draft()
  318. voucher.unlink()
  319. order.state = 'draft'
  320. return
  321. def approve_order(self):
  322. ''' 成品入库 '''
  323. for order in self:
  324. if order.state == 'done':
  325. raise UserError('请不要重复执行成品入库')
  326. if order.state != 'feeding':
  327. raise UserError('请先投料')
  328. order.move_id.check_qc_result() # 检验质检报告是否上传
  329. if order.lot: # 入库批次
  330. for line in order.line_in_ids:
  331. line.lot = order.lot
  332. order.line_in_ids.action_done() # 完成成品入库
  333. wh_internal = self.env['wh.internal'].search([('ref', '=', order.move_id.name)])
  334. if wh_internal:
  335. wh_internal.approve_order()
  336. order.update_parent_cost()
  337. order.wh_assembly_create_voucher() # 生成入库凭证并审核
  338. order.approve_uid = self.env.uid
  339. order.approve_date = fields.Datetime.now(self)
  340. order.state = 'done'
  341. order.move_id.state = 'done'
  342. return
  343. def cancel_approved_order(self):
  344. for order in self:
  345. if order.state == 'feeding':
  346. raise UserError('请不要重复撤销 %s' % self._description)
  347. # 反审核入库到废品仓的移库单
  348. wh_internal = self.env['wh.internal'].search([('ref', '=', order.move_id.name)])
  349. if wh_internal:
  350. wh_internal.cancel_approved_order()
  351. wh_internal.unlink()
  352. order.line_in_ids.action_draft()
  353. # 删除入库凭证
  354. voucher, order.voucher_id = order.voucher_id, False
  355. if voucher.state == 'done':
  356. voucher.voucher_draft()
  357. voucher.unlink()
  358. order.approve_uid = False
  359. order.approve_date = False
  360. order.state = 'feeding'
  361. order.move_id.state = 'draft'
  362. @ inherits()
  363. def unlink(self):
  364. for order in self:
  365. if order.state != 'draft':
  366. raise UserError('只能删除草稿状态的单据')
  367. return order.move_id.unlink()
  368. @api.model_create_multi
  369. @create_name
  370. @create_origin
  371. def create(self, vals_list):
  372. for vals in vals_list:
  373. vals.update({'finance_category_id': self.env.ref(
  374. 'finance.categ_ass_disass').id})
  375. new_ids = super(WhAssembly, self).create(vals_list)
  376. for assembly in new_ids:
  377. assembly.update_parent_cost()
  378. return new_ids
  379. def write(self, vals):
  380. if 'line_in_ids' in vals:
  381. vals.pop('line_in_ids')
  382. if 'line_out_ids' in vals:
  383. vals.pop('line_out_ids')
  384. res = super(WhAssembly, self).write(vals)
  385. self.update_parent_cost()
  386. return res
  387. @api.onchange('bom_id')
  388. def onchange_bom(self):
  389. line_out_ids, line_in_ids = [], []
  390. domain = {}
  391. warehouse_id = self.env['warehouse'].search(
  392. [('type', '=', 'stock')], limit=1)
  393. if self.bom_id:
  394. line_in_ids = [(0, 0, {
  395. 'type': 'in',
  396. 'goods_id': line.goods_id.id,
  397. 'warehouse_id': self.env['warehouse'].get_warehouse_by_type(
  398. 'production').id,
  399. 'warehouse_dest_id': warehouse_id.id,
  400. 'uom_id': line.goods_id.uom_id.id,
  401. 'goods_qty': line.goods_qty,
  402. 'goods_uos_qty': line.goods_qty / line.goods_id.conversion,
  403. 'uos_id': line.goods_id.uos_id.id,
  404. 'attribute_id': line.attribute_id.id,
  405. }) for line in self.bom_id.line_parent_ids]
  406. for line in self.bom_id.line_child_ids:
  407. cost, cost_unit = line.goods_id. \
  408. get_suggested_cost_by_warehouse(
  409. warehouse_id[0], line.goods_qty)
  410. line_out_ids.append((0,0,{
  411. 'type': 'out',
  412. 'goods_id': line.goods_id.id,
  413. 'warehouse_id': warehouse_id.id,
  414. 'warehouse_dest_id': self.env[
  415. 'warehouse'].get_warehouse_by_type('production').id,
  416. 'uom_id': line.goods_id.uom_id.id,
  417. 'goods_qty': line.goods_qty,
  418. 'cost_unit': cost_unit,
  419. 'cost': cost,
  420. 'goods_uos_qty': line.goods_qty / line.goods_id.conversion,
  421. 'uos_id': line.goods_id.uos_id.id,
  422. 'attribute_id': line.attribute_id.id,
  423. 'designator': line.designator,
  424. }))
  425. self.line_in_ids = False
  426. self.line_out_ids = False
  427. else:
  428. self.goods_qty = 1
  429. if len(line_in_ids) == 1:
  430. """当物料清单中只有一个组合件的时候,默认本单据只有一个组合件 设置is_many_to_many_combinations 为False
  431. 使试图只能在 many2one中选择一个商品(并且只能选择在物料清单中的商品),并且回写数量"""
  432. self.goods_qty = line_in_ids[0][2].get("goods_qty")
  433. self.goods_id = line_in_ids[0][2].get("goods_id")
  434. domain = {'goods_id': [('id', '=', self.goods_id.id)]}
  435. line_ids = []
  436. if line_out_ids:
  437. line_ids += line_out_ids
  438. if line_in_ids:
  439. line_ids += line_in_ids
  440. self.line_ids = False
  441. self.line_ids = line_ids
  442. return {'domain': domain}
  443. def update_bom(self):
  444. for assembly in self:
  445. if assembly.bom_id:
  446. return assembly.save_bom()
  447. else:
  448. return {
  449. 'type': 'ir.actions.act_window',
  450. 'res_model': 'save.bom.memory',
  451. 'view_mode': 'form',
  452. 'target': 'new',
  453. }
  454. def save_bom(self, name=''):
  455. for assembly in self:
  456. for line in assembly.line_ids:
  457. line_parent_ids = []
  458. line_child_ids = []
  459. if line.type == 'in':
  460. line_parent_ids.append((0, 0, {
  461. 'goods_id': line.goods_id.id,
  462. 'goods_qty': line.goods_qty,
  463. }))
  464. if line.type == 'out':
  465. line_child_ids.append((0, 0, {
  466. 'goods_id': line.goods_id.id,
  467. 'goods_qty': line.goods_qty,
  468. }))
  469. if assembly.bom_id:
  470. assembly.bom_id.line_parent_ids.unlink()
  471. assembly.bom_id.line_child_ids.unlink()
  472. assembly.bom_id.write({
  473. 'line_parent_ids': line_parent_ids,
  474. 'line_child_ids': line_child_ids})
  475. else:
  476. bom_id = self.env['wh.bom'].create({
  477. 'name': name,
  478. 'type': 'assembly',
  479. 'line_parent_ids': line_parent_ids,
  480. 'line_child_ids': line_child_ids,
  481. })
  482. assembly.bom_id = bom_id
  483. return True
  484. class OutSource(models.Model):
  485. _name = 'outsource'
  486. _description = '委外加工单'
  487. _inherit = ['mail.thread']
  488. _order = 'date DESC, id DESC'
  489. _inherits = {
  490. 'wh.move': 'move_id',
  491. }
  492. state = fields.Selection([('draft', '草稿'),
  493. ('feeding', '已发料'),
  494. ('done', '完成'),
  495. ('cancel', '已作废')],
  496. '状态', copy=False, default='draft',
  497. index=True,
  498. help='委外加工单状态标识,新建时状态为草稿;发料后状态为已发料,可以多次投料;成品入库后状态为完成。')
  499. move_id = fields.Many2one('wh.move', '移库单', required=True, index=True, ondelete='cascade',
  500. help='委外加工单对应的移库单')
  501. bom_id = fields.Many2one('wh.bom', '物料清单', domain=[('type', '=', 'outsource')],
  502. context={'type': 'outsource'}, ondelete='restrict',
  503. readonly=True,
  504. help='委外加工单对应的物料清单')
  505. goods_id = fields.Many2one('goods', string='组合件商品',
  506. readonly=True)
  507. lot = fields.Char('批号')
  508. goods_qty = fields.Float('组合件数量', default=1, digits='Quantity',
  509. readonly=True,
  510. help=u"(选择使用物料清单后)当更改这个数量的时候后自动的改变相应的子件的数量")
  511. voucher_id = fields.Many2one(
  512. 'voucher', copy=False, ondelete='restrict', string='入库凭证号')
  513. out_voucher_id = fields.Many2one(
  514. 'voucher', copy=False, ondelete='restrict', string='出库凭证号')
  515. outsource_partner_id = fields.Many2one('partner', string='委外供应商',
  516. readonly=True,
  517. required=True)
  518. address_id = fields.Many2one('partner.address', '地址',
  519. domain="[('partner_id', '=', outsource_partner_id)]",
  520. help='联系地址')
  521. wh_assembly_id = fields.Many2one('wh.assembly', string='关联的组装单',
  522. readonly=True)
  523. outsource_fee = fields.Float(string='委外费用(含税)',
  524. digits='Amount',
  525. readonly=True)
  526. tax_amount = fields.Float(string='税额',
  527. digits='Amount',
  528. readonly=True)
  529. invoice_id = fields.Many2one('money.invoice',
  530. copy=False,
  531. ondelete='set null',
  532. string='发票号')
  533. company_id = fields.Many2one(
  534. 'res.company',
  535. string='公司',
  536. change_default=True,
  537. default=lambda self: self.env.company)
  538. details = fields.Html('明细', compute='_compute_details')
  539. @api.depends('move_id.line_in_ids', 'move_id.line_out_ids')
  540. def _compute_details(self):
  541. for v in self:
  542. vl = {'col': [], 'val': []}
  543. vl['col'] = ['', '商品', '属性', '数量']
  544. for l in v.move_id.line_in_ids:
  545. vl['val'].append(['产出', l.goods_id.name or '', l.attribute_id.name or '', l.goods_qty or ''])
  546. for l in v.move_id.line_out_ids:
  547. vl['val'].append(['投入', l.goods_id.name or '', l.attribute_id.name or '', l.goods_qty or ''])
  548. v.details = v.company_id._get_html_table(vl)
  549. @api.onchange('goods_id')
  550. def onchange_goods_id(self):
  551. if self.goods_id and not self.bom_id:
  552. self.line_in_ids = False
  553. self.line_in_ids = [(0,0,{'goods_id': self.goods_id.id, 'goods_uos_qty': 1, 'goods_qty': 1,
  554. 'uom_id': self.goods_id.uom_id.id, 'uos_id': self.goods_id.uos_id.id,
  555. 'type': 'in'})]
  556. @api.onchange('goods_qty')
  557. def onchange_goods_qty(self):
  558. """
  559. 改变商品数量时(outsource 中的goods_qty) 根据 物料清单 中的数量的比例
  560. 计算出新的组合件或子件的数量
  561. (line.goods_qty / parent_line_goods_qty * self.goods_qty
  562. line.goods_qty 子件商品数量
  563. parent_line_goods_qty 物料清单组合件商品数量
  564. self.goods_qty 所要的组合件的商品数量
  565. line.goods_qty /parent_line_goods_qty 得出子件和组合件的比例
  566. line.goods_qty / parent_line_goods_qty * self.goods_qty 得出子件实际的数量的数量
  567. )
  568. :return:line_out_ids ,line_in_ids
  569. """
  570. line_out_ids, line_in_ids = [], []
  571. warehouse_id = self.env['warehouse'].search(
  572. [('type', '=', 'stock')], limit=1)
  573. if self.bom_id: # 存在 物料清单
  574. line_in_ids = [(0, 0, {'goods_id': line.goods_id.id,
  575. 'attribute_id': line.attribute_id.id,
  576. 'warehouse_id': self.env['warehouse'].get_warehouse_by_type('production').id,
  577. 'warehouse_dest_id': warehouse_id.id,
  578. 'uom_id': line.goods_id.uom_id.id,
  579. 'goods_qty': self.goods_qty,
  580. 'goods_uos_qty': self.goods_qty / line.goods_id.conversion,
  581. 'uos_id': line.goods_id.uos_id.id,
  582. 'type': 'in'
  583. }) for line in self.bom_id.line_parent_ids]
  584. parent_line_goods_qty = self.bom_id.line_parent_ids[0].goods_qty
  585. for line in self.bom_id.line_child_ids:
  586. cost, cost_unit = line.goods_id.get_suggested_cost_by_warehouse(
  587. warehouse_id[0], line.goods_qty / parent_line_goods_qty * self.goods_qty)
  588. local_goods_qty = line.goods_qty / parent_line_goods_qty * self.goods_qty
  589. line_out_ids.append((0,0,{
  590. 'goods_id': line.goods_id.id,
  591. 'attribute_id': line.attribute_id.id,
  592. 'designator': line.designator,
  593. 'warehouse_id': warehouse_id.id,
  594. 'warehouse_dest_id': self.env['warehouse'].get_warehouse_by_type('production'),
  595. 'uom_id': line.goods_id.uom_id.id,
  596. 'goods_qty': local_goods_qty,
  597. 'cost_unit': cost_unit,
  598. 'cost': cost,
  599. 'goods_uos_qty': local_goods_qty / line.goods_id.conversion,
  600. 'uos_id': line.goods_id.uos_id.id,
  601. 'type': 'out',
  602. }))
  603. line_ids = []
  604. if line_out_ids:
  605. line_ids += line_out_ids
  606. if line_in_ids:
  607. line_ids += line_in_ids
  608. self.line_ids = False
  609. self.line_ids = line_ids
  610. @api.onchange('bom_id')
  611. def onchange_bom(self):
  612. line_out_ids, line_in_ids = [], []
  613. domain = {}
  614. warehouse_id = self.env['warehouse'].search(
  615. [('type', '=', 'stock')], limit=1)
  616. if self.bom_id:
  617. line_in_ids = [(0,0,{
  618. 'goods_id': line.goods_id.id,
  619. 'attribute_id': line.attribute_id.id,
  620. 'warehouse_id': self.env['warehouse'].get_warehouse_by_type('production').id,
  621. 'warehouse_dest_id': warehouse_id.id,
  622. 'uom_id': line.goods_id.uom_id.id,
  623. 'goods_qty': line.goods_qty,
  624. 'goods_uos_qty': line.goods_qty / line.goods_id.conversion,
  625. 'uos_id': line.goods_id.uos_id.id,
  626. 'type': 'in',
  627. }) for line in self.bom_id.line_parent_ids]
  628. for line in self.bom_id.line_child_ids:
  629. cost, cost_unit = line.goods_id. \
  630. get_suggested_cost_by_warehouse(
  631. warehouse_id[0], line.goods_qty)
  632. line_out_ids.append((0,0,{
  633. 'goods_id': line.goods_id.id,
  634. 'attribute_id': line.attribute_id.id,
  635. 'designator': line.designator,
  636. 'warehouse_id': warehouse_id.id,
  637. 'warehouse_dest_id': self.env[
  638. 'warehouse'].get_warehouse_by_type('production').id,
  639. 'uom_id': line.goods_id.uom_id.id,
  640. 'goods_qty': line.goods_qty,
  641. 'cost_unit': cost_unit,
  642. 'cost': cost,
  643. 'goods_uos_qty': line.goods_qty / line.goods_id.conversion,
  644. 'uos_id': line.goods_id.uos_id.id,
  645. 'type': 'out',
  646. }))
  647. self.line_in_ids = False
  648. self.line_out_ids = False
  649. else:
  650. self.goods_qty = 1
  651. if len(line_in_ids) == 1:
  652. """当物料清单中只有一个组合件的时候,默认本单据只有一个组合件 设置is_many_to_many_combinations 为False
  653. 使视图只能在 many2one中选择一个商品(并且只能选择在物料清单中的商品),并且回写数量"""
  654. self.goods_qty = line_in_ids[0][-1].get("goods_qty")
  655. self.goods_id = line_in_ids[0][-1].get("goods_id")
  656. domain = {'goods_id': [('id', '=', self.goods_id.id)]}
  657. line_ids = []
  658. if line_out_ids:
  659. line_ids += line_out_ids
  660. if line_in_ids:
  661. line_ids += line_in_ids
  662. self.line_ids = False
  663. self.line_ids = line_ids
  664. return {'domain': domain}
  665. def apportion_cost(self, cost):
  666. for outsource in self:
  667. if not outsource.line_in_ids:
  668. continue
  669. collects = []
  670. ignore_move = [line.id for line in outsource.line_in_ids]
  671. for parent in outsource.line_in_ids:
  672. collects.append((parent,
  673. parent.goods_id.get_suggested_cost_by_warehouse(
  674. parent.warehouse_dest_id, parent.goods_qty,
  675. lot_id=parent.lot_id,
  676. attribute=parent.attribute_id,
  677. ignore_move=ignore_move)[0]
  678. ))
  679. amount_total, collect_parent_cost = sum(
  680. collect[1] for collect in collects), 0
  681. for parent, amount in islice(collects, 0, len(collects) - 1):
  682. parent_cost = safe_division(amount, amount_total) * cost
  683. collect_parent_cost += parent_cost
  684. parent.write({
  685. 'cost_unit': safe_division(
  686. parent_cost, parent.goods_qty),
  687. 'cost': parent_cost,
  688. })
  689. # 最后一行数据使用总金额减去已经消耗的金额来计算
  690. last_parent_cost = cost - collect_parent_cost
  691. collects[-1][0].write({
  692. 'cost_unit': safe_division(
  693. last_parent_cost, collects[-1][0].goods_qty),
  694. 'cost': last_parent_cost,
  695. })
  696. return True
  697. def update_parent_cost(self):
  698. for outsource in self:
  699. if not outsource.outsource_fee:
  700. outsource_fee = sum(
  701. p.cost for p in outsource.line_in_ids)
  702. tax_amount = sum(
  703. p.price_taxed for p in outsource.line_in_ids)
  704. if outsource_fee:
  705. outsource.outsource_fee = outsource_fee
  706. outsource.tax_amount = tax_amount
  707. cost = sum(child.cost for child in outsource.line_out_ids) + \
  708. outsource.outsource_fee - outsource.tax_amount
  709. outsource.apportion_cost(cost)
  710. return True
  711. @inherits()
  712. def unlink(self):
  713. for order in self:
  714. if order.state != 'draft':
  715. raise UserError('只删除草稿状态的单据')
  716. return order.move_id.unlink()
  717. @api.model_create_multi
  718. @create_name
  719. @create_origin
  720. def create(self, vals_list):
  721. for vals in vals_list:
  722. vals.update({'finance_category_id': self.env.ref(
  723. 'finance.categ_outsource').id})
  724. new_ids = super(OutSource, self).create(vals_list)
  725. for outsource in new_ids:
  726. outsource.update_parent_cost()
  727. return new_ids
  728. def write(self, vals):
  729. if 'line_in_ids' in vals:
  730. vals.pop('line_in_ids')
  731. if 'line_out_ids' in vals:
  732. vals.pop('line_out_ids')
  733. res = super(OutSource, self).write(vals)
  734. if vals.get('outsource_fee') or vals.get('tax_amount'):
  735. return res
  736. self.update_parent_cost()
  737. return res
  738. def check_parent_length(self):
  739. for outsource_line in self:
  740. if not len(outsource_line.line_in_ids) or not len(outsource_line.line_out_ids):
  741. raise UserError('委外加工单必须存在组合件和子件明细行。')
  742. def _create_money_invoice(self):
  743. categ = self.env.ref('money.core_category_purchase')
  744. source_id = self.env['money.invoice'].create({
  745. 'name': self.name,
  746. 'partner_id': self.outsource_partner_id.id,
  747. 'category_id': categ.id,
  748. 'date': fields.Date.context_today(self),
  749. 'amount': self.outsource_fee,
  750. 'tax_amount':self.tax_amount,
  751. 'reconciled': 0,
  752. 'to_reconcile': self.outsource_fee,
  753. 'date_due': fields.Date.context_today(self),
  754. 'note': self.note or '',
  755. })
  756. if source_id:
  757. self.invoice_id = source_id.id
  758. return source_id
  759. def create_voucher_line(self, data):
  760. return [self.env['voucher.line'].create(data_line) for data_line in data]
  761. def create_vourcher_line_data(self, outsource, voucher_row):
  762. """
  763. 准备入库凭证行数据
  764. 借:库存商品(商品上)
  765. 贷:生产成本-基本生产成本(核算分类上)
  766. :param outsource: 委外加工单
  767. :param voucher_row: 入库凭证
  768. :return:
  769. """
  770. line_out_data, line_in_data = [], []
  771. line_out_credit = 0.0
  772. for line_out in outsource.line_in_ids:
  773. if line_out.cost:
  774. line_out_credit += line_out.cost
  775. if round(line_out_credit - outsource.outsource_fee + outsource.tax_amount, 2) > 0: # 贷方行
  776. account_id = self.finance_category_id.account_id.id
  777. line_out_data.append({'credit': line_out_credit - outsource.outsource_fee + outsource.tax_amount,
  778. 'goods_id': False,
  779. 'voucher_id': voucher_row.id,
  780. 'account_id': account_id,
  781. 'name': '%s 原料 %s' % (outsource.move_id.name, outsource.move_id.note or '')
  782. })
  783. for line_in in outsource.line_in_ids: # 借方行
  784. if line_in.cost:
  785. account_id = line_in.goods_id.category_id.account_id.id
  786. line_in_data.append({'debit': line_in.cost,
  787. 'goods_id': line_in.goods_id.id,
  788. 'goods_qty': line_in.goods_qty,
  789. 'voucher_id': voucher_row.id,
  790. 'account_id': account_id,
  791. 'name': '%s 成品 %s' % (outsource.move_id.name, outsource.move_id.note or '')
  792. })
  793. return line_out_data + line_in_data
  794. def pre_out_vourcher_line_data(self, outsource, voucher):
  795. """
  796. 准备出库凭证行数据
  797. 借:委托加工物资(核算分类上)
  798. 贷:库存商品(商品上)
  799. :param outsource: 委外加工单
  800. :param voucher: 出库凭证
  801. :return: 出库凭证行数据
  802. """
  803. line_out_data, line_in_data = [], []
  804. line_out_debit = 0.0
  805. for line_out in outsource.line_out_ids:
  806. if line_out.cost:
  807. line_out_debit += line_out.cost
  808. if line_out_debit: # 借方行
  809. account_id = self.finance_category_id.account_id.id
  810. line_in_data.append({'debit': line_out_debit,
  811. 'goods_id': False,
  812. 'voucher_id': voucher.id,
  813. 'account_id': account_id,
  814. 'name': '%s 成品 %s' % (outsource.move_id.name, outsource.move_id.note or '')
  815. })
  816. for line_out in outsource.line_out_ids: # 贷方行
  817. if line_out.cost:
  818. account_id = line_out.goods_id.category_id.account_id.id
  819. line_out_data.append({'credit': line_out.cost,
  820. 'goods_id': line_out.goods_id.id,
  821. 'goods_qty': line_out.goods_qty,
  822. 'voucher_id': voucher.id,
  823. 'account_id': account_id,
  824. 'name': '%s 原料 %s' % (outsource.move_id.name, outsource.move_id.note or '')})
  825. return line_out_data + line_in_data
  826. def outsource_create_voucher_line(self, outsource, voucher_row):
  827. """
  828. 创建入库凭证行
  829. :param outsource: 委外加工单
  830. :param voucher_row: 入库凭证
  831. :return:
  832. """
  833. voucher_line_data = []
  834. if outsource.outsource_fee:
  835. # 贷方行
  836. voucher_line_data.append({'name': '委外费用',
  837. 'account_id': self.env.ref('money.core_category_purchase').account_id.id, # 采购发票类别对应的科目
  838. 'credit': outsource.outsource_fee-outsource.tax_amount, 'voucher_id': voucher_row.id})
  839. voucher_line_data += self.create_vourcher_line_data(
  840. outsource, voucher_row)
  841. self.create_voucher_line(voucher_line_data)
  842. def create_out_voucher_line(self, outsource, voucher):
  843. """
  844. 创建出库凭证行
  845. :param outsource: 委外加工单
  846. :param voucher: 出库凭证
  847. :return:
  848. """
  849. voucher_line_data = self.pre_out_vourcher_line_data(outsource, voucher)
  850. self.create_voucher_line(voucher_line_data)
  851. def outsource_create_voucher(self):
  852. """
  853. 生成入库凭证并审核
  854. :return:
  855. """
  856. for outsource in self:
  857. voucher_row = self.env['voucher'].create({
  858. 'date': outsource.date,
  859. })
  860. self.outsource_create_voucher_line(outsource, voucher_row) # 入库凭证
  861. outsource.voucher_id = voucher_row.id
  862. voucher_row.voucher_done()
  863. def create_out_voucher(self):
  864. """
  865. 生成出库凭证并审核
  866. :return:
  867. """
  868. for outsource in self:
  869. out_voucher = self.env['voucher'].create({
  870. 'date': outsource.date,
  871. })
  872. self.create_out_voucher_line(outsource, out_voucher) # 出库凭证
  873. if not out_voucher.line_ids:
  874. out_voucher.unlink()
  875. return
  876. old_voucher = outsource.out_voucher_id
  877. outsource.out_voucher_id = out_voucher.id
  878. out_voucher.voucher_done()
  879. if old_voucher:
  880. old_voucher.voucher_draft()
  881. old_voucher.unlink()
  882. def check_is_child_enable(self):
  883. for child_line in self.line_out_ids:
  884. for parent_line in self.line_in_ids:
  885. if child_line.goods_id.id == parent_line.goods_id.id and child_line.attribute_id.id == parent_line.attribute_id.id:
  886. raise UserError('子件中不能包含与组合件中相同的 产品+属性,%s' % parent_line.goods_id.name)
  887. def approve_feeding(self):
  888. ''' 发料 '''
  889. for order in self:
  890. if order.state == 'feeding':
  891. raise UserError('请不要重复发料')
  892. # order.check_parent_length()
  893. order.check_is_child_enable()
  894. for line_out in order.line_out_ids:
  895. if line_out.state != 'done':
  896. line_out.action_done()
  897. order.create_out_voucher() # 生成出库凭证并审核
  898. order.state = 'feeding'
  899. return
  900. def cancel_feeding(self):
  901. ''' 退料 '''
  902. for order in self:
  903. if order.state == 'done':
  904. raise UserError('已入库不可退料')
  905. for line_out in order.line_out_ids:
  906. if line_out.state != 'draft':
  907. line_out.action_draft()
  908. # 删除出库凭证
  909. voucher, order.out_voucher_id = order.out_voucher_id, False
  910. if voucher.state == 'done':
  911. voucher.voucher_draft()
  912. voucher.unlink()
  913. order.state = 'draft'
  914. return
  915. def approve_order(self):
  916. ''' 成品入库 '''
  917. for order in self:
  918. if order.state == 'done':
  919. raise UserError('请不要重复执行成品入库')
  920. if order.state != 'feeding':
  921. raise UserError('请先投料')
  922. order.move_id.check_qc_result() # 检验质检报告是否上传
  923. if order.lot: # 入库批次
  924. for line in order.line_in_ids:
  925. line.lot = order.lot
  926. order.line_in_ids.action_done() # 完成成品入库
  927. order.date = fields.Date.context_today(self)
  928. wh_internal = self.env['wh.internal'].search([('ref', '=', order.move_id.name)])
  929. if wh_internal:
  930. wh_internal.approve_order()
  931. # 如果委外费用存在,生成 结算单
  932. if order.outsource_fee:
  933. order._create_money_invoice()
  934. order.update_parent_cost()
  935. order.outsource_create_voucher() # 生成入库凭证并审核
  936. order.approve_uid = self.env.uid
  937. order.approve_date = fields.Datetime.now(self)
  938. order.state = 'done'
  939. order.move_id.state = 'done'
  940. return
  941. def cancel_approved_order(self):
  942. for order in self:
  943. if order.state == 'feeding':
  944. raise UserError('请不要重复撤销 %s' % self._description)
  945. # 反审核入库到废品仓的移库单
  946. wh_internal = self.env['wh.internal'].search([('ref', '=', order.move_id.name)])
  947. if wh_internal:
  948. wh_internal.cancel_approved_order()
  949. wh_internal.unlink()
  950. order.line_in_ids.action_draft()
  951. # 删除入库凭证
  952. voucher, order.voucher_id = order.voucher_id, False
  953. if voucher.state == 'done':
  954. voucher.voucher_draft()
  955. voucher.unlink()
  956. if order.invoice_id:
  957. if order.invoice_id.state == 'done':
  958. order.invoice_id.money_invoice_draft()
  959. order.invoice_id.unlink()
  960. order.approve_uid = False
  961. order.approve_date = False
  962. order.state = 'feeding'
  963. order.move_id.state = 'draft'
  964. class WhDisassembly(models.Model):
  965. _name = 'wh.disassembly'
  966. _description = '拆卸单'
  967. _inherit = ['mail.thread']
  968. _order = 'date DESC, id DESC'
  969. _inherits = {
  970. 'wh.move': 'move_id',
  971. }
  972. state = fields.Selection([('draft', '草稿'),
  973. ('feeding', '已发料'),
  974. ('done', '完成'),
  975. ('cancel', '已作废')],
  976. '状态', copy=False, default='draft',
  977. index=True,
  978. help='拆卸单状态标识,新建时状态为草稿;发料后状态为已发料,可以多次投料;成品入库后状态为完成。')
  979. move_id = fields.Many2one(
  980. 'wh.move', '移库单', required=True, index=True, ondelete='cascade',
  981. help='拆卸单对应的移库单')
  982. bom_id = fields.Many2one(
  983. 'wh.bom', '物料清单', domain=[('type', '=', 'disassembly')],
  984. context={'type': 'disassembly'}, ondelete='restrict',
  985. readonly=True,
  986. help='拆卸单对应的物料清单')
  987. fee = fields.Float(
  988. '拆卸费用', digits='Amount',
  989. readonly=True,
  990. help='拆卸单对应的拆卸费用, 拆卸费用+拆卸行出库成本作为子件的入库成本')
  991. goods_id = fields.Many2one('goods', string='组合件商品',
  992. readonly=True)
  993. lot_id = fields.Many2one('wh.move.line', '批号',
  994. help='用于拆卸的组合件批号')
  995. goods_qty = fields.Float('组合件数量', default=1, digits='Quantity',
  996. readonly=True,
  997. help=u"(选择使用物料清单后)当更改这个数量的时候后自动的改变相应的子件的数量")
  998. voucher_id = fields.Many2one(
  999. 'voucher', copy=False, ondelete='restrict', string='入库凭证号')
  1000. out_voucher_id = fields.Many2one(
  1001. 'voucher', copy=False, ondelete='restrict', string='出库凭证号')
  1002. company_id = fields.Many2one(
  1003. 'res.company',
  1004. string='公司',
  1005. change_default=True,
  1006. default=lambda self: self.env.company)
  1007. details = fields.Html('明细', compute='_compute_details')
  1008. @api.depends('move_id.line_in_ids', 'move_id.line_out_ids')
  1009. def _compute_details(self):
  1010. for v in self:
  1011. vl = {'col': [], 'val': []}
  1012. vl['col'] = ['', '商品', '属性', '数量']
  1013. for l in v.move_id.line_out_ids:
  1014. vl['val'].append(['组合件', l.goods_id.name or '', l.attribute_id.name or '', l.goods_qty or ''])
  1015. for l in v.move_id.line_in_ids:
  1016. vl['val'].append(['子件', l.goods_id.name or '', l.attribute_id.name or '', l.goods_qty or ''])
  1017. v.details = v.company_id._get_html_table(vl)
  1018. def apportion_cost(self, cost):
  1019. for assembly in self:
  1020. if not assembly.line_in_ids:
  1021. continue
  1022. collects = []
  1023. ignore_move = [line.id for line in assembly.line_in_ids]
  1024. for child in assembly.line_in_ids:
  1025. collects.append((
  1026. child, child.goods_id.get_suggested_cost_by_warehouse(
  1027. child.warehouse_dest_id, child.goods_qty,
  1028. lot_id=child.lot_id, attribute=child.attribute_id,
  1029. ignore_move=ignore_move)[0]))
  1030. amount_total, collect_child_cost = \
  1031. sum(collect[1] for collect in collects), 0
  1032. for child, amount in islice(collects, 0, len(collects) - 1):
  1033. child_cost = safe_division(amount, amount_total) * cost
  1034. collect_child_cost += child_cost
  1035. child.write({
  1036. 'cost_unit': safe_division(
  1037. child_cost, child.goods_qty),
  1038. 'cost': child_cost,
  1039. })
  1040. # 最后一行数据使用总金额减去已经消耗的金额来计算
  1041. last_child_cost = cost - collect_child_cost
  1042. collects[-1][0].write({
  1043. 'cost_unit': safe_division(
  1044. last_child_cost, collects[-1][0].goods_qty),
  1045. 'cost': last_child_cost,
  1046. })
  1047. return True
  1048. def update_child_cost(self):
  1049. for assembly in self:
  1050. cost = sum(child.cost for child in assembly.line_out_ids) + \
  1051. assembly.fee
  1052. assembly.apportion_cost(cost)
  1053. return True
  1054. def check_parent_length(self):
  1055. for whd in self:
  1056. if not len(whd.line_in_ids) or not len(whd.line_out_ids):
  1057. raise UserError('组合件和子件的商品都必须输入')
  1058. def create_voucher_line(self, data):
  1059. return [self.env['voucher.line'].create(data_line) for data_line in data]
  1060. def create_vourcher_line_data(self, disassembly, voucher_row):
  1061. """
  1062. 准备入库凭证行数据
  1063. 借:库存商品(商品上)
  1064. 贷:生产成本-基本生产成本(核算分类上)
  1065. :param disassembly: 拆卸单
  1066. :param voucher_row: 入库库凭证
  1067. :return:
  1068. """
  1069. line_out_data, line_in_data = [], []
  1070. line_in_credit = 0.0
  1071. for line_in in disassembly.line_in_ids:
  1072. if line_in.cost:
  1073. line_in_credit += line_in.cost
  1074. if line_in_credit: # 贷方行
  1075. account_id = self.finance_category_id.account_id.id
  1076. line_out_data.append({'credit': line_in_credit,
  1077. 'goods_id': False,
  1078. 'voucher_id': voucher_row.id,
  1079. 'account_id': account_id,
  1080. 'name': '%s 原料 %s' % (disassembly.move_id.name, disassembly.move_id.note or '')
  1081. })
  1082. for line_in in disassembly.line_in_ids: # 借方行
  1083. if line_in.cost:
  1084. account_id = line_in.goods_id.category_id.account_id.id
  1085. line_in_data.append({'debit': line_in.cost,
  1086. 'goods_id': line_in.goods_id.id,
  1087. 'goods_qty': line_in.goods_qty,
  1088. 'voucher_id': voucher_row.id,
  1089. 'account_id': account_id,
  1090. 'name': '%s 成品 %s' % (disassembly.move_id.name, disassembly.move_id.note or '')
  1091. })
  1092. return line_out_data + line_in_data
  1093. def pre_out_vourcher_line_data(self, disassembly, voucher):
  1094. """
  1095. 准备出库凭证行数据
  1096. 借:生产成本-基本生产成本(核算分类上)
  1097. 贷:库存商品(商品上)
  1098. :param disassembly: 拆卸单
  1099. :param voucher: 出库凭证
  1100. :return: 出库凭证行数据
  1101. """
  1102. line_out_data, line_in_data = [], []
  1103. line_out_debit = 0.0
  1104. for line_out in disassembly.line_out_ids:
  1105. line_out_debit += line_out.cost
  1106. if line_out_debit: # 借方行
  1107. account_id = self.finance_category_id.account_id.id
  1108. line_in_data.append({'debit': line_out_debit,
  1109. 'goods_id': False,
  1110. 'voucher_id': voucher.id,
  1111. 'account_id': account_id,
  1112. 'name': '%s 成品 %s' % (disassembly.move_id.name, disassembly.move_id.note or '')
  1113. })
  1114. for line_out in disassembly.line_out_ids: # 贷方行
  1115. if line_out.cost:
  1116. account_id = line_out.goods_id.category_id.account_id.id
  1117. line_out_data.append({'credit': line_out.cost + disassembly.fee,
  1118. 'goods_id': line_out.goods_id.id,
  1119. 'goods_qty': line_out.goods_qty,
  1120. 'voucher_id': voucher.id,
  1121. 'account_id': account_id,
  1122. 'name': '%s 原料 %s' % (disassembly.move_id.name, disassembly.move_id.note or '')})
  1123. return line_out_data + line_in_data
  1124. def wh_disassembly_create_voucher_line(self, disassembly, voucher_row):
  1125. """
  1126. 创建入库凭证行
  1127. :param disassembly:
  1128. :param voucher_row:
  1129. :return:
  1130. """
  1131. voucher_line_data = []
  1132. voucher_line_data += self.create_vourcher_line_data(
  1133. disassembly, voucher_row)
  1134. self.create_voucher_line(voucher_line_data)
  1135. def create_out_voucher_line(self, disassembly, voucher):
  1136. """
  1137. 创建出库凭证行
  1138. :param disassembly: 拆卸单
  1139. :param voucher: 出库凭证
  1140. :return:
  1141. """
  1142. voucher_line_data = []
  1143. # 借方行
  1144. if disassembly.fee:
  1145. account = disassembly.create_uid.company_id.operating_cost_account_id
  1146. voucher_line_data.append({'name': '拆卸费用', 'account_id': account.id,
  1147. 'debit': disassembly.fee, 'voucher_id': voucher.id})
  1148. voucher_line_data += self.pre_out_vourcher_line_data(
  1149. disassembly, voucher)
  1150. self.create_voucher_line(voucher_line_data)
  1151. def wh_disassembly_create_voucher(self):
  1152. """
  1153. 生成入库凭证并审核
  1154. :return:
  1155. """
  1156. for disassembly in self:
  1157. voucher_row = self.env['voucher'].create({
  1158. 'date': self.date,
  1159. })
  1160. self.wh_disassembly_create_voucher_line(
  1161. disassembly, voucher_row) # 入库凭证
  1162. disassembly.voucher_id = voucher_row.id
  1163. voucher_row.voucher_done()
  1164. def create_out_voucher(self):
  1165. """
  1166. 生成出库凭证并审核
  1167. :return:
  1168. """
  1169. for disassembly in self:
  1170. out_voucher = self.env['voucher'].create({
  1171. 'date': self.date,
  1172. })
  1173. self.create_out_voucher_line(disassembly, out_voucher) # 出库凭证
  1174. old_voucher = disassembly.out_voucher_id
  1175. disassembly.out_voucher_id = out_voucher.id
  1176. out_voucher.voucher_done()
  1177. if old_voucher:
  1178. old_voucher.voucher_draft()
  1179. old_voucher.unlink()
  1180. def check_is_child_enable(self):
  1181. for child_line in self.line_in_ids:
  1182. for parent_line in self.line_out_ids:
  1183. if child_line.goods_id.id == parent_line.goods_id.id and child_line.attribute_id.id == parent_line.attribute_id.id:
  1184. raise UserError('子件中不能包含与组合件中相同的 产品+属性,%s' % parent_line.goods_id.name)
  1185. def approve_feeding(self):
  1186. ''' 发料 '''
  1187. for order in self:
  1188. if order.state == 'feeding':
  1189. raise UserError('请不要重复发料')
  1190. order.check_parent_length()
  1191. order.check_is_child_enable()
  1192. for line_out in order.line_out_ids:
  1193. if line_out.state != 'done':
  1194. if order.lot_id: #出库批次
  1195. line_out.lot_id = order.lot_id
  1196. line_out.action_done()
  1197. order.create_out_voucher() # 生成出库凭证并审核
  1198. order.state = 'feeding'
  1199. return
  1200. def cancel_feeding(self):
  1201. ''' 退料 '''
  1202. for order in self:
  1203. if order.state == 'done':
  1204. raise UserError('已入库不可退料')
  1205. for line_out in order.line_out_ids:
  1206. if line_out.state != 'draft':
  1207. line_out.action_draft()
  1208. # 删除出库凭证
  1209. voucher, order.out_voucher_id = order.out_voucher_id, False
  1210. if voucher.state == 'done':
  1211. voucher.voucher_draft()
  1212. voucher.unlink()
  1213. order.state = 'draft'
  1214. return
  1215. def approve_order(self):
  1216. ''' 成品入库 '''
  1217. for order in self:
  1218. if order.state == 'done':
  1219. raise UserError('请不要重复执行成品入库')
  1220. if order.state != 'feeding':
  1221. raise UserError('请先投料')
  1222. order.move_id.check_qc_result() # 检验质检报告是否上传
  1223. order.line_in_ids.action_done() # 完成成品入库
  1224. wh_internal = self.env['wh.internal'].search([('ref', '=', order.move_id.name)])
  1225. if wh_internal:
  1226. wh_internal.approve_order()
  1227. order.update_child_cost()
  1228. order.wh_disassembly_create_voucher() # 生成入库凭证并审核
  1229. order.approve_uid = self.env.uid
  1230. order.approve_date = fields.Datetime.now(self)
  1231. order.state = 'done'
  1232. order.move_id.state = 'done'
  1233. return
  1234. def cancel_approved_order(self):
  1235. for order in self:
  1236. if order.state == 'feeding':
  1237. raise UserError('请不要重复撤销 %s' % self._description)
  1238. # 反审核入库到废品仓的移库单
  1239. wh_internal = self.env['wh.internal'].search([('ref', '=', order.move_id.name)])
  1240. if wh_internal:
  1241. wh_internal.cancel_approved_order()
  1242. wh_internal.unlink()
  1243. order.line_in_ids.action_draft()
  1244. # 删除入库凭证
  1245. voucher, order.voucher_id = order.voucher_id, False
  1246. if voucher.state == 'done':
  1247. voucher.voucher_draft()
  1248. voucher.unlink()
  1249. order.approve_uid = False
  1250. order.approve_date = False
  1251. order.state = 'feeding'
  1252. order.move_id.state = 'draft'
  1253. @inherits()
  1254. def unlink(self):
  1255. for order in self:
  1256. if order.state != 'draft':
  1257. raise UserError('只删除草稿状态的单据')
  1258. return order.move_id.unlink()
  1259. @api.model_create_multi
  1260. @create_name
  1261. @create_origin
  1262. def create(self, vals_list):
  1263. for vals in vals_list:
  1264. vals.update({'finance_category_id': self.env.ref(
  1265. 'finance.categ_ass_disass').id})
  1266. new_ids = super(WhDisassembly, self).create(vals_list)
  1267. for disassembly in new_ids:
  1268. disassembly.update_child_cost()
  1269. return new_ids
  1270. def write(self, vals):
  1271. if 'line_ids' in vals:
  1272. if 'line_in_ids' in vals:
  1273. vals.pop('line_in_ids')
  1274. if 'line_out_ids' in vals:
  1275. vals.pop('line_out_ids')
  1276. res = super(WhDisassembly, self).write(vals)
  1277. self.update_child_cost()
  1278. return res
  1279. @api.onchange('goods_id')
  1280. def onchange_goods_id(self):
  1281. if self.goods_id and not self.bom_id:
  1282. warehouse_id = self.env['warehouse'].search(
  1283. [('type', '=', 'stock')], limit=1)
  1284. self.line_out_ids = [(0,0,{'goods_id': self.goods_id.id, 'goods_uos_qty': 1, 'goods_qty': 1,
  1285. 'warehouse_id': self.env['warehouse'].get_warehouse_by_type('production').id,
  1286. 'warehouse_dest_id': warehouse_id.id,
  1287. 'uom_id': self.goods_id.uom_id.id,
  1288. 'uos_id': self.goods_id.uos_id.id,
  1289. 'type': 'out',
  1290. })]
  1291. @api.onchange('goods_qty')
  1292. def onchange_goods_qty(self):
  1293. """
  1294. 改变商品数量时(wh_assembly 中的goods_qty) 根据物料清单的 数量的比例及成本价的计算
  1295. 算出新的组合件或者子件的 数量 (line.goods_qty / parent_line_goods_qty * self.goods_qty
  1296. line.goods_qty 子件商品数量
  1297. parent_line_goods_qty 物料清单组合件商品数量
  1298. self.goods_qty 所要的组合件的商品数量
  1299. line.goods_qty /parent_line_goods_qty 得出子件和组合件的比例
  1300. line.goods_qty / parent_line_goods_qty * self.goods_qty 得出子件实际的数量的数量
  1301. )
  1302. :return:line_out_ids ,line_in_ids
  1303. """
  1304. warehouse_id = self.env['warehouse'].search(
  1305. [('type', '=', 'stock')], limit=1)
  1306. line_out_ids, line_in_ids = [], []
  1307. parent_line = self.bom_id.line_parent_ids
  1308. if warehouse_id and self.bom_id and parent_line and self.bom_id.line_child_ids:
  1309. cost, cost_unit = parent_line.goods_id \
  1310. .get_suggested_cost_by_warehouse(
  1311. warehouse_id, self.goods_qty)
  1312. line_out_ids.append((0,0,{
  1313. 'goods_id': parent_line.goods_id.id,
  1314. 'attribute_id': parent_line.attribute_id.id,
  1315. 'warehouse_id': self.env[
  1316. 'warehouse'].get_warehouse_by_type('production').id,
  1317. 'warehouse_dest_id': warehouse_id.id,
  1318. 'uom_id': parent_line.goods_id.uom_id.id,
  1319. 'goods_qty': self.goods_qty,
  1320. 'goods_uos_qty': self.goods_qty / parent_line.goods_id.conversion,
  1321. 'uos_id': parent_line.goods_id.uos_id.id,
  1322. 'cost_unit': cost_unit,
  1323. 'cost': cost,
  1324. 'type': 'out',
  1325. }))
  1326. line_in_ids = [(0,0,{
  1327. 'goods_id': line.goods_id.id,
  1328. 'attribute_id': line.attribute_id.id,
  1329. 'warehouse_id': warehouse_id.id,
  1330. 'warehouse_dest_id': self.env[
  1331. 'warehouse'].get_warehouse_by_type('production').id,
  1332. 'uom_id': line.goods_id.uom_id.id,
  1333. 'goods_qty': line.goods_qty / parent_line.goods_qty * self.goods_qty,
  1334. 'goods_uos_qty': line.goods_qty / parent_line.goods_qty * self.goods_qty / line.goods_id.conversion,
  1335. 'uos_id':line.goods_id.uos_id.id,
  1336. 'type': 'in',
  1337. }) for line in self.bom_id.line_child_ids]
  1338. # self.line_in_ids = False
  1339. # self.line_out_ids = False
  1340. # self.line_out_ids = line_out_ids
  1341. # self.line_in_ids = line_in_ids
  1342. if line_out_ids:
  1343. line_ids += line_out_ids
  1344. if line_in_ids:
  1345. line_ids += line_in_ids
  1346. self.line_ids = False
  1347. self.line_ids = line_ids
  1348. @api.onchange('bom_id')
  1349. def onchange_bom(self):
  1350. line_out_ids, line_in_ids = [], []
  1351. domain = {}
  1352. warehouse_id = self.env['warehouse'].search(
  1353. [('type', '=', 'stock')], limit=1)
  1354. if self.bom_id:
  1355. line_out_ids = []
  1356. for line in self.bom_id.line_parent_ids:
  1357. cost, cost_unit = line.goods_id \
  1358. .get_suggested_cost_by_warehouse(
  1359. warehouse_id, line.goods_qty)
  1360. line_out_ids.append((0,0,{
  1361. 'goods_id': line.goods_id,
  1362. 'attribute_id': line.attribute_id.id,
  1363. 'designator': line.designator,
  1364. 'warehouse_id': self.env[
  1365. 'warehouse'].get_warehouse_by_type('production').id,
  1366. 'warehouse_dest_id': warehouse_id.id,
  1367. 'uom_id': line.goods_id.uom_id.id,
  1368. 'goods_qty': line.goods_qty,
  1369. 'goods_uos_qty': line.goods_qty / line.goods_id.conversion,
  1370. 'uos_id': line.goods_id.uos_id.id,
  1371. 'cost_unit': cost_unit,
  1372. 'cost': cost,
  1373. 'type': 'out',
  1374. }))
  1375. line_in_ids = [(0,0,{
  1376. 'goods_id': line.goods_id.id,
  1377. 'attribute_id': line.attribute_id.id,
  1378. 'warehouse_id': warehouse_id,
  1379. 'warehouse_dest_id': self.env[
  1380. 'warehouse'].get_warehouse_by_type('production').id,
  1381. 'uom_id': line.goods_id.uom_id.id,
  1382. 'goods_qty': line.goods_qty,
  1383. 'goods_uos_qty': line.goods_qty / line.goods_id.conversion,
  1384. 'uos_id':line.goods_id.uos_id.id,
  1385. 'type': 'in',
  1386. }) for line in self.bom_id.line_child_ids]
  1387. self.line_in_ids = False
  1388. self.line_out_ids = False
  1389. else:
  1390. self.goods_qty = 1
  1391. if len(line_out_ids) == 1 and line_out_ids:
  1392. """当物料清单中只有一个组合件的时候,默认本单据只有一个组合件 设置is_many_to_many_combinations 为False
  1393. 使试图只能在 many2one中选择一个商品(并且只能选择在物料清单中的商品),并且回写数量"""
  1394. self.goods_qty = line_out_ids[0][-1].get("goods_qty")
  1395. self.goods_id = line_out_ids[0][-1].get("goods_id")
  1396. domain = {'goods_id': [('id', '=', self.goods_id.id)]}
  1397. line_ids = []
  1398. if line_out_ids:
  1399. line_ids += line_out_ids
  1400. if line_in_ids:
  1401. line_ids += line_in_ids
  1402. self.line_ids = False
  1403. self.line_ids = line_ids
  1404. return {'domain': domain}
  1405. def update_bom(self):
  1406. for disassembly in self:
  1407. if disassembly.bom_id:
  1408. return disassembly.save_bom()
  1409. else:
  1410. return {
  1411. 'type': 'ir.actions.act_window',
  1412. 'res_model': 'save.bom.memory',
  1413. 'view_mode': 'form',
  1414. 'target': 'new',
  1415. }
  1416. def save_bom(self, name=''):
  1417. for disassembly in self:
  1418. line_child_ids = [[0, False, {
  1419. 'goods_id': line.goods_id.id,
  1420. 'goods_qty': line.goods_qty,
  1421. }] for line in disassembly.line_in_ids]
  1422. line_parent_ids = [[0, False, {
  1423. 'goods_id': line.goods_id.id,
  1424. 'goods_qty': line.goods_qty,
  1425. }] for line in disassembly.line_out_ids]
  1426. if disassembly.bom_id:
  1427. disassembly.bom_id.line_parent_ids.unlink()
  1428. disassembly.bom_id.line_child_ids.unlink()
  1429. disassembly.bom_id.write({
  1430. 'line_parent_ids': line_parent_ids,
  1431. 'line_child_ids': line_child_ids})
  1432. else:
  1433. bom_id = self.env['wh.bom'].create({
  1434. 'name': name,
  1435. 'type': 'disassembly',
  1436. 'line_parent_ids': line_parent_ids,
  1437. 'line_child_ids': line_child_ids,
  1438. })
  1439. disassembly.bom_id = bom_id
  1440. return True
  1441. class WhBom(models.Model):
  1442. _name = 'wh.bom'
  1443. _inherit = ['mail.thread']
  1444. _description = '物料清单'
  1445. BOM_TYPE = [
  1446. ('assembly', '组装单'),
  1447. ('disassembly', '拆卸单'),
  1448. ('outsource', '委外加工单'),
  1449. ]
  1450. name = fields.Char('物料清单名称',
  1451. help='组装/拆卸物料清单名称')
  1452. type = fields.Selection(
  1453. BOM_TYPE, '类型', default=lambda self: self.env.context.get('type'),
  1454. help='类型: 组装单、拆卸单')
  1455. version_control_id = fields.Many2one('core.value',
  1456. string='版本控制',
  1457. ondelete='restrict',
  1458. domain=[('type', '=', 'bom_version')],
  1459. help='版本控制:数据来源于系统基础配置')
  1460. line_parent_ids = fields.One2many(
  1461. 'wh.bom.line', 'bom_id', '组合件', domain=[('type', '=', 'parent')],
  1462. context={'type': 'parent'}, copy=True,
  1463. help='物料清单对应的组合件行')
  1464. line_child_ids = fields.One2many(
  1465. 'wh.bom.line', 'bom_id', '子件', domain=[('type', '=', 'child')],
  1466. context={'type': 'child'}, copy=True,
  1467. help='物料清单对应的子件行')
  1468. active = fields.Boolean('启用', default=True)
  1469. company_id = fields.Many2one(
  1470. 'res.company',
  1471. string='公司',
  1472. change_default=True,
  1473. default=lambda self: self.env.company)
  1474. goods_id = fields.Many2one('goods', related='line_parent_ids.goods_id', string='组合商品')
  1475. @api.constrains('line_parent_ids', 'line_child_ids')
  1476. def check_parent_child_unique(self):
  1477. """判断同一个产品不能是组合件又是子件"""
  1478. for wb in self:
  1479. for child_line in wb.line_child_ids:
  1480. for parent_line in wb.line_parent_ids:
  1481. if child_line.goods_id == parent_line.goods_id and child_line.attribute_id == parent_line.attribute_id:
  1482. raise UserError('组合件和子件不能相同,产品:%s' % parent_line.goods_id.name)
  1483. @api.onchange('line_parent_ids', 'line_child_ids')
  1484. def check_parent_length(self):
  1485. """判断组合件必填"""
  1486. for p in self:
  1487. if not p.name: # 新增记录,不处理
  1488. continue
  1489. elif not len(p.line_parent_ids) and len(p.line_child_ids): # 控制物料清单优先录入组合件,再录入子件
  1490. raise UserError('请先选择组合件,然后选择子件')
  1491. details = fields.Html('明细', compute='_compute_details')
  1492. @api.depends('line_parent_ids', 'line_child_ids')
  1493. def _compute_details(self):
  1494. for v in self:
  1495. vl = {'col': [], 'val': []}
  1496. vl['col'] = ['', '商品', '属性', '数量']
  1497. for l in v.line_parent_ids:
  1498. vl['val'].append(['组合件', l.goods_id.name or '', l.attribute_id.name or '', l.goods_qty or ''])
  1499. for l in v.line_child_ids:
  1500. vl['val'].append(['子件', l.goods_id.name or '', l.attribute_id.name or '', l.goods_qty or ''])
  1501. v.details = v.company_id._get_html_table(vl)
  1502. def name_get(self):
  1503. ret = super().name_get()
  1504. result = []
  1505. for r in ret:
  1506. result.append((r[0], '%s %s' %
  1507. (r[1], self.browse(r[0]).version_control_id.name or '')))
  1508. return result
  1509. @api.model_create_multi
  1510. def create(self, vals_list):
  1511. ret = super().create(vals_list)
  1512. for r in ret:
  1513. for l in r.line_parent_ids:
  1514. if r.type == 'assembly' and l.goods_id.get_way != 'self':
  1515. l.goods_id.get_way = 'self'
  1516. if r.type == 'outsource' and l.goods_id.get_way != 'ous':
  1517. l.goods_id.get_way = 'ous'
  1518. return ret
  1519. class WhBomLine(models.Model):
  1520. _name = 'wh.bom.line'
  1521. _description = '物料清单明细'
  1522. BOM_LINE_TYPE = [
  1523. ('parent', '组合件'),
  1524. ('child', '子件'),
  1525. ]
  1526. bom_id = fields.Many2one('wh.bom', '物料清单', ondelete='cascade',
  1527. help='子件行/组合件行对应的物料清单')
  1528. type = fields.Selection(
  1529. BOM_LINE_TYPE, '类型',
  1530. default=lambda self: self.env.context.get('type'),
  1531. help='类型: 组合件、子件')
  1532. goods_id = fields.Many2one('goods', '商品', ondelete='restrict',
  1533. help='子件行/组合件行上的商品')
  1534. goods_qty = fields.Float(
  1535. '数量', digits='Quantity',
  1536. default=1.0,
  1537. help='子件行/组合件行上的商品数量')
  1538. attribute_id = fields.Many2one('attribute', '属性', ondelete='restrict')
  1539. designator = fields.Char('位号') #电子行业用,需带到单据上
  1540. last_cost = fields.Float('最近成本', digits='Amount', compute='_get_last_cost')
  1541. company_id = fields.Many2one(
  1542. 'res.company',
  1543. string='公司',
  1544. change_default=True,
  1545. default=lambda self: self.env.company)
  1546. @api.constrains('goods_qty')
  1547. def check_goods_qty(self):
  1548. """验证商品数量大于0"""
  1549. for wbl in self:
  1550. if wbl.goods_qty <= 0:
  1551. raise UserError('商品 %s 的数量必须大于0' % wbl.goods_id.name)
  1552. def _get_last_cost(self):
  1553. for wbl in self:
  1554. wbl.last_cost = 0
  1555. last_in = self.env['wh.move.line'].search(
  1556. [('goods_id', '=', wbl.goods_id.id),
  1557. ('type', '=', 'in'),
  1558. ('state', '=', 'done')],
  1559. order="date desc", limit=1)
  1560. if last_in:
  1561. wbl.last_cost = last_in.cost_unit * wbl.goods_qty
上海开阖软件有限公司 沪ICP备12045867号-1