672 lines
32 KiB
Python
672 lines
32 KiB
Python
# -*- coding: utf-8 -*-
|
|
##############################################################################
|
|
#
|
|
# OpenERP, Open Source Management Solution
|
|
# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Affero General Public License as
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
# License, or (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU Affero General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
#
|
|
##############################################################################
|
|
from lxml import etree
|
|
import datetime
|
|
import time
|
|
|
|
from tools.translate import _
|
|
from osv import fields, osv
|
|
|
|
class project_task_type(osv.osv):
|
|
_name = 'project.task.type'
|
|
_description = 'Task Stage'
|
|
_columns = {
|
|
'name': fields.char('Stage Name', required=True, size=64, translate=True),
|
|
'description': fields.text('Description'),
|
|
'sequence': fields.integer('Sequence'),
|
|
}
|
|
_order = 'sequence'
|
|
|
|
_defaults = {
|
|
'sequence': 1
|
|
}
|
|
project_task_type()
|
|
|
|
class project(osv.osv):
|
|
_name = "project.project"
|
|
_description = "Project"
|
|
_inherits = {'account.analytic.account':"category_id"}
|
|
|
|
def search(self, cr, user, args, offset=0, limit=None, order=None, context=None, count=False):
|
|
if context is None:
|
|
context = {}
|
|
if user == 1:
|
|
return super(project, self).search(cr, user, args, offset=offset, limit=limit, order=order, context=context, count=count)
|
|
if context and context.has_key('user_prefence') and context['user_prefence']:
|
|
cr.execute("""SELECT project.id FROM project_project project
|
|
LEFT JOIN account_analytic_account account ON account.id = project.category_id
|
|
LEFT JOIN project_user_rel rel ON rel.project_id = project.category_id
|
|
WHERE (account.user_id = %s or rel.uid = %s)"""%(user, user))
|
|
res = cr.fetchall()
|
|
return [(r[0]) for r in res]
|
|
return super(project, self).search(cr, user, args, offset=offset, limit=limit, order=order,
|
|
context=context, count=count)
|
|
|
|
def _complete_name(self, cr, uid, ids, name, args, context=None):
|
|
if context is None:
|
|
context = {}
|
|
res = {}
|
|
for m in self.browse(cr, uid, ids, context=context):
|
|
res[m.id] = (m.parent_id and (m.parent_id.name + '/') or '') + m.name
|
|
return res
|
|
|
|
def check_recursion(self, cursor, user, ids, parent=None):
|
|
return super(project, self).check_recursion(cursor, user, ids,
|
|
parent=parent)
|
|
|
|
def onchange_partner_id(self, cr, uid, ids, part=False, context=None):
|
|
if context is None:
|
|
context = {}
|
|
partner_obj = self.pool.get('res.partner')
|
|
if not part:
|
|
return {'value':{'contact_id': False, 'pricelist_id': False}}
|
|
addr = partner_obj.address_get(cr, uid, [part], ['contact'])
|
|
pricelist = partner_obj.read(cr, uid, part, ['property_product_pricelist'], context=context)
|
|
pricelist_id = pricelist.get('property_product_pricelist', False) and pricelist.get('property_product_pricelist')[0] or False
|
|
return {'value':{'contact_id': addr['contact'], 'pricelist_id': pricelist_id}}
|
|
|
|
def _progress_rate(self, cr, uid, ids, names, arg, context=None):
|
|
if context is None:
|
|
context = {}
|
|
res = {}.fromkeys(ids, 0.0)
|
|
progress = {}
|
|
if not ids:
|
|
return res
|
|
ids2 = self.search(cr, uid, [('parent_id','child_of',ids)], context=context)
|
|
if ids2:
|
|
cr.execute('''SELECT
|
|
project_id, sum(planned_hours), sum(total_hours), sum(effective_hours)
|
|
FROM
|
|
project_task
|
|
WHERE
|
|
project_id IN %s AND
|
|
state<>'cancelled'
|
|
GROUP BY
|
|
project_id''',(tuple(ids2),))
|
|
progress = dict(map(lambda x: (x[0], (x[1], x[2], x[3])), cr.fetchall()))
|
|
for project in self.browse(cr, uid, ids, context=context):
|
|
s = [0.0, 0.0, 0.0]
|
|
tocompute = [project]
|
|
while tocompute:
|
|
p = tocompute.pop()
|
|
tocompute += p.child_ids
|
|
for i in range(3):
|
|
s[i] += progress.get(p.id, (0.0, 0.0, 0.0))[i]
|
|
res[project.id] = {
|
|
'planned_hours': s[0],
|
|
'effective_hours': s[2],
|
|
'total_hours': s[1],
|
|
'progress_rate': s[1] and (100.0 * s[2] / s[1]) or 0.0
|
|
}
|
|
return res
|
|
|
|
def unlink(self, cr, uid, ids, *args, **kwargs):
|
|
for proj in self.browse(cr, uid, ids):
|
|
if proj.tasks:
|
|
raise osv.except_osv(_('Operation Not Permitted !'), _('You can not delete a project with tasks. I suggest you to deactivate it.'))
|
|
return super(project, self).unlink(cr, uid, ids, *args, **kwargs)
|
|
|
|
_columns = {
|
|
'complete_name': fields.function(_complete_name, method=True, string="Project Name", type='char', size=250),
|
|
'active': fields.boolean('Active', help="If the active field is set to true, it will allow you to hide the project without removing it."),
|
|
'sequence': fields.integer('Sequence', help="Gives the sequence order when displaying a list of Projects."),
|
|
'category_id': fields.many2one('account.analytic.account', 'Analytic Account', help="Link this project to an analytic account if you need financial management on projects. It enables you to connect projects with budgets, planning, cost and revenue analysis, timesheets on projects, etc."),
|
|
'priority': fields.integer('Sequence', help="Gives the sequence order when displaying a list of task"),
|
|
'warn_manager': fields.boolean('Warn Manager', help="If you check this field, the project manager will receive a request each time a task is completed by his team."),
|
|
'members': fields.many2many('res.users', 'project_user_rel', 'project_id', 'uid', 'Project Members', help="Project's member. Not used in any computation, just for information purpose."),
|
|
'tasks': fields.one2many('project.task', 'project_id', "Project tasks"),
|
|
'planned_hours': fields.function(_progress_rate, multi="progress", method=True, string='Planned Time', help="Sum of planned hours of all tasks related to this project."),
|
|
'effective_hours': fields.function(_progress_rate, multi="progress", method=True, string='Time Spent', help="Sum of spent hours of all tasks related to this project."),
|
|
'total_hours': fields.function(_progress_rate, multi="progress", method=True, string='Total Time', help="Sum of total hours of all tasks related to this project."),
|
|
'progress_rate': fields.function(_progress_rate, multi="progress", method=True, string='Progress', type='float', help="Percent of tasks closed according to the total of tasks todo."),
|
|
'warn_customer': fields.boolean('Warn Partner', help="If you check this, the user will have a popup when closing a task that propose a message to send by email to the customer."),
|
|
'warn_header': fields.text('Mail Header', help="Header added at the beginning of the email for the warning message sent to the customer when a task is closed."),
|
|
'warn_footer': fields.text('Mail Footer', help="Footer added at the beginning of the email for the warning message sent to the customer when a task is closed."),
|
|
'type_ids': fields.many2many('project.task.type', 'project_task_type_rel', 'project_id', 'type_id', 'Tasks Stages'),
|
|
'project_escalation_id': fields.many2one('project.project','Project Escalation', help='If any issue is escalated from the current Project, it will be listed under the project selected here.'),
|
|
}
|
|
_order = "sequence"
|
|
_defaults = {
|
|
'active': True,
|
|
'priority': 1,
|
|
'sequence': 10,
|
|
'warn_manager': True,
|
|
}
|
|
def _check_dates(self, cr, uid, ids):
|
|
leave = self.read(cr, uid, ids[0], ['date_start', 'date'])
|
|
if leave['date_start'] and leave['date']:
|
|
if leave['date_start'] > leave['date']:
|
|
return False
|
|
return True
|
|
|
|
def _check_escalation(self, cr, uid, ids):
|
|
project_obj = self.browse(cr, uid, ids[0])
|
|
if project_obj.project_escalation_id:
|
|
if project_obj.project_escalation_id.id == project_obj.id:
|
|
return False
|
|
return True
|
|
|
|
_constraints = [
|
|
(_check_dates, 'Error! project start-date must be lower then project end-date.', ['date_start', 'date']),
|
|
(_check_escalation, 'Error! You cannot assign escalation to the same project!', ['project_escalation_id'])
|
|
]
|
|
def set_template(self, cr, uid, ids, context=None):
|
|
res = self.setActive(cr, uid, ids, value=False, context=context)
|
|
return res
|
|
|
|
def set_done(self, cr, uid, ids, context=None):
|
|
self.write(cr, uid, ids, {'state':'close'}, context=context)
|
|
return True
|
|
|
|
def set_cancel(self, cr, uid, ids, context=None):
|
|
self.write(cr, uid, ids, {'state':'cancelled'}, context=context)
|
|
return True
|
|
|
|
def set_pending(self, cr, uid, ids, context=None):
|
|
self.write(cr, uid, ids, {'state':'pending'}, context=context)
|
|
return True
|
|
|
|
def set_open(self, cr, uid, ids, context=None):
|
|
self.write(cr, uid, ids, {'state':'open'}, context=context)
|
|
return True
|
|
|
|
def reset_project(self, cr, uid, ids, context=None):
|
|
res = self.setActive(cr, uid, ids, value=True, context=context)
|
|
return res
|
|
|
|
def copy(self, cr, uid, id, default={}, context=None):
|
|
if context is None:
|
|
context = {}
|
|
proj = self.browse(cr, uid, id, context=context)
|
|
default = default or {}
|
|
context['active_test'] = False
|
|
default['state'] = 'open'
|
|
if not default.get('name', False):
|
|
default['name'] = proj.name+_(' (copy)')
|
|
res = super(project, self).copy(cr, uid, id, default, context)
|
|
ids = self.search(cr, uid, [('parent_id','child_of', [res])])
|
|
if ids:
|
|
cr.execute('update project_task set active=True where project_id IN %s',(tuple(ids),))
|
|
return res
|
|
|
|
def duplicate_template(self, cr, uid, ids, context=None):
|
|
if context is None:
|
|
context = {}
|
|
project_obj = self.pool.get('project.project')
|
|
data_obj = self.pool.get('ir.model.data')
|
|
task_obj = self.pool.get('project.task')
|
|
result = []
|
|
for proj in self.browse(cr, uid, ids, context=context):
|
|
parent_id = context.get('parent_id', False) # check me where to pass context for parent id ??
|
|
context.update({'analytic_project_copy': True})
|
|
new_id = project_obj.copy(cr, uid, proj.id, default = {
|
|
'name': proj.name +_(' (copy)'),
|
|
'state':'open',
|
|
'parent_id':parent_id}, context=context)
|
|
result.append(new_id)
|
|
cr.execute('select id from project_task where project_id=%s', (proj.id,))
|
|
res = cr.fetchall()
|
|
child_ids = self.search(cr, uid, [('parent_id','=', proj.category_id.id)], context=context)
|
|
parent_id = self.read(cr, uid, new_id, ['category_id'])['category_id'][0]
|
|
if child_ids:
|
|
self.duplicate_template(cr, uid, child_ids, context={'parent_id': parent_id})
|
|
|
|
if result and len(result):
|
|
res_id = result[0]
|
|
form_view_id = data_obj._get_id(cr, uid, 'project', 'edit_project')
|
|
form_view = data_obj.read(cr, uid, form_view_id, ['res_id'])
|
|
tree_view_id = data_obj._get_id(cr, uid, 'project', 'view_project_list')
|
|
tree_view = data_obj.read(cr, uid, tree_view_id, ['res_id'])
|
|
search_view_id = data_obj._get_id(cr, uid, 'project', 'view_project_project_filter')
|
|
search_view = data_obj.read(cr, uid, search_view_id, ['res_id'])
|
|
return {
|
|
'name': _('Projects'),
|
|
'view_type': 'form',
|
|
'view_mode': 'form,tree',
|
|
'res_model': 'project.project',
|
|
'view_id': False,
|
|
'res_id': res_id,
|
|
'views': [(form_view['res_id'],'form'),(tree_view['res_id'],'tree')],
|
|
'type': 'ir.actions.act_window',
|
|
'search_view_id': search_view['res_id'],
|
|
'nodestroy': True
|
|
}
|
|
# return result
|
|
|
|
# set active value for a project, its sub projects and its tasks
|
|
def setActive(self, cr, uid, ids, value=True, context=None):
|
|
if context is None:
|
|
context = {}
|
|
task_obj = self.pool.get('project.task')
|
|
for proj in self.browse(cr, uid, ids, context=None):
|
|
if context is None:
|
|
context = {}
|
|
self.write(cr, uid, [proj.id], {'state': value and 'open' or 'template'}, context)
|
|
cr.execute('select id from project_task where project_id=%s', (proj.id,))
|
|
tasks_id = [x[0] for x in cr.fetchall()]
|
|
if tasks_id:
|
|
task_obj.write(cr, uid, tasks_id, {'active': value}, context=context)
|
|
child_ids = self.search(cr, uid, [('parent_id','=', proj.id)])
|
|
if child_ids:
|
|
self.setActive(cr, uid, child_ids, value, context=None)
|
|
if context is None:
|
|
context = {}
|
|
return True
|
|
project()
|
|
|
|
class task(osv.osv):
|
|
_name = "project.task"
|
|
_description = "Task"
|
|
_log_create = True
|
|
_date_name = "date_start"
|
|
|
|
def _str_get(self, task, level=0, border='***', context=None):
|
|
if context is None:
|
|
context = {}
|
|
return border+' '+(task.user_id and task.user_id.name.upper() or '')+(level and (': L'+str(level)) or '')+(' - %.1fh / %.1fh'%(task.effective_hours or 0.0,task.planned_hours))+' '+border+'\n'+ \
|
|
border[0]+' '+(task.name or '')+'\n'+ \
|
|
(task.description or '')+'\n\n'
|
|
|
|
# Compute: effective_hours, total_hours, progress
|
|
def _hours_get(self, cr, uid, ids, field_names, args, context=None):
|
|
if context is None:
|
|
context = {}
|
|
res = {}
|
|
cr.execute("SELECT task_id, COALESCE(SUM(hours),0) FROM project_task_work WHERE task_id IN %s GROUP BY task_id",(tuple(ids),))
|
|
hours = dict(cr.fetchall())
|
|
for task in self.browse(cr, uid, ids, context=context):
|
|
res[task.id] = {'effective_hours': hours.get(task.id, 0.0), 'total_hours': task.remaining_hours + hours.get(task.id, 0.0)}
|
|
res[task.id]['progress'] = 0.0
|
|
if (task.remaining_hours + hours.get(task.id, 0.0)):
|
|
if task.state != 'done':
|
|
res[task.id]['progress'] = round(min(100.0 * hours.get(task.id, 0.0) / res[task.id]['total_hours'], 99.99), 2)
|
|
|
|
if task.state in ('done', 'cancel'):
|
|
res[task.id]['progress'] = 100.0
|
|
res[task.id]['delay_hours'] = res[task.id]['total_hours'] - task.planned_hours
|
|
return res
|
|
|
|
def onchange_planned(self, cr, uid, ids, planned = 0.0, effective = 0.0):
|
|
return {'value':{'remaining_hours': planned - effective}}
|
|
|
|
def _default_project(self, cr, uid, context=None):
|
|
if 'project_id' in context and context['project_id']:
|
|
return int(context['project_id'])
|
|
return False
|
|
|
|
#_sql_constraints = [
|
|
# ('remaining_hours', 'CHECK (remaining_hours>=0)', 'Please increase and review remaining hours ! It can not be smaller than 0.'),
|
|
#]
|
|
|
|
def copy_data(self, cr, uid, id, default={}, context=None):
|
|
if context is None:
|
|
context = {}
|
|
default = default or {}
|
|
default['work_ids'] = []
|
|
return super(task, self).copy_data(cr, uid, id, default, context)
|
|
|
|
def _check_dates(self, cr, uid, ids, context=None):
|
|
if context is None:
|
|
context = {}
|
|
task = self.read(cr, uid, ids[0], ['date_start', 'date_end'])
|
|
if task['date_start'] and task['date_end']:
|
|
if task['date_start'] > task['date_end']:
|
|
return False
|
|
return True
|
|
|
|
_columns = {
|
|
'active': fields.boolean('Active', help="If the active field is set to true, it will allow you to hide the task without removing it."),
|
|
'name': fields.char('Task Summary', size=128, required=True),
|
|
'description': fields.text('Description'),
|
|
'priority' : fields.selection([('4','Very Low'), ('3','Low'), ('2','Medium'), ('1','Urgent'), ('0','Very urgent')], 'Importance'),
|
|
'sequence': fields.integer('Sequence', help="Gives the sequence order when displaying a list of tasks."),
|
|
'type': fields.many2one('project.task.type', 'Stage',),
|
|
'state': fields.selection([('draft', 'Draft'),('open', 'In Progress'),('pending', 'Pending'), ('cancelled', 'Cancelled'), ('done', 'Done')], 'State', readonly=True, required=True,
|
|
help='If the task is created the state \'Draft\'.\n If the task is started, the state becomes \'In Progress\'.\n If review is needed the task is in \'Pending\' state.\
|
|
\n If the task is over, the states is set to \'Done\'.'),
|
|
'date_start': fields.datetime('Starting Date'),
|
|
'date_end': fields.datetime('Ending Date'),
|
|
'date_deadline': fields.date('Deadline'),
|
|
'project_id': fields.many2one('project.project', 'Project', ondelete='cascade',
|
|
help="If you have [?] in the project name, it means there are no analytic account linked to this project."),
|
|
'parent_ids': fields.many2many('project.task', 'project_task_parent_rel', 'task_id', 'parent_id', 'Parent Tasks'),
|
|
'child_ids': fields.many2many('project.task', 'project_task_parent_rel', 'parent_id', 'task_id', 'Delegated Tasks'),
|
|
'notes': fields.text('Notes'),
|
|
'planned_hours': fields.float('Planned Hours', required=True, help='Estimated time to do the task, usually set by the project manager when the task is in draft state.'),
|
|
'effective_hours': fields.function(_hours_get, method=True, string='Hours Spent', multi='hours', store=True, help="Computed using the sum of the task work done."),
|
|
'remaining_hours': fields.float('Remaining Hours', digits=(16,4), help="Total remaining time, can be re-estimated periodically by the assignee of the task."),
|
|
'total_hours': fields.function(_hours_get, method=True, string='Total Hours', multi='hours', store=True, help="Computed as: Time Spent + Remaining Time."),
|
|
'progress': fields.function(_hours_get, method=True, string='Progress (%)', multi='hours', store=True, help="Computed as: Time Spent / Total Time."),
|
|
'delay_hours': fields.function(_hours_get, method=True, string='Delay Hours', multi='hours', store=True, help="Computed as difference of the time estimated by the project manager and the real time to close the task."),
|
|
|
|
'user_id': fields.many2one('res.users', 'Assigned to'),
|
|
'delegated_user_id': fields.related('child_ids', 'user_id', type='many2one', relation='res.users', string='Delegated To'),
|
|
'partner_id': fields.many2one('res.partner', 'Partner'),
|
|
'work_ids': fields.one2many('project.task.work', 'task_id', 'Work done'),
|
|
'manager_id': fields.related('project_id', 'category_id', 'user_id', type='many2one', relation='res.users', string='Project Manager'),
|
|
'company_id': fields.many2one('res.company', 'Company'),
|
|
}
|
|
_defaults = {
|
|
'state': 'draft',
|
|
'priority': '2',
|
|
'progress': 0,
|
|
'sequence': 10,
|
|
'active': True,
|
|
'date_start': time.strftime('%Y-%m-%d %H:%M:%S'),
|
|
'project_id': _default_project,
|
|
'company_id': lambda self, cr, uid, c: self.pool.get('res.company')._company_default_get(cr, uid, 'project.task', context=c)
|
|
}
|
|
_order = "sequence, priority, date_start, id"
|
|
|
|
_constraints = [
|
|
(_check_dates, 'Error! task start-date must be lower then task end-date.', ['date_start', 'date_end'])
|
|
]
|
|
#
|
|
# Override view according to the company definition
|
|
#
|
|
def fields_view_get(self, cr, uid, view_id=None, view_type='form', context=None, toolbar=False, submenu=False):
|
|
users_obj = self.pool.get('res.users')
|
|
if context is None:
|
|
context = {}
|
|
obj_tm = users_obj.browse(cr, uid, uid, context).company_id.project_time_mode_id
|
|
tm = obj_tm and obj_tm.name or 'Hours'
|
|
|
|
res = super(task, self).fields_view_get(cr, uid, view_id, view_type, context, toolbar, submenu=submenu)
|
|
|
|
if tm in ['Hours','Hour']:
|
|
return res
|
|
|
|
eview = etree.fromstring(res['arch'])
|
|
|
|
def _check_rec(eview):
|
|
if eview.attrib.get('widget','') == 'float_time':
|
|
eview.set('widget','float')
|
|
for child in eview:
|
|
_check_rec(child)
|
|
return True
|
|
|
|
_check_rec(eview)
|
|
|
|
res['arch'] = etree.tostring(eview)
|
|
|
|
for f in res['fields']:
|
|
if 'Hours' in res['fields'][f]['string']:
|
|
res['fields'][f]['string'] = res['fields'][f]['string'].replace('Hours',tm)
|
|
|
|
return res
|
|
|
|
def do_close(self, cr, uid, ids, *args):
|
|
mail_send = False
|
|
mod_obj = self.pool.get('ir.model.data')
|
|
request = self.pool.get('res.request')
|
|
tasks = self.browse(cr, uid, ids)
|
|
for task in tasks:
|
|
project = task.project_id
|
|
if project:
|
|
if project.warn_manager and project.user_id and (project.user_id.id != uid):
|
|
request.create(cr, uid, {
|
|
'name': _("Task '%s' closed") % task.name,
|
|
'state': 'waiting',
|
|
'act_from': uid,
|
|
'act_to': project.user_id.id,
|
|
'ref_partner_id': task.partner_id.id,
|
|
'ref_doc1': 'project.task,%d'% (task.id,),
|
|
'ref_doc2': 'project.project,%d'% (project.id,),
|
|
})
|
|
elif project.warn_manager:
|
|
task_id = ids[0]
|
|
mail_send = True
|
|
self.write(cr, uid, [task.id], {'state': 'done', 'date_end':time.strftime('%Y-%m-%d %H:%M:%S'), 'remaining_hours': 0.0})
|
|
for parent_id in task.parent_ids:
|
|
if parent_id.state in ('pending','draft'):
|
|
reopen = True
|
|
for child in parent_id.child_ids:
|
|
if child.id != task.id and child.state not in ('done','cancelled'):
|
|
reopen = False
|
|
if reopen:
|
|
self.do_reopen(cr, uid, [parent_id.id])
|
|
if mail_send:
|
|
model_data_ids = mod_obj.search(cr,uid,[('model','=','ir.ui.view'),('name','=','view_project_close_task')])
|
|
resource_id = mod_obj.read(cr, uid, model_data_ids, fields=['res_id'])[0]['res_id']
|
|
args[0].update({'task_id': task_id})
|
|
return {
|
|
'name': _('Email Send to Customer'),
|
|
'view_type': 'form',
|
|
'context': args[0], # improve me
|
|
'view_mode': 'tree,form',
|
|
'res_model': 'close.task',
|
|
'views': [(resource_id,'form')],
|
|
'type': 'ir.actions.act_window',
|
|
'target': 'new',
|
|
'nodestroy': True
|
|
}
|
|
return True
|
|
|
|
def do_reopen(self, cr, uid, ids, *args):
|
|
request = self.pool.get('res.request')
|
|
tasks = self.browse(cr, uid, ids)
|
|
for task in tasks:
|
|
project = task.project_id
|
|
if project and project.warn_manager and project.user_id.id and (project.user_id.id != uid):
|
|
request.create(cr, uid, {
|
|
'name': _("Task '%s' set in progress") % task.name,
|
|
'state': 'waiting',
|
|
'act_from': uid,
|
|
'act_to': project.user_id.id,
|
|
'ref_partner_id': task.partner_id.id,
|
|
'ref_doc1': 'project.task,%d' % task.id,
|
|
'ref_doc2': 'project.project,%d' % project.id,
|
|
})
|
|
|
|
self.write(cr, uid, [task.id], {'state': 'open'})
|
|
return True
|
|
|
|
def do_cancel(self, cr, uid, ids, *args):
|
|
request = self.pool.get('res.request')
|
|
tasks = self.browse(cr, uid, ids)
|
|
for task in tasks:
|
|
project = task.project_id
|
|
if project.warn_manager and project.user_id and (project.user_id.id != uid):
|
|
request.create(cr, uid, {
|
|
'name': _("Task '%s' cancelled") % task.name,
|
|
'state': 'waiting',
|
|
'act_from': uid,
|
|
'act_to': project.user_id.id,
|
|
'ref_partner_id': task.partner_id.id,
|
|
'ref_doc1': 'project.task,%d' % task.id,
|
|
'ref_doc2': 'project.project,%d' % project.id,
|
|
})
|
|
self.write(cr, uid, [task.id], {'state': 'cancelled', 'remaining_hours':0.0})
|
|
return True
|
|
|
|
def do_open(self, cr, uid, ids, *args):
|
|
tasks= self.browse(cr,uid,ids)
|
|
for t in tasks:
|
|
self.write(cr, uid, [t.id], {'state': 'open'})
|
|
return True
|
|
|
|
def do_draft(self, cr, uid, ids, *args):
|
|
self.write(cr, uid, ids, {'state': 'draft'})
|
|
return True
|
|
|
|
|
|
def do_pending(self, cr, uid, ids, *args):
|
|
self.write(cr, uid, ids, {'state': 'pending'})
|
|
return True
|
|
|
|
def next_type(self, cr, uid, ids, *args):
|
|
for typ in self.browse(cr, uid, ids):
|
|
typeid = typ.type.id
|
|
types = map(lambda x:x.id, typ.project_id.type_ids or [])
|
|
if types:
|
|
if not typeid:
|
|
self.write(cr, uid, typ.id, {'type': types[0]})
|
|
elif typeid and typeid in types and types.index(typeid) != len(types)-1 :
|
|
index = types.index(typeid)
|
|
self.write(cr, uid, typ.id, {'type': types[index+1]})
|
|
return True
|
|
|
|
def prev_type(self, cr, uid, ids, *args):
|
|
for typ in self.browse(cr, uid, ids):
|
|
typeid = typ.type.id
|
|
types = map(lambda x:x.id, typ.project_id.type_ids)
|
|
if types:
|
|
if typeid and typeid in types:
|
|
index = types.index(typeid)
|
|
self.write(cr, uid, typ.id, {'type': index and types[index-1] or False})
|
|
return True
|
|
|
|
task()
|
|
|
|
class project_work(osv.osv):
|
|
_name = "project.task.work"
|
|
_description = "Task Work"
|
|
_columns = {
|
|
'name': fields.char('Work summary', size=128),
|
|
'date': fields.datetime('Date'),
|
|
'task_id': fields.many2one('project.task', 'Task', ondelete='cascade', required=True),
|
|
'hours': fields.float('Time Spent'),
|
|
'user_id': fields.many2one('res.users', 'Done by', required=True),
|
|
'company_id': fields.related('task_id', 'company_id', type='many2one', relation='res.company', string='Company', store=True)
|
|
}
|
|
_defaults = {
|
|
'user_id': lambda obj, cr, uid, context: uid,
|
|
'date': time.strftime('%Y-%m-%d %H:%M:%S')
|
|
}
|
|
_order = "date desc"
|
|
def create(self, cr, uid, vals, *args, **kwargs):
|
|
if 'hours' in vals and (not vals['hours']):
|
|
vals['hours'] = 0.00
|
|
if 'task_id' in vals:
|
|
cr.execute('update project_task set remaining_hours=remaining_hours - %s where id=%s', (vals.get('hours',0.0), vals['task_id']))
|
|
return super(project_work,self).create(cr, uid, vals, *args, **kwargs)
|
|
|
|
def write(self, cr, uid, ids, vals, context=None):
|
|
if context is None:
|
|
context = {}
|
|
if 'hours' in vals and (not vals['hours']):
|
|
vals['hours'] = 0.00
|
|
if 'hours' in vals:
|
|
for work in self.browse(cr, uid, ids, context):
|
|
cr.execute('update project_task set remaining_hours=remaining_hours - %s + (%s) where id=%s', (vals.get('hours',0.0), work.hours, work.task_id.id))
|
|
return super(project_work,self).write(cr, uid, ids, vals, context)
|
|
|
|
def unlink(self, cr, uid, ids, *args, **kwargs):
|
|
for work in self.browse(cr, uid, ids):
|
|
cr.execute('update project_task set remaining_hours=remaining_hours + %s where id=%s', (work.hours, work.task_id.id))
|
|
return super(project_work,self).unlink(cr, uid, ids, *args, **kwargs)
|
|
project_work()
|
|
|
|
class config_compute_remaining(osv.osv_memory):
|
|
_name='config.compute.remaining'
|
|
def _get_remaining(self,cr, uid, ctx):
|
|
task_obj = self.pool.get('project.task')
|
|
if 'active_id' in ctx:
|
|
return task_obj.browse(cr, uid, ctx['active_id'], context=ctx).remaining_hours
|
|
return False
|
|
|
|
_columns = {
|
|
'remaining_hours' : fields.float('Remaining Hours', digits=(16,2), help="Put here the remaining hours required to close the task."),
|
|
}
|
|
|
|
_defaults = {
|
|
'remaining_hours': _get_remaining
|
|
}
|
|
|
|
def compute_hours(self, cr, uid, ids, context=None):
|
|
if context is None:
|
|
context = {}
|
|
task_obj = self.pool.get('project.task')
|
|
request = self.pool.get('res.request')
|
|
if 'active_id' in context:
|
|
remaining_hrs=self.browse(cr,uid,ids)[0].remaining_hours
|
|
task_obj.write(cr,uid,context['active_id'],{'remaining_hours':remaining_hrs})
|
|
if context.get('button_reactivate', False):
|
|
tasks = task_obj.browse(cr, uid, [context['active_id']], context=context)
|
|
for task in tasks:
|
|
project = task.project_id
|
|
if project and project.warn_manager and project.user_id.id and (project.user_id.id != uid):
|
|
request.create(cr, uid, {
|
|
'name': _("Task '%s' set in progress") % task.name,
|
|
'state': 'waiting',
|
|
'act_from': uid,
|
|
'act_to': project.user_id.id,
|
|
'ref_partner_id': task.partner_id.id,
|
|
'ref_doc1': 'project.task,%d' % task.id,
|
|
'ref_doc2': 'project.project,%d' % project.id,
|
|
})
|
|
task_obj.write(cr, uid, [task.id], {'state': 'open'})
|
|
return {
|
|
'type': 'ir.actions.act_window_close',
|
|
}
|
|
|
|
config_compute_remaining()
|
|
|
|
class message(osv.osv):
|
|
_name = "project.message"
|
|
_description = "Message"
|
|
_columns = {
|
|
'subject': fields.char('Subject', size=128, required="True"),
|
|
'project_id': fields.many2one('project.project', 'Project', ondelete='cascade'),
|
|
'date': fields.date('Date', required=1),
|
|
'user_id': fields.many2one('res.users', 'User', required="True"),
|
|
'description': fields.text('Description'),
|
|
}
|
|
|
|
def _default_project(self, cr, uid, context=None):
|
|
if context is None:
|
|
context = {}
|
|
if 'project_id' in context and context['project_id']:
|
|
return int(context['project_id'])
|
|
return False
|
|
|
|
_defaults = {
|
|
'user_id': lambda self,cr,uid,ctx: uid,
|
|
'date': lambda self,cr,uid,ctx: time.strftime('%Y-%m-%d'),
|
|
'project_id': _default_project
|
|
}
|
|
|
|
message()
|
|
|
|
class users(osv.osv):
|
|
_inherit = 'res.users'
|
|
_description = "Users"
|
|
_columns = {
|
|
'context_project_id': fields.many2one('project.project', 'Project')
|
|
}
|
|
users()
|
|
|
|
class account_analytic_account(osv.osv):
|
|
|
|
_inherit = 'account.analytic.account'
|
|
_description = 'Analytic Account'
|
|
|
|
def create(self, cr, uid, vals, context=None):
|
|
if context is None:
|
|
context = {}
|
|
if vals.get('child_ids', False) and context.get('analytic_project_copy', False):
|
|
vals['child_ids'] = []
|
|
return super(account_analytic_account, self).create(cr, uid, vals, context=context)
|
|
|
|
account_analytic_account()
|
|
|
|
# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
|