GoodERP
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

610 lines
27KB

  1. # Copyright 2016 上海开阖软件有限公司 (http://www.osbzr.com)
  2. # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
  3. from odoo import fields, models, api
  4. from odoo.exceptions import UserError
  5. import datetime
  6. from odoo.tools import float_compare, float_is_zero
  7. # 字段只读状态
  8. READONLY_STATES = {
  9. 'done': [('readonly', True)],
  10. }
  11. ISODATEFORMAT = '%Y-%m-%d'
  12. class BuyReceipt(models.Model):
  13. _name = "buy.receipt"
  14. _inherits = {'wh.move': 'buy_move_id'}
  15. _inherit = ['mail.thread']
  16. _description = "采购入库单"
  17. _order = 'date desc, id desc'
  18. @api.depends('line_in_ids.subtotal', 'discount_amount',
  19. 'payment', 'line_out_ids.subtotal', 'delivery_fee')
  20. def _compute_all_amount(selfs):
  21. '''当优惠金额改变时,改变成交金额'''
  22. for self in selfs:
  23. total = 0
  24. if self.line_in_ids:
  25. # 入库时优惠前总金额
  26. total = sum(line.subtotal for line in self.line_in_ids)
  27. elif self.line_out_ids:
  28. # 退货时优惠前总金额
  29. total = sum(line.subtotal for line in self.line_out_ids)
  30. self.amount = total - self.discount_amount + self.delivery_fee
  31. @api.depends('is_return', 'invoice_id.reconciled', 'invoice_id.amount')
  32. def _get_buy_money_state(selfs):
  33. '''返回付款状态'''
  34. for self in selfs:
  35. if not self.is_return:
  36. if self.invoice_id.reconciled == 0:
  37. self.money_state = '未付款'
  38. elif self.invoice_id.reconciled < self.invoice_id.amount:
  39. self.money_state = '部分付款'
  40. elif self.invoice_id.reconciled == self.invoice_id.amount:
  41. self.money_state = '全部付款'
  42. # 返回退款状态
  43. if self.is_return:
  44. if self.invoice_id.reconciled == 0:
  45. self.return_state = '未退款'
  46. elif abs(self.invoice_id.reconciled) < abs(self.invoice_id.amount):
  47. self.return_state = '部分退款'
  48. elif self.invoice_id.reconciled == self.invoice_id.amount:
  49. self.return_state = '全部退款'
  50. buy_move_id = fields.Many2one('wh.move', '入库单',
  51. required=True, ondelete='cascade',
  52. help='入库单号')
  53. is_return = fields.Boolean('是否退货',
  54. default=lambda self: self.env.context.get(
  55. 'is_return'),
  56. help='是否为退货类型')
  57. order_id = fields.Many2one('buy.order', '订单号',
  58. copy=False, ondelete='cascade',
  59. help='产生入库单/退货单的采购订单')
  60. invoice_id = fields.Many2one('money.invoice', '发票号', copy=False,
  61. ondelete='set null',
  62. help='产生的发票号')
  63. date_due = fields.Date('到期日期', copy=False,
  64. default=lambda self: fields.Date.context_today(
  65. self),
  66. help='付款截止日期')
  67. discount_rate = fields.Float('优惠率(%)', states=READONLY_STATES,
  68. help='整单优惠率')
  69. discount_amount = fields.Float('优惠金额', states=READONLY_STATES,
  70. digits='Amount',
  71. help='整单优惠金额,可由优惠率自动计算得出,也可手动输入')
  72. invoice_by_receipt = fields.Boolean(string="按收货结算", default=True,
  73. help='如未勾选此项,可在资金行里输入付款金额,订单保存后,采购人员可以单击资金行上的【确认】按钮。')
  74. amount = fields.Float('成交金额', compute=_compute_all_amount,
  75. store=True, readonly=True,
  76. digits='Amount',
  77. help='总金额减去优惠金额')
  78. payment = fields.Float('本次付款', states=READONLY_STATES,
  79. digits='Amount',
  80. help='本次付款金额')
  81. bank_account_id = fields.Many2one('bank.account', '结算账户',
  82. ondelete='restrict',
  83. help='用来核算和监督企业与其他单位或个人之间的债权债务的结算情况')
  84. cost_line_ids = fields.One2many('cost.line', 'buy_id', '采购费用', copy=False,
  85. help='采购费用明细行')
  86. money_state = fields.Char('付款状态', compute=_get_buy_money_state,
  87. store=True, default='未付款',
  88. help="采购入库单的付款状态",
  89. index=True, copy=False)
  90. return_state = fields.Char('退款状态', compute=_get_buy_money_state,
  91. store=True, default='未退款',
  92. help="采购退货单的退款状态",
  93. index=True, copy=False)
  94. voucher_id = fields.Many2one('voucher', '入库凭证', readonly=True,
  95. copy=False,
  96. help='入库时产生的入库凭证')
  97. origin_id = fields.Many2one('buy.receipt', '来源单据', copy=False)
  98. currency_id = fields.Many2one('res.currency',
  99. '外币币别',
  100. help='外币币别')
  101. currency_rate = fields.Float('汇率',digits='Price')
  102. delivery_fee = fields.Float('运费')
  103. money_order_id = fields.Many2one(
  104. 'money.order',
  105. '付款单',
  106. readonly=True,
  107. copy=False,
  108. help='输入本次付款确认时产生的付款单')
  109. def set_today(self):
  110. self.date = fields.Date.today()
  111. def _compute_total(self, line_ids):
  112. return sum(line.subtotal for line in line_ids)
  113. @api.onchange('discount_rate', 'line_in_ids', 'line_out_ids')
  114. def onchange_discount_rate(self):
  115. '''当优惠率或订单行发生变化时,单据优惠金额发生变化'''
  116. line = self.line_in_ids or self.line_out_ids
  117. total = self._compute_total(line)
  118. if self.discount_rate:
  119. self.discount_amount = total * self.discount_rate * 0.01
  120. @api.onchange('partner_id')
  121. def onchange_partner_id(self):
  122. if self.partner_id:
  123. for line in self.line_in_ids:
  124. line.tax_rate = line.goods_id.get_tax_rate(line.goods_id, self.partner_id, 'buy')
  125. def get_move_origin(self, vals):
  126. return self._name + (self.env.context.get('is_return') and
  127. '.return' or '.buy')
  128. @api.model
  129. def create(self, vals):
  130. '''创建采购入库单时生成有序编号'''
  131. if not self.env.context.get('is_return'):
  132. name = self._name
  133. else:
  134. name = 'buy.return'
  135. if vals.get('name', '/') == '/':
  136. vals['name'] = self.env['ir.sequence'].next_by_code(name) or '/'
  137. vals.update({
  138. 'origin': self.get_move_origin(vals),
  139. 'finance_category_id': self.env.ref('finance.categ_buy_goods').id,
  140. })
  141. return super(BuyReceipt, self).create(vals)
  142. def unlink(self):
  143. for receipt in self:
  144. receipt.buy_move_id.unlink()
  145. def _wrong_receipt_done(self):
  146. self.ensure_one()
  147. if self.state == 'done':
  148. raise UserError('请不要重复入库')
  149. batch_one_list_wh = []
  150. batch_one_list = []
  151. for line in self.line_in_ids:
  152. if line.amount < 0:
  153. raise UserError('采购金额不能小于 0!请修改。')
  154. if line.goods_id.force_batch_one:
  155. wh_move_lines = self.env['wh.move.line'].search(
  156. [('state', '=', 'done'), ('type', '=', 'in'), ('goods_id', '=', line.goods_id.id)])
  157. for move_line in wh_move_lines:
  158. if (move_line.goods_id.id, move_line.lot) not in batch_one_list_wh and move_line.lot:
  159. batch_one_list_wh.append(
  160. (move_line.goods_id.id, move_line.lot))
  161. if (line.goods_id.id, line.lot) in batch_one_list_wh:
  162. raise UserError('仓库已存在相同序列号的商品!\n商品:%s 序列号:%s' %
  163. (line.goods_id.name, line.lot))
  164. for line in self.line_in_ids:
  165. if line.goods_qty <= 0 or line.price_taxed < 0:
  166. raise UserError('商品 %s 的数量和含税单价不能小于0!' % line.goods_id.name)
  167. if line.goods_id.force_batch_one:
  168. batch_one_list.append((line.goods_id.id, line.lot))
  169. if len(batch_one_list) > len(set(batch_one_list)):
  170. raise UserError('不能创建相同序列号的商品!\n 序列号列表为%s' %
  171. [lot[1] for lot in batch_one_list])
  172. for line in self.line_out_ids:
  173. if line.amount < 0:
  174. raise UserError('退货金额不能小于 0!请修改。')
  175. if line.goods_qty <= 0 or line.price_taxed < 0:
  176. raise UserError('商品 %s 的数量和含税单价不能小于0!' % line.goods_id.name)
  177. if not self.bank_account_id and self.payment:
  178. raise UserError('付款额不为空时,请选择结算账户!')
  179. decimal_amount = self.env.ref('core.decimal_amount')
  180. if float_compare(self.payment, self.amount, precision_digits=decimal_amount.digits) == 1:
  181. raise UserError('本次付款金额不能大于折后金额!\n付款金额:%s 折后金额:%s' %
  182. (self.payment, self.amount))
  183. if float_compare(sum(cost_line.amount for cost_line in self.cost_line_ids),
  184. sum(line.share_cost for line in self.line_in_ids),
  185. precision_digits=decimal_amount.digits) != 0:
  186. raise UserError('采购费用还未分摊或分摊不正确!\n采购费用:%s 分摊总费用:%s' %
  187. (sum(cost_line.amount for cost_line in self.cost_line_ids),
  188. sum(line.share_cost for line in self.line_in_ids)))
  189. return
  190. def _line_qty_write(self):
  191. self.ensure_one()
  192. if self.order_id:
  193. for line in self.line_in_ids:
  194. decimal_quantity = self.env.ref('core.decimal_quantity')
  195. if float_compare(
  196. line.buy_line_id.quantity_in + line.goods_qty,
  197. line.buy_line_id.quantity,
  198. decimal_quantity.digits) == 1:
  199. if not line.goods_id.excess:
  200. raise UserError('%s收货数量大于订单数量' % line.goods_id.name)
  201. line.buy_line_id.quantity_in += line.goods_qty
  202. for line in self.line_out_ids: # 退货单行
  203. if self.order_id.type == 'return': # 退货类型的buy_order生成的采购退货单审核
  204. line.buy_line_id.quantity_in += line.goods_qty
  205. else:
  206. line.buy_line_id.quantity_in -= line.goods_qty
  207. return
  208. def _get_invoice_vals(self, partner_id, category_id, date, amount, tax_amount):
  209. '''返回创建 money_invoice 时所需数据'''
  210. return {
  211. 'move_id': self.buy_move_id.id,
  212. 'name': self.name,
  213. 'partner_id': partner_id.id,
  214. 'pay_method': self.order_id.pay_method.id or partner_id.pay_method.id,
  215. 'currency_id': partner_id.s_category_id.account_id.currency_id.id,
  216. 'category_id': category_id.id,
  217. 'date': date,
  218. 'amount': amount,
  219. 'reconciled': 0,
  220. 'to_reconcile': amount,
  221. 'tax_amount': tax_amount,
  222. 'date_due': self.date_due,
  223. 'state': 'draft',
  224. }
  225. def _receipt_make_invoice(self):
  226. '''入库单/退货单 生成结算单'''
  227. invoice_id = False
  228. if not self.is_return:
  229. if not self.invoice_by_receipt:
  230. return False
  231. amount = sum(line.buy_line_id.price_taxed*line.goods_qty
  232. for line in self.line_in_ids)
  233. tax_amount = sum(line.tax_amount for line in self.line_in_ids)
  234. else:
  235. amount = -self.amount
  236. tax_amount = - sum(line.tax_amount for line in self.line_out_ids)
  237. categ = self.env.ref('money.core_category_purchase')
  238. if not float_is_zero(amount, 2):
  239. invoice_id = self.env['money.invoice'].create(
  240. self._get_invoice_vals(
  241. self.partner_id, categ, self.date, amount, tax_amount)
  242. )
  243. return invoice_id
  244. def _buy_amount_to_invoice(self):
  245. '''采购费用产生结算单'''
  246. self.ensure_one()
  247. if sum(cost_line.amount for cost_line in self.cost_line_ids) > 0:
  248. for line in self.cost_line_ids:
  249. if not float_is_zero(line.amount, 2):
  250. self.env['money.invoice'].create(
  251. self._get_invoice_vals(line.partner_id, line.category_id, self.date, line.amount + line.tax,
  252. line.tax)
  253. )
  254. return
  255. def _make_payment(self, invoice_id, amount, this_reconcile):
  256. '''根据传入的invoice_id生成付款单'''
  257. categ = self.env.ref('money.core_category_purchase')
  258. money_lines = [
  259. {'bank_id': self.bank_account_id.id, 'amount': this_reconcile}]
  260. source_lines = [{'name': invoice_id.id,
  261. 'category_id': categ.id,
  262. 'date': invoice_id.date,
  263. 'amount': amount,
  264. 'reconciled': 0.0,
  265. 'to_reconcile': amount,
  266. 'this_reconcile': this_reconcile}]
  267. rec = self.with_context(type='pay')
  268. money_order = rec.env['money.order'].create({
  269. 'partner_id': self.partner_id.id,
  270. 'bank_name': self.partner_id.bank_name,
  271. 'bank_num': self.partner_id.bank_num,
  272. 'date' : fields.Date.context_today(self),
  273. 'line_ids':
  274. [(0, 0, line) for line in money_lines],
  275. 'source_ids':
  276. [(0, 0, line) for line in source_lines] if invoice_id.state == 'done' else False,
  277. 'amount': amount,
  278. 'reconciled': this_reconcile,
  279. 'to_reconcile': amount,
  280. 'state': 'draft',
  281. 'origin_name': self.name,
  282. 'buy_id': self.order_id.id,
  283. })
  284. return money_order
  285. def _create_voucher_line(self, account_id, debit, credit, voucher_id, goods_id, goods_qty, partner_id):
  286. '''返回voucher line'''
  287. voucher = self.env['voucher.line'].create({
  288. 'name': '%s %s' % (self.name, ''),
  289. 'account_id': account_id and account_id.id,
  290. 'partner_id': partner_id and partner_id.id,
  291. 'debit': debit,
  292. 'credit': credit,
  293. 'voucher_id': voucher_id and voucher_id.id,
  294. 'goods_id': goods_id and goods_id.id,
  295. 'goods_qty': goods_qty,
  296. })
  297. return voucher
  298. def create_voucher(self):
  299. '''
  300. 借: 商品分类对应的会计科目 一般是库存商品
  301. 贷:类型为支出的类别对应的会计科目 一般是材料采购
  302. 当一张入库单有多个商品的时候,按对应科目汇总生成多个借方凭证行。
  303. 采购退货单生成的金额为负
  304. '''
  305. self.ensure_one()
  306. vouch_id = self.env['voucher'].create({'date': self.date, 'ref': '%s,%s' % (self._name, self.id)})
  307. sum_amount = 0
  308. if not self.is_return:
  309. for line in self.line_in_ids:
  310. if line.amount:
  311. # 借方明细
  312. self._create_voucher_line(line.goods_id.category_id.account_id,
  313. line.amount + line.share_cost, 0, vouch_id, line.goods_id, line.goods_qty, False)
  314. sum_amount += line.amount
  315. if sum_amount:
  316. # 贷方明细
  317. self._create_voucher_line(self.buy_move_id.finance_category_id.account_id,
  318. 0, sum_amount, vouch_id, False, 0, self.partner_id)
  319. for l in self.cost_line_ids:
  320. self._create_voucher_line(self.buy_move_id.finance_category_id.account_id,
  321. 0, l.amount, vouch_id, False, 0, l.partner_id)
  322. if self.is_return:
  323. for line in self.line_out_ids:
  324. if line.amount:
  325. # 借方明细
  326. self._create_voucher_line(line.goods_id.category_id.account_id,
  327. -line.amount, 0, vouch_id, line.goods_id, line.goods_qty, False)
  328. sum_amount += line.amount
  329. if sum_amount:
  330. # 贷方明细
  331. self._create_voucher_line(self.buy_move_id.finance_category_id.account_id,
  332. 0, -sum_amount, vouch_id, False, 0, self.partner_id)
  333. if len(vouch_id.line_ids) > 0:
  334. vouch_id.voucher_done()
  335. return vouch_id
  336. else:
  337. vouch_id.unlink()
  338. def multi_currency_rate(self):
  339. if self.currency_rate:
  340. for l in self.line_in_ids:
  341. l.price = l.buy_line_id.price * self.currency_rate
  342. l.onchange_price()
  343. l.cost = l.price * l.goods_qty - l.discount_amount + l.share_cost
  344. def buy_receipt_done(self):
  345. '''审核采购入库单/退货单,更新本单的付款状态/退款状态,并生成结算单和付款单'''
  346. self.ensure_one()
  347. # 入库单/退货单 生成结算单
  348. invoice_id = self._receipt_make_invoice()
  349. self.multi_currency_rate()
  350. self._wrong_receipt_done()
  351. # 调用wh.move中审核方法,更新审核人和审核状态
  352. self.buy_move_id.approve_order()
  353. # 将收货/退货数量写入订单行
  354. self._line_qty_write()
  355. # 创建入库的会计凭证
  356. voucher = self.create_voucher()
  357. # 采购费用产生结算单
  358. self._buy_amount_to_invoice()
  359. # 生成付款单
  360. money_order = False
  361. if self.payment:
  362. flag = not self.is_return and 1 or -1
  363. amount = flag * self.amount
  364. this_reconcile = flag * self.payment
  365. money_order = self._make_payment(invoice_id, amount, this_reconcile)
  366. self.write({
  367. 'voucher_id': voucher and voucher.id,
  368. 'invoice_id': invoice_id and invoice_id.id,
  369. 'money_order_id': money_order and money_order.id,
  370. 'state': 'done', # 为保证审批流程顺畅,否则,未审批就可审核
  371. })
  372. if self.order_id:
  373. # 如果已退货也已退款,不生成新的分单
  374. if self.is_return and self.payment:
  375. return True
  376. #产生新的入库单时,如果已经存在草稿的入库单时,先将已经存在的草稿进行删除
  377. self.env['buy.receipt'].search(['&',('state', '=', 'draft'),'&',('order_id','=', self.order_id.id),('is_return', '=', False)]).unlink()
  378. return self.order_id.buy_generate_receipt()
  379. def buy_receipt_draft(self):
  380. '''反审核采购入库单/退货单,更新本单的付款状态/退款状态,并删除生成的结算单、付款单及凭证'''
  381. self.ensure_one()
  382. if self.state == 'draft':
  383. raise UserError('请不要重复撤销 %s' % self._description)
  384. # 查找产生的结算单(除了供应商的发票还有可能是采购费用发票)
  385. invoice_ids = self.env['money.invoice'].search([('name', '=', self.name)])
  386. for invoice in invoice_ids:
  387. # 不能反审核已核销的发货单
  388. if invoice.reconciled != 0:
  389. raise UserError('发票已核销,不能撤销入库!')
  390. if invoice.state == 'done':
  391. if self.env.company.draft_invoice:
  392. raise UserError('发票已收不可撤销入库')
  393. invoice.money_invoice_draft()
  394. invoice.unlink()
  395. # 反审核采购入库单时删除对应的入库凭证
  396. voucher = self.voucher_id
  397. if voucher.state == 'done':
  398. voucher.voucher_draft()
  399. voucher.unlink()
  400. self.write({
  401. 'state': 'draft',
  402. })
  403. # 修改订单行中已执行数量
  404. if self.order_id:
  405. for line in self.line_in_ids:
  406. line.buy_line_id.quantity_in -= line.goods_qty
  407. for line in self.line_out_ids:
  408. if self.order_id.type == 'return':
  409. line.buy_line_id.quantity_in -= line.goods_qty
  410. else:
  411. line.buy_line_id.quantity_in += line.goods_qty
  412. # 调用wh.move中反审核方法,更新审核人和审核状态
  413. self.buy_move_id.cancel_approved_order()
  414. def buy_share_cost(self):
  415. '''入库单上的采购费用分摊到入库单明细行上'''
  416. self.ensure_one()
  417. total_amount = 0
  418. for line in self.line_in_ids:
  419. total_amount += line.amount
  420. cost = sum(cost_line.amount for cost_line in self.cost_line_ids)
  421. for line in self.line_in_ids:
  422. line.share_cost = cost / total_amount * line.amount
  423. share_cost = sum(line.share_cost for line in self.line_in_ids)
  424. diff_cost = cost - share_cost
  425. self.line_in_ids[0].share_cost = self.line_in_ids[0].share_cost + diff_cost
  426. return True
  427. def buy_to_return(self):
  428. '''采购入库单转化为采购退货单'''
  429. return_goods = {}
  430. return_order_draft = self.search([
  431. ('is_return', '=', True),
  432. ('origin_id', '=', self.id),
  433. ('state', '=', 'draft')
  434. ])
  435. if return_order_draft:
  436. raise UserError('采购入库单存在草稿状态的退货单!')
  437. return_order = self.search([
  438. ('is_return', '=', True),
  439. ('origin_id', '=', self.id),
  440. ('state', '=', 'done')
  441. ])
  442. for order in return_order:
  443. for return_line in order.line_out_ids:
  444. # 用产品、属性、批次做key记录已退货数量
  445. t_key = (return_line.goods_id.id,
  446. return_line.attribute_id.id, return_line.lot_id.lot)
  447. if return_goods.get(t_key):
  448. return_goods[t_key] += return_line.goods_qty
  449. else:
  450. return_goods[t_key] = return_line.goods_qty
  451. receipt_line = []
  452. for line in self.line_in_ids:
  453. qty = line.goods_qty
  454. l_key = (line.goods_id.id, line.attribute_id.id, line.lot)
  455. if return_goods.get(l_key):
  456. qty = qty - return_goods[l_key]
  457. if qty > 0:
  458. dic = {
  459. 'goods_id': line.goods_id.id,
  460. 'attribute_id': line.attribute_id.id,
  461. 'uom_id': line.uom_id.id,
  462. 'warehouse_id': line.warehouse_dest_id.id,
  463. 'warehouse_dest_id': line.warehouse_id.id,
  464. 'buy_line_id': line.buy_line_id.id,
  465. 'goods_qty': qty,
  466. 'price_taxed': line.price_taxed,
  467. 'price': line.price,
  468. 'tax_rate':line.tax_rate,
  469. 'cost_unit': line.cost_unit,
  470. 'discount_rate': line.discount_rate,
  471. 'discount_amount': line.discount_amount,
  472. 'type': 'out'
  473. }
  474. receipt_line.append(dic)
  475. if len(receipt_line) == 0:
  476. raise UserError('该订单已全部退货!')
  477. vals = {'partner_id': self.partner_id.id,
  478. 'is_return': True,
  479. 'order_id': self.order_id.id,
  480. 'origin_id': self.id,
  481. 'origin': 'buy.receipt.return',
  482. 'warehouse_dest_id': self.warehouse_id.id,
  483. 'warehouse_id': self.warehouse_dest_id.id,
  484. 'bank_account_id': self.bank_account_id.id,
  485. 'date_due': (datetime.datetime.now()).strftime(ISODATEFORMAT),
  486. 'date': (datetime.datetime.now()).strftime(ISODATEFORMAT),
  487. 'line_out_ids': [(0, 0, line) for line in receipt_line],
  488. 'discount_amount': self.discount_amount,
  489. }
  490. delivery_return = self.with_context(is_return=True).create(vals)
  491. view_id = self.env.ref('buy.buy_return_form').id
  492. name = '采购退货单'
  493. return {
  494. 'name': name,
  495. 'view_mode': 'form',
  496. 'view_id': False,
  497. 'views': [(view_id, 'form')],
  498. 'res_model': 'buy.receipt',
  499. 'type': 'ir.actions.act_window',
  500. 'res_id': delivery_return.id,
  501. 'target': 'current'
  502. }
  503. class WhMoveLine(models.Model):
  504. _inherit = 'wh.move.line'
  505. buy_line_id = fields.Many2one('buy.order.line',
  506. '采购单行', ondelete='cascade',
  507. help='对应的采购订单行')
  508. def _buy_get_price_and_tax(self):
  509. self.tax_rate = self.env.user.company_id.import_tax_rate
  510. self.price_taxed = self.goods_id.cost
  511. order_id = self.buy_line_id and self.buy_line_id.order_id.id or self.env.context.get('order_id')
  512. if order_id:
  513. line_domain = [
  514. ('order_id', '=', order_id),
  515. ('goods_id', '=', self.goods_id.id)
  516. ]
  517. # 如果行有属性,添加进搜索条件
  518. if self.attribute_id:
  519. line_domain.append(('attribute_id', '=', self.attribute_id.id))
  520. else:
  521. pass
  522. line = self.env['buy.order.line'].search(line_domain, limit=1)
  523. if line:
  524. self.buy_line_id = line.id
  525. self.uos_id = line.goods_id.uos_id.id
  526. self.uom_id = line.uom_id.id
  527. self.cost_unit = line.price
  528. self.price = line.price
  529. self.price_taxed = line.price_taxed
  530. self.discount_rate = line.discount_rate
  531. self.tax_rate = line.tax_rate
  532. self.plan_date = line.order_id.planned_date
  533. else:
  534. raise UserError('无此商品的订单行')
  535. @api.onchange('attribute_id')
  536. def onchange_attribute_id(self):
  537. '''当订单行的商品属性变化时,计算准确的采购订单行'''
  538. self.ensure_one()
  539. if self.attribute_id:
  540. self._buy_get_price_and_tax()
  541. @api.onchange('goods_id')
  542. def onchange_goods_id(self):
  543. '''当订单行的商品变化时,带出商品上的成本价,以及公司的进项税'''
  544. self.ensure_one()
  545. if self.goods_id:
  546. is_return = self.env.context.get('default_is_return')
  547. # 如果是采购入库单行 或 采购退货单行
  548. if is_return is not None and \
  549. ((self.type == 'in' and not is_return) or (self.type == 'out' and is_return)):
  550. self._buy_get_price_and_tax()
  551. return super(WhMoveLine, self).onchange_goods_id()
上海开阖软件有限公司 沪ICP备12045867号-1