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.

322 lines
15 KiB

  1. # -*- coding: utf-8 -*-
  2. from datetime import datetime
  3. from openerp import api, fields, models, _
  4. from openerp.exceptions import ValidationError
  5. class operation_request(models.Model):
  6. _name = 'operation.request'
  7. def get_date_now(self):
  8. return datetime.strftime(datetime.now(), '%Y-%m-%d')
  9. @api.multi
  10. @api.depends('share_product_id', 'share_product_id.list_price', 'quantity')
  11. def _compute_subscription_amount(self):
  12. for operation_request in self:
  13. operation_request.subscription_amount = (operation_request.
  14. share_product_id.
  15. list_price *
  16. operation_request.
  17. quantity)
  18. request_date = fields.Date(string='Request date',
  19. default=lambda self: self.get_date_now())
  20. effective_date = fields.Date(string='Effective date', required=False)
  21. partner_id = fields.Many2one('res.partner',
  22. string='Cooperator',
  23. domain=[('member', '=', True)],
  24. required=True)
  25. partner_id_to = fields.Many2one('res.partner',
  26. string='Transfered to',
  27. domain=[('cooperator', '=', True)])
  28. operation_type = fields.Selection([('subscription', 'Subscription'),
  29. ('transfer', 'Transfer'),
  30. ('sell_back', 'Sell Back'),
  31. ('convert', 'Conversion')],
  32. string='Operation Type',
  33. required=True)
  34. share_product_id = fields.Many2one('product.product',
  35. string='Share type',
  36. domain=[('is_share', '=', True)],
  37. required=True)
  38. share_to_product_id = fields.Many2one('product.product',
  39. string='Convert to this share type',
  40. domain=[('is_share', '=', True)])
  41. share_short_name = fields.Char(related='share_product_id.short_name',
  42. string='Share type name')
  43. share_to_short_name = fields.Char(related='share_to_product_id.short_name',
  44. string='Share to type name')
  45. share_unit_price = fields.Float(related='share_product_id.list_price',
  46. string='Share price')
  47. share_to_unit_price = fields.Float(related='share_to_product_id.list_price',
  48. string='Share to price')
  49. subscription_amount = fields.Float(compute='_compute_subscription_amount',
  50. string='Operation amount')
  51. quantity = fields.Integer(string='Number of share',
  52. required=True)
  53. state = fields.Selection([('draft', 'Draft'),
  54. ('waiting', 'Waiting'),
  55. ('approved', 'Approved'),
  56. ('done', 'Done'),
  57. ('cancelled', 'Cancelled'),
  58. ('refused', 'Refused')],
  59. string='State',
  60. required=True,
  61. default='draft')
  62. user_id = fields.Many2one('res.users',
  63. string='Responsible',
  64. readonly=True,
  65. default=lambda self: self.env.user)
  66. subscription_request = fields.One2many('subscription.request',
  67. 'operation_request_id',
  68. string="Share Receiver Info",
  69. help="In case on a transfer of"
  70. " share. If the share receiver"
  71. " isn't a effective member then a"
  72. " subscription form should"
  73. " be filled.")
  74. receiver_not_member = fields.Boolean(string='Receiver is not a member')
  75. company_id = fields.Many2one('res.company',
  76. string='Company',
  77. required=True,
  78. change_default=True,
  79. readonly=True,
  80. default=lambda self: self.env['res.company']._company_default_get())
  81. invoice = fields.Many2one('account.invoice',
  82. string="Invoice")
  83. @api.multi
  84. @api.constrains("effective_date")
  85. def _constrain_effective_date(self):
  86. for obj in self:
  87. if obj.effective_date > fields.Datetime.now():
  88. raise ValidationError("The effective date can not be in the future.")\
  89. @api.multi
  90. def approve_operation(self):
  91. for rec in self:
  92. rec.write({'state': 'approved'})
  93. @api.multi
  94. def refuse_operation(self):
  95. for rec in self:
  96. rec.write({'state': 'refused'})
  97. @api.multi
  98. def submit_operation(self):
  99. for rec in self:
  100. rec.validate()
  101. rec.write({'state': 'waiting'})
  102. @api.multi
  103. def cancel_operation(self):
  104. for rec in self:
  105. rec.write({'state': 'cancelled'})
  106. @api.multi
  107. def reset_to_draft(self):
  108. for rec in self:
  109. rec.write({'state': 'draft'})
  110. def get_total_share_dic(self, partner):
  111. total_share_dic = {}
  112. prod_template_obj = self.env['product.product']
  113. share_products = prod_template_obj.search([('is_share', '=', True)])
  114. for share_product in share_products:
  115. total_share_dic[share_product.id] = 0
  116. for line in partner.share_ids:
  117. total_share_dic[line.share_product_id.id] += line.share_number
  118. return total_share_dic
  119. # This function doesn't handle the case of a cooperator can own
  120. # different kinds of share type
  121. def hand_share_over(self, partner, share_product_id, quantity):
  122. if not partner.member:
  123. raise ValidationError(_("This operation can't be executed if the"
  124. " cooperator is not an effective member"))
  125. share_ind = len(partner.share_ids)
  126. i = 1
  127. while quantity > 0:
  128. line = self.partner_id.share_ids[share_ind-i]
  129. if line.share_product_id.id == share_product_id.id:
  130. if quantity > line.share_number:
  131. quantity -= line.share_number
  132. line.unlink()
  133. else:
  134. share_left = line.share_number - quantity
  135. quantity = 0
  136. line.write({'share_number': share_left})
  137. i += 1
  138. # if the cooperator sold all his shares he's no more
  139. # an effective member
  140. remaning_share_dict = 0
  141. for share_quant in self.get_total_share_dic(partner).values():
  142. remaning_share_dict += share_quant
  143. if remaning_share_dict == 0:
  144. self.partner_id.write({'member': False, 'old_member': True})
  145. def has_share_type(self):
  146. for line in self.partner_id.share_ids:
  147. if line.share_product_id.id == self.share_product_id.id:
  148. return True
  149. return False
  150. def validate(self):
  151. if not self.has_share_type() and \
  152. self.operation_type in ['sell_back', 'transfer']:
  153. raise ValidationError(_("The cooperator doesn't own this share"
  154. " type. Please choose the appropriate"
  155. " share type."))
  156. if self.operation_type in ['sell_back', 'convert', 'transfer']:
  157. total_share_dic = self.get_total_share_dic(self.partner_id)
  158. if self.quantity > total_share_dic[self.share_product_id.id]:
  159. raise ValidationError(_("The cooperator can't hand over more"
  160. " shares that he/she owns."))
  161. if self.operation_type == 'convert':
  162. amount_to_convert = self.share_unit_price * self.quantity
  163. share_price = self.share_to_product_id.list_price
  164. remainder = amount_to_convert % share_price
  165. if remainder != 0:
  166. raise ValidationError(_("The conversion must give a whole"
  167. " number for quantity"))
  168. if self.company_id.unmix_share_type:
  169. if self.share_product_id.code == self.share_to_product_id.code:
  170. raise ValidationError(_("You can't convert the share to"
  171. " the same share type."))
  172. if self.subscription_amount != self.partner_id.total_value:
  173. raise ValidationError(_("You must convert all the shares"
  174. " to the selected type."))
  175. else:
  176. if self.subscription_amount != self.partner_id.total_value:
  177. raise ValidationError(_("Converting just part of the"
  178. " shares is not yet implemented"))
  179. elif self.operation_type == 'transfer':
  180. if not self.receiver_not_member and self.company_id.unmix_share_type \
  181. and (self.partner_id_to.cooperator_type
  182. and self.partner_id.cooperator_type != self.partner_id_to.cooperator_type):
  183. raise ValidationError(_("This share type could not be"
  184. " transfered to " +
  185. self.partner_id_to.name))
  186. if self.receiver_not_member and self.subscription_request \
  187. and not self.subscription_request.validated:
  188. raise ValidationError(_("The information of the receiver"
  189. " are not correct. Please correct"
  190. " the information before"
  191. " submitting"))
  192. @api.multi
  193. def execute_operation(self):
  194. self.ensure_one()
  195. if self.effective_date:
  196. effective_date = self.effective_date
  197. else:
  198. effective_date = self.get_date_now()
  199. sub_request = self.env['subscription.request']
  200. for rec in self:
  201. rec.validate()
  202. if rec.state != 'approved':
  203. raise ValidationError(_("This operation must be approved"
  204. " before to be executed"))
  205. values = {
  206. 'partner_id': rec.partner_id.id, 'quantity': rec.quantity,
  207. 'share_product_id': rec.share_product_id.id,
  208. 'type': rec.operation_type,
  209. 'share_unit_price': rec.share_unit_price,
  210. 'date': effective_date,
  211. }
  212. if rec.operation_type == 'sell_back':
  213. self.hand_share_over(rec.partner_id, rec.share_product_id,
  214. rec.quantity)
  215. elif rec.operation_type == 'convert':
  216. amount_to_convert = rec.share_unit_price * rec.quantity
  217. convert_quant = int(amount_to_convert / rec.share_to_product_id.list_price)
  218. remainder = amount_to_convert % rec.share_to_product_id.list_price
  219. if convert_quant > 0 and remainder == 0:
  220. share_ids = rec.partner_id.share_ids
  221. line = share_ids[0]
  222. if len(share_ids) > 1:
  223. share_ids[1:len(share_ids)].unlink()
  224. line.write({
  225. 'share_number': convert_quant,
  226. 'share_product_id': rec.share_to_product_id.id,
  227. 'share_unit_price': rec.share_to_unit_price,
  228. 'share_short_name': rec.share_to_short_name
  229. })
  230. values['share_to_product_id'] = rec.share_to_product_id.id
  231. values['quantity_to'] = convert_quant
  232. else:
  233. raise ValidationError(_("Converting just part of the"
  234. " shares is not yet implemented"))
  235. elif rec.operation_type == 'transfer':
  236. sequence_id = self.env.ref('easy_my_coop.sequence_subscription', False)
  237. if rec.receiver_not_member:
  238. partner = rec.subscription_request.create_coop_partner()
  239. # get cooperator number
  240. sub_reg_num = int(sequence_id.next_by_id())
  241. partner_vals = sub_request.get_eater_vals(partner, rec.share_product_id)
  242. partner_vals['member'] = True
  243. partner_vals['cooperator_register_number'] = sub_reg_num
  244. partner.write(partner_vals)
  245. rec.partner_id_to = partner
  246. else:
  247. # means an old member or cooperator candidate
  248. if not rec.partner_id_to.member:
  249. if rec.partner_id_to.cooperator_register_number == 0:
  250. sub_reg_num = int(sequence_id.next_by_id())
  251. partner_vals['cooperator_register_number'] = sub_reg_num
  252. partner_vals = sub_request.get_eater_vals(
  253. rec.partner_id_to,
  254. rec.share_product_id)
  255. partner_vals['member'] = True
  256. partner_vals['old_member'] = False
  257. rec.partner_id_to.write(partner_vals)
  258. # remove the parts to the giver
  259. self.hand_share_over(rec.partner_id,
  260. rec.share_product_id,
  261. rec.quantity)
  262. # give the share to the receiver
  263. self.env['share.line'].create({
  264. 'share_number': rec.quantity,
  265. 'partner_id': rec.partner_id_to.id,
  266. 'share_product_id': rec.share_product_id.id,
  267. 'share_unit_price': rec.share_unit_price,
  268. 'effective_date': effective_date})
  269. values['partner_id_to'] = rec.partner_id_to.id
  270. else:
  271. raise ValidationError(_("This operation is not yet"
  272. " implemented."))
  273. sequence_operation = self.env.ref('easy_my_coop.sequence_register_operation', False)
  274. sub_reg_operation = sequence_operation.next_by_id()
  275. values['name'] = sub_reg_operation
  276. values['register_number_operation'] = int(sub_reg_operation)
  277. rec.write({'state': 'done'})
  278. # send mail to the receiver
  279. if rec.operation_type == 'transfer':
  280. certificat_email_template = self.env.ref('easy_my_coop.email_template_share_transfer', False)
  281. certificat_email_template.send_mail(rec.partner_id_to.id, False)
  282. self.env['subscription.register'].create(values)
  283. certificat_email_template = self.env.ref('easy_my_coop.email_template_share_update', False)
  284. certificat_email_template.send_mail(rec.partner_id.id, False)