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.

449 lines
13KB

  1. # Copyright 2016 上海开阖软件有限公司 (http://www.osbzr.com)
  2. # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
  3. from odoo import api, fields, models
  4. from odoo.tools import float_is_zero
  5. from odoo.exceptions import UserError
  6. # 状态可选值
  7. TASK_STATES = [
  8. ('todo', '新建'),
  9. ('doing', '正在进行'),
  10. ('done', '已完成'),
  11. ('cancel', '已取消'),
  12. ]
  13. AVAILABLE_PRIORITIES = [
  14. ('0', '一般'),
  15. ('1', '低'),
  16. ('2', '中'),
  17. ('3', '高'),
  18. ]
  19. class Project(models.Model):
  20. _name = 'project'
  21. _description = '项目'
  22. _inherits = {'auxiliary.financing': 'auxiliary_id'}
  23. _inherit = ['mail.thread']
  24. @api.depends('task_ids.hours')
  25. def _compute_hours(self):
  26. '''计算项目的实际工时'''
  27. for Project in self:
  28. for Task in Project.task_ids:
  29. Project.hours += Task.hours
  30. type_id = fields.Many2one(
  31. string='分类',
  32. comodel_name='core.value',
  33. ondelete='restrict',
  34. domain=[('type', '=', 'project_type')],
  35. context={'type': 'project_type'},
  36. )
  37. auxiliary_id = fields.Many2one(
  38. string='辅助核算',
  39. comodel_name='auxiliary.financing',
  40. ondelete='cascade',
  41. required=True,
  42. )
  43. task_ids = fields.One2many(
  44. string='任务',
  45. comodel_name='task',
  46. inverse_name='project_id',
  47. )
  48. customer_id = fields.Many2one(
  49. string='客户',
  50. comodel_name='partner',
  51. ondelete='restrict',
  52. )
  53. invoice_ids = fields.One2many(
  54. string='发票行',
  55. comodel_name='project.invoice',
  56. inverse_name='project_id',
  57. )
  58. plan_hours = fields.Float('计划工时')
  59. hours = fields.Float('实际工时',
  60. compute=_compute_hours,
  61. store=True)
  62. address = fields.Char('地址')
  63. note = fields.Text('备注')
  64. active = fields.Boolean('启用', default=True)
  65. class ProjectInvoice(models.Model):
  66. _name = 'project.invoice'
  67. _description = '项目的发票'
  68. @api.depends('tax_rate', 'amount')
  69. def _compute_tax_amount(self):
  70. '''计算税额'''
  71. self.ensure_one()
  72. if self.tax_rate > 100:
  73. raise UserError('税率不能输入超过100的数\n当前税率:%s' % self.tax_rate)
  74. if self.tax_rate < 0:
  75. raise UserError('税率不能输入负数\n当前税率:%s' % self.tax_rate)
  76. self.tax_amount = self.amount / (100 + self.tax_rate) * self.tax_rate
  77. project_id = fields.Many2one(
  78. string='项目',
  79. comodel_name='project',
  80. ondelete='cascade',
  81. )
  82. tax_rate = fields.Float(
  83. string='税率',
  84. default=lambda self: self.env.user.company_id.output_tax_rate,
  85. help='默认值取公司销项税率',
  86. )
  87. tax_amount = fields.Float(
  88. string='税额',
  89. compute=_compute_tax_amount,
  90. )
  91. amount = fields.Float(
  92. string='含税金额',
  93. help='含税金额',
  94. )
  95. date_due = fields.Date(
  96. string='到期日',
  97. default=lambda self: fields.Date.context_today(self),
  98. required=True,
  99. help='收款截止日期',
  100. )
  101. invoice_id = fields.Many2one(
  102. string='发票号',
  103. comodel_name='money.invoice',
  104. readonly=True,
  105. copy=False,
  106. ondelete='set null',
  107. help='产生的发票号',
  108. )
  109. company_id = fields.Many2one(
  110. 'res.company',
  111. string='公司',
  112. change_default=True,
  113. default=lambda self: self.env.company
  114. )
  115. def _get_invoice_vals(self, category_id, project_id, amount, tax_amount):
  116. '''返回创建 money_invoice 时所需数据'''
  117. return {
  118. 'name': project_id.name,
  119. 'partner_id': project_id.customer_id and project_id.customer_id.id,
  120. 'category_id': category_id.id,
  121. 'auxiliary_id': project_id.auxiliary_id.id,
  122. 'date' : fields.Date.context_today(self),
  123. 'amount': amount,
  124. 'reconciled': 0,
  125. 'to_reconcile': amount,
  126. 'tax_amount': tax_amount,
  127. 'date_due': self.date_due,
  128. 'state': 'draft',
  129. }
  130. def make_invoice(self):
  131. '''生成结算单'''
  132. for line in self:
  133. invoice_id = False
  134. if not line.project_id.customer_id:
  135. raise UserError('生成发票前请输入客户')
  136. category = self.env.ref('money.core_category_sale')
  137. if not float_is_zero(self.amount, 2):
  138. invoice_id = self.env['money.invoice'].create(
  139. self._get_invoice_vals(
  140. category, line.project_id, line.amount, line.tax_amount)
  141. )
  142. line.invoice_id = invoice_id.id
  143. return invoice_id
  144. class Task(models.Model):
  145. _name = 'task'
  146. _description = '任务'
  147. _inherit = ['mail.thread']
  148. _order = 'sequence, priority desc, id'
  149. @api.depends('timeline_ids.hours')
  150. def _compute_hours(self):
  151. """计算任务的实际工时"""
  152. for Task in self:
  153. for line in Task.timeline_ids:
  154. Task.hours += line.hours
  155. def _default_status_impl(self):
  156. """任务阶段默认值的实现方法"""
  157. status_id = self.env['task.status'].search(
  158. [('state', '=', 'todo')])
  159. return status_id and status_id[0]
  160. @api.model
  161. def _default_status(self):
  162. '''创建任务时,任务阶段默认为todo状态的阶段'''
  163. return self._default_status_impl()
  164. @api.depends('project_id.type_id')
  165. def _compute_project_type_id(self):
  166. for line in self:
  167. line.project_type_id = False
  168. if line.project_id:
  169. line.project_type_id = self.project_id.type_id
  170. name = fields.Char(
  171. string='名称',
  172. required=True,
  173. )
  174. user_id = fields.Many2one(
  175. string='指派给',
  176. comodel_name='res.users',
  177. )
  178. project_id = fields.Many2one(
  179. string='项目',
  180. comodel_name='project',
  181. ondelete='cascade',
  182. )
  183. project_type_id = fields.Many2one(
  184. string='分类',
  185. comodel_name='core.value',
  186. ondelete='restrict',
  187. compute=_compute_project_type_id,
  188. store=True)
  189. timeline_ids = fields.One2many(
  190. string='工作记录',
  191. comodel_name='timeline',
  192. inverse_name='task_id',
  193. )
  194. next_action = fields.Char(
  195. string='下一步计划',
  196. required=False,
  197. help='针对此任务下一步的计划',
  198. )
  199. next_datetime = fields.Datetime(
  200. string='下一步计划时间',
  201. help='下一步计划预计开始的时间',
  202. )
  203. status = fields.Many2one(
  204. 'task.status',
  205. string='状态',
  206. default=_default_status,
  207. ondelete='restrict',
  208. domain="['|', ('project_type_id', '=', project_type_id), ('state','!=','doing')]",
  209. )
  210. plan_hours = fields.Float('计划工时')
  211. hours = fields.Float('实际工时',
  212. compute=_compute_hours,
  213. store=True)
  214. sequence = fields.Integer('顺序')
  215. is_schedule = fields.Boolean('列入计划')
  216. note = fields.Text('描述')
  217. priority = fields.Selection(AVAILABLE_PRIORITIES,
  218. string='优先级',
  219. default=AVAILABLE_PRIORITIES[0][0])
  220. color = fields.Integer('Color Index',
  221. default=0)
  222. company_id = fields.Many2one(
  223. 'res.company',
  224. string='公司',
  225. change_default=True,
  226. default=lambda self: self.env.company
  227. )
  228. tag_ids = fields.Many2many('core.value',
  229. ondelete='restrict',
  230. string='标签',
  231. domain=[('type', '=', 'task_tag')],
  232. context={'type': 'task_tag'})
  233. def assign_to_me(self):
  234. '''将任务指派给自己,并修改状态'''
  235. self.ensure_one()
  236. next_status = self.env['task.status'].search([('state', '=', 'doing')])
  237. self.user_id = self.env.user
  238. self.status = next_status and next_status[0]
  239. class TaskStatus(models.Model):
  240. _name = 'task.status'
  241. _description = '任务阶段'
  242. _order = 'sequence, id'
  243. name = fields.Char('名称', required=True)
  244. project_type_id = fields.Many2one(
  245. string='项目分类',
  246. comodel_name='core.value',
  247. ondelete='restrict',
  248. domain=[('type', '=', 'project_type')],
  249. )
  250. state = fields.Selection(TASK_STATES,
  251. string='任务状态',
  252. index=True,
  253. required=True,
  254. default='doing')
  255. sequence = fields.Integer('顺序', default=10)
  256. company_id = fields.Many2one(
  257. 'res.company',
  258. string='公司',
  259. change_default=True,
  260. default=lambda self: self.env.company)
  261. class Timesheet(models.Model):
  262. _name = 'timesheet'
  263. _description = '今日工作日志'
  264. date = fields.Date(
  265. string='日期',
  266. required=True,
  267. readonly=True,
  268. default=fields.Date.context_today)
  269. user_id = fields.Many2one(
  270. string='用户',
  271. required=True,
  272. readonly=True,
  273. default=lambda self: self.env.user,
  274. comodel_name='res.users',
  275. )
  276. timeline_ids = fields.One2many(
  277. string='工作记录',
  278. comodel_name='timeline',
  279. inverse_name='timesheet_id',
  280. )
  281. task_ids = fields.Many2many(
  282. string='待办事项',
  283. required=False,
  284. readonly=False,
  285. default=lambda self: [(4, t.id) for t in self.env['task'].search(
  286. [('user_id', '=', self.env.user.id),
  287. ('status.state', '=', 'doing')])],
  288. help=False,
  289. comodel_name='task',
  290. domain=[],
  291. context={}
  292. )
  293. company_id = fields.Many2one(
  294. 'res.company',
  295. string='公司',
  296. change_default=True,
  297. default=lambda self: self.env.company
  298. )
  299. color = fields.Integer('Color Index',
  300. default=0)
  301. _sql_constraints = [
  302. ('user_uniq', 'unique(user_id,date)', '同一个人一天只能创建一个工作日志')
  303. ]
  304. def name_get(self):
  305. ret = []
  306. for s in self:
  307. ret.append((s.id, '%s %s' % (s.user_id.name, s.date)))
  308. return ret
  309. class Timeline(models.Model):
  310. _name = 'timeline'
  311. _description = '工作记录'
  312. timesheet_id = fields.Many2one(
  313. string='记录表',
  314. comodel_name='timesheet',
  315. ondelete='cascade',
  316. )
  317. task_id = fields.Many2one(
  318. string='任务',
  319. required=True,
  320. readonly=False,
  321. comodel_name='task',
  322. )
  323. project_id = fields.Many2one(
  324. string='项目',
  325. related='task_id.project_id',
  326. store=True,
  327. ondelete='cascade',
  328. )
  329. user_id = fields.Many2one(
  330. string='指派给',
  331. comodel_name='res.users',
  332. )
  333. start_time = fields.Datetime('开始时间', default=fields.Datetime.now)
  334. end_time = fields.Datetime('结束时间', default=fields.Datetime.now)
  335. hours = fields.Float(
  336. string='小时数',
  337. default=0.5,
  338. digits=(16, 1),
  339. help='实际花的小时数',
  340. )
  341. just_done = fields.Text(
  342. string='具体工作内容',
  343. required=True,
  344. help='在此时长内针对此任务实际完成的工作内容',
  345. )
  346. need_help = fields.Char(
  347. string='需要的帮助',
  348. )
  349. # TODO 以下三个字段用于更新task的同名字段
  350. next_action = fields.Char(
  351. string='下一步计划',
  352. required=False,
  353. help='针对此任务下一步的计划',
  354. )
  355. next_datetime = fields.Datetime(
  356. string='下一步计划时间',
  357. help='下一步计划预计开始的时间',
  358. )
  359. set_status = fields.Many2one('task.status',
  360. string='状态更新到')
  361. company_id = fields.Many2one(
  362. 'res.company',
  363. string='公司',
  364. change_default=True,
  365. default=lambda self: self.env.company
  366. )
  367. color = fields.Integer('Color Index',
  368. default=0)
  369. @api.model_create_multi
  370. def create(self, vals_list):
  371. '''创建工作记录时,更新对应task的status等字段'''
  372. res = super(Timeline, self).create(vals_list)
  373. for vals in vals_list:
  374. set_status = vals.get('set_status')
  375. task_id = vals.get('task_id')
  376. next_action = vals.get('next_action')
  377. next_datetime = vals.get('next_datetime')
  378. user_id = vals.get('user_id')
  379. if task_id:
  380. Task = self.env['task'].search([('id', '=', task_id)])
  381. if set_status:
  382. Task.write({'status': set_status})
  383. if next_action:
  384. Task.write({'next_action': next_action})
  385. if next_datetime:
  386. Task.write({'next_datetime': next_datetime})
  387. if user_id:
  388. Task.write({'user_id': user_id})
  389. return res
上海开阖软件有限公司 沪ICP备12045867号-1