1 # -*- coding: utf-8 -*-
2 ##############################################################################
4 # OpenERP, Open Source Management Solution
5 # Copyright (C) 2004-2009 Tiny SPRL (<http://tiny.be>).
7 # This program is free software: you can redistribute it and/or modify
8 # it under the terms of the GNU Affero General Public License as
9 # published by the Free Software Foundation, either version 3 of the
10 # License, or (at your option) any later version.
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU Affero General Public License for more details.
17 # You should have received a copy of the GNU Affero General Public License
18 # along with this program. If not, see <http://www.gnu.org/licenses/>.
20 ##############################################################################
23 from datetime import datetime, timedelta
25 from osv import fields, osv
31 from tools.translate import _
32 from crm import wizard
34 wizard.mail_compose_message.SUPPORTED_MODELS.append('hr.applicant')
38 ('open', 'In Progress'),
39 ('cancel', 'Refused'),
41 ('pending', 'Pending')
44 AVAILABLE_PRIORITIES = [
53 class hr_recruitment_source(osv.osv):
54 """ Sources of HR Recruitment """
55 _name = "hr.recruitment.source"
56 _description = "Source of Applicants"
58 'name': fields.char('Source Name', size=64, required=True, translate=True),
60 hr_recruitment_source()
63 class hr_recruitment_stage(osv.osv):
64 """ Stage of HR Recruitment """
65 _name = "hr.recruitment.stage"
66 _description = "Stage of Recruitment"
69 'name': fields.char('Name', size=64, required=True, translate=True),
70 'sequence': fields.integer('Sequence', help="Gives the sequence order when displaying a list of stages."),
71 'department_id':fields.many2one('hr.department', 'Specific to a Department', help="Stages of the recruitment process may be different per department. If this stage is common to all departments, keep tempy this field."),
72 'requirements': fields.text('Requirements')
77 hr_recruitment_stage()
79 class hr_recruitment_degree(osv.osv):
80 """ Degree of HR Recruitment """
81 _name = "hr.recruitment.degree"
82 _description = "Degree of Recruitment"
84 'name': fields.char('Name', size=64, required=True, translate=True),
85 'sequence': fields.integer('Sequence', help="Gives the sequence order when displaying a list of degrees."),
91 ('name_uniq', 'unique (name)', 'The name of the Degree of Recruitment must be unique!')
93 hr_recruitment_degree()
95 class hr_applicant(crm.crm_case, osv.osv):
96 _name = "hr.applicant"
97 _description = "Applicant"
99 _inherit = ['ir.needaction_mixin', 'mail.thread']
101 def _compute_day(self, cr, uid, ids, fields, args, context=None):
103 @param cr: the current row, from the database cursor,
104 @param uid: the current user’s ID for security checks,
105 @param ids: List of Openday’s IDs
106 @return: difference between current date and log date
107 @param context: A standard dictionary for contextual values
110 for issue in self.browse(cr, uid, ids, context=context):
117 if field in ['day_open']:
119 date_create = datetime.strptime(issue.create_date, "%Y-%m-%d %H:%M:%S")
120 date_open = datetime.strptime(issue.date_open, "%Y-%m-%d %H:%M:%S")
121 ans = date_open - date_create
122 date_until = issue.date_open
124 elif field in ['day_close']:
125 if issue.date_closed:
126 date_create = datetime.strptime(issue.create_date, "%Y-%m-%d %H:%M:%S")
127 date_close = datetime.strptime(issue.date_closed, "%Y-%m-%d %H:%M:%S")
128 date_until = issue.date_closed
129 ans = date_close - date_create
131 duration = float(ans.days)
132 res[issue.id][field] = abs(float(duration))
136 'name': fields.char('Name', size=128, required=True),
137 'message_ids': fields.one2many('mail.message', 'res_id', 'Messages', domain=[('model','=',_name)]),
138 'active': fields.boolean('Active', help="If the active field is set to false, it will allow you to hide the case without removing it."),
139 'description': fields.text('Description'),
140 'email_from': fields.char('Email', size=128, help="These people will receive email."),
141 'email_cc': fields.text('Watchers Emails', size=252, help="These email addresses will be added to the CC field of all inbound and outbound emails for this record before being sent. Separate multiple email addresses with a comma"),
142 'probability': fields.float('Probability'),
143 'partner_id': fields.many2one('res.partner', 'Partner'),
144 'create_date': fields.datetime('Creation Date', readonly=True, select=True),
145 'write_date': fields.datetime('Update Date', readonly=True),
146 'stage_id': fields.many2one ('hr.recruitment.stage', 'Stage'),
147 'state': fields.selection(AVAILABLE_STATES, 'State', size=16, readonly=True,
148 help='The state is set to \'Draft\', when a case is created.\
149 \nIf the case is in progress the state is set to \'Open\'.\
150 \nWhen the case is over, the state is set to \'Done\'.\
151 \nIf the case needs to be reviewed then the state is set to \'Pending\'.'),
152 'company_id': fields.many2one('res.company', 'Company'),
153 'user_id': fields.many2one('res.users', 'Responsible'),
155 'date_closed': fields.datetime('Closed', readonly=True, select=True),
156 'date_open': fields.datetime('Opened', readonly=True, select=True),
157 'date': fields.datetime('Date'),
158 'date_action': fields.date('Next Action Date'),
159 'title_action': fields.char('Next Action', size=64),
160 'priority': fields.selection(AVAILABLE_PRIORITIES, 'Appreciation'),
161 'job_id': fields.many2one('hr.job', 'Applied Job'),
162 'salary_proposed_extra': fields.char('Proposed Salary Extra', size=100, help="Salary Proposed by the Organisation, extra advantages"),
163 'salary_expected_extra': fields.char('Expected Salary Extra', size=100, help="Salary Expected by Applicant, extra advantages"),
164 'salary_proposed': fields.float('Proposed Salary', help="Salary Proposed by the Organisation"),
165 'salary_expected': fields.float('Expected Salary', help="Salary Expected by Applicant"),
166 'availability': fields.integer('Availability (Days)'),
167 'partner_name': fields.char("Applicant's Name", size=64),
168 'partner_phone': fields.char('Phone', size=32),
169 'partner_mobile': fields.char('Mobile', size=32),
170 'type_id': fields.many2one('hr.recruitment.degree', 'Degree'),
171 'department_id': fields.many2one('hr.department', 'Department'),
172 'state': fields.selection(AVAILABLE_STATES, 'State', size=16, readonly=True),
173 'survey': fields.related('job_id', 'survey_id', type='many2one', relation='survey', string='Survey'),
174 'response': fields.integer("Response"),
175 'reference': fields.char('Refered By', size=128),
176 'source_id': fields.many2one('hr.recruitment.source', 'Source'),
177 'day_open': fields.function(_compute_day, string='Days to Open', \
178 multi='day_open', type="float", store=True),
179 'day_close': fields.function(_compute_day, string='Days to Close', \
180 multi='day_close', type="float", store=True),
181 'color': fields.integer('Color Index'),
182 'emp_id': fields.many2one('hr.employee', 'employee'),
183 'user_email': fields.related('user_id', 'user_email', type='char', string='User Email', readonly=True),
187 'active': lambda *a: 1,
188 'user_id': lambda self, cr, uid, context: uid,
189 'email_from': crm.crm_case. _get_default_email,
190 'state': lambda *a: 'draft',
191 'priority': lambda *a: '',
192 'company_id': lambda s, cr, uid, c: s.pool.get('res.company')._company_default_get(cr, uid, 'crm.helpdesk', context=c),
196 def _read_group_stage_ids(self, cr, uid, ids, domain, read_group_order=None, access_rights_uid=None, context=None):
197 access_rights_uid = access_rights_uid or uid
198 stage_obj = self.pool.get('hr.recruitment.stage')
199 order = stage_obj._order
200 if read_group_order == 'stage_id desc':
201 # lame hack to allow reverting search, should just work in the trivial case
202 order = "%s desc" % order
203 stage_ids = stage_obj._search(cr, uid, ['|',('id','in',ids),('department_id','=',False)], order=order,
204 access_rights_uid=access_rights_uid, context=context)
205 result = stage_obj.name_get(cr, access_rights_uid, stage_ids, context=context)
206 # restore order of the search
207 result.sort(lambda x,y: cmp(stage_ids.index(x[0]), stage_ids.index(y[0])))
211 'stage_id': _read_group_stage_ids
215 def onchange_job(self,cr, uid, ids, job, context=None):
219 job_obj = self.pool.get('hr.job')
220 result['department_id'] = job_obj.browse(cr, uid, job, context=context).department_id.id
221 return {'value': result}
222 return {'value': {'department_id': False}}
224 def onchange_department_id(self, cr, uid, ids, department_id=False, context=None):
225 if not department_id:
226 return {'value': {'stage_id': False}}
227 obj_recru_stage = self.pool.get('hr.recruitment.stage')
228 stage_ids = obj_recru_stage.search(cr, uid, ['|',('department_id','=',department_id),('department_id','=',False)], context=context)
229 stage_id = stage_ids and stage_ids[0] or False
230 return {'value': {'stage_id': stage_id}}
232 def stage_previous(self, cr, uid, ids, context=None):
233 """This function computes previous stage for case from its current stage
234 using available stage for that case type
235 @param self: The object pointer
236 @param cr: the current row, from the database cursor,
237 @param uid: the current user’s ID for security checks,
238 @param ids: List of case IDs
239 @param context: A standard dictionary for contextual values"""
240 stage_obj = self.pool.get('hr.recruitment.stage')
241 for case in self.browse(cr, uid, ids, context=context):
242 department = (case.department_id.id or False)
243 st = case.stage_id.id or False
244 stage_ids = stage_obj.search(cr, uid, ['|',('department_id','=',department),('department_id','=',False)], context=context)
245 if st and stage_ids.index(st):
246 self.write(cr, uid, [case.id], {'stage_id': stage_ids[stage_ids.index(st)-1]}, context=context)
248 self.write(cr, uid, [case.id], {'stage_id': False}, context=context)
251 def stage_next(self, cr, uid, ids, context=None):
252 """This function computes next stage for case from its current stage
253 using available stage for that case type
254 @param self: The object pointer
255 @param cr: the current row, from the database cursor,
256 @param uid: the current user’s ID for security checks,
257 @param ids: List of case IDs
258 @param context: A standard dictionary for contextual values"""
259 stage_obj = self.pool.get('hr.recruitment.stage')
260 for case in self.browse(cr, uid, ids, context=context):
261 department = (case.department_id.id or False)
262 st = case.stage_id.id or False
263 stage_ids = stage_obj.search(cr, uid, ['|',('department_id','=',department),('department_id','=',False)], context=context)
265 if st and len(stage_ids) != stage_ids.index(st)+1:
266 val = stage_ids[stage_ids.index(st)+1]
267 elif (not st) and stage_ids:
271 self.write(cr, uid, [case.id], {'stage_id': val}, context=context)
274 def action_makeMeeting(self, cr, uid, ids, context=None):
276 This opens Meeting's calendar view to schedule meeting on current Opportunity
277 @param self: The object pointer
278 @param cr: the current row, from the database cursor,
279 @param uid: the current user’s ID for security checks,
280 @param ids: List of Opportunity to Meeting IDs
281 @param context: A standard dictionary for contextual values
283 @return: Dictionary value for created Meeting view
285 data_obj = self.pool.get('ir.model.data')
289 for opp in self.browse(cr, uid, ids, context=context):
291 result = data_obj._get_id(cr, uid, 'crm', 'view_crm_case_meetings_filter')
292 res = data_obj.read(cr, uid, result, ['res_id'], context=context)
293 id1 = data_obj._get_id(cr, uid, 'crm', 'crm_case_calendar_view_meet')
294 id2 = data_obj._get_id(cr, uid, 'crm', 'crm_case_form_view_meet')
295 id3 = data_obj._get_id(cr, uid, 'crm', 'crm_case_tree_view_meet')
297 id1 = data_obj.browse(cr, uid, id1, context=context).res_id
299 id2 = data_obj.browse(cr, uid, id2, context=context).res_id
301 id3 = data_obj.browse(cr, uid, id3, context=context).res_id
304 'default_partner_id': opp.partner_id and opp.partner_id.id or False,
305 'default_email_from': opp.email_from,
306 'default_state': 'open',
307 'default_name': opp.name
310 'name': ('Meetings'),
311 'domain': "[('user_id','=',%s)]" % (uid),
314 'view_mode': 'calendar,form,tree',
315 'res_model': 'crm.meeting',
317 'views': [(id1, 'calendar'), (id2, 'form'), (id3, 'tree')],
318 'type': 'ir.actions.act_window',
319 'search_view_id': res['res_id'],
324 def action_print_survey(self, cr, uid, ids, context=None):
326 If response is available then print this response otherwise print survey form(print template of the survey).
328 @param self: The object pointer
329 @param cr: the current row, from the database cursor,
330 @param uid: the current user’s ID for security checks,
331 @param ids: List of Survey IDs
332 @param context: A standard dictionary for contextual values
333 @return: Dictionary value for print survey form.
337 record = self.browse(cr, uid, ids, context=context)
338 record = record and record[0]
339 context.update({'survey_id': record.survey.id, 'response_id': [record.response], 'response_no': 0, })
340 value = self.pool.get("survey").action_print_survey(cr, uid, ids, context=context)
343 def message_new(self, cr, uid, msg, custom_values=None, context=None):
344 """Automatically called when new email message arrives"""
345 res_id = super(hr_applicant,self).message_new(cr, uid, msg, custom_values=custom_values, context=context)
346 subject = msg.get('subject') or _("No Subject")
347 body = msg.get('body_text')
348 msg_from = msg.get('from')
349 priority = msg.get('priority')
352 'email_from': msg_from,
353 'email_cc': msg.get('cc'),
358 vals['priority'] = priority
359 vals.update(self.message_partner_by_email(cr, uid, msg.get('from', False)))
360 self.write(cr, uid, [res_id], vals, context)
363 def message_update(self, cr, uid, ids, msg, vals=None, default_act='pending', context=None):
364 if isinstance(ids, (str, int, long)):
368 msg_from = msg['from']
370 'description': msg['body_text']
372 if msg.get('priority', False):
373 vals['priority'] = msg.get('priority')
376 'cost':'planned_cost',
377 'revenue': 'planned_revenue',
378 'probability':'probability'
381 for line in msg['body_text'].split('\n'):
383 res = tools.misc.command_re.match(line)
384 if res and maps.get(res.group(1).lower(), False):
385 key = maps.get(res.group(1).lower())
386 vls[key] = res.group(2).lower()
389 res = self.write(cr, uid, ids, vals, context=context)
390 self.message_append_dict(cr, uid, ids, msg, context=context)
393 def create(self, cr, uid, vals, context=None):
394 obj_id = super(hr_applicant, self).create(cr, uid, vals, context=context)
395 self.create_send_note(cr, uid, [obj_id], context=context)
398 def case_open(self, cr, uid, ids, context=None):
400 open Request of the applicant for the hr_recruitment
402 res = super(hr_applicant, self).case_open(cr, uid, ids, context)
403 date = self.read(cr, uid, ids, ['date_open'])[0]
404 if not date['date_open']:
405 self.write(cr, uid, ids, {'date_open': time.strftime('%Y-%m-%d %H:%M:%S'),})
408 def case_close(self, cr, uid, ids, context=None):
409 res = super(hr_applicant, self).case_close(cr, uid, ids, context)
412 def case_close_with_emp(self, cr, uid, ids, context=None):
415 hr_employee = self.pool.get('hr.employee')
416 model_data = self.pool.get('ir.model.data')
417 act_window = self.pool.get('ir.actions.act_window')
419 for applicant in self.browse(cr, uid, ids, context=context):
421 if applicant.partner_id:
422 address_id = applicant.partner_id.address_get(['contact'])['contact']
424 applicant.job_id.write({'no_of_recruitment': applicant.job_id.no_of_recruitment - 1})
425 emp_id = hr_employee.create(cr,uid,{'name': applicant.partner_name or applicant.name,
426 'job_id': applicant.job_id.id,
427 'address_home_id': address_id,
428 'department_id': applicant.department_id.id
430 self.write(cr, uid, [applicant.id], {'emp_id': emp_id}, context=context)
431 self.case_close(cr, uid, [applicant.id], context)
433 raise osv.except_osv(_('Warning!'),_('You must define Applied Job for this applicant.'))
435 action_model, action_id = model_data.get_object_reference(cr, uid, 'hr', 'open_view_employee_list')
436 dict_act_window = act_window.read(cr, uid, action_id, [])
438 dict_act_window['res_id'] = emp_id
439 dict_act_window['view_mode'] = 'form,tree'
440 return dict_act_window
442 def case_cancel(self, cr, uid, ids, context=None):
443 """Overrides cancel for crm_case for setting probability
445 res = super(hr_applicant, self).case_cancel(cr, uid, ids, context)
446 self.write(cr, uid, ids, {'probability' : 0.0})
449 def case_pending(self, cr, uid, ids, context=None):
450 """Marks case as pending"""
451 res = super(hr_applicant, self).case_pending(cr, uid, ids, context)
452 self.write(cr, uid, ids, {'probability' : 0.0})
455 def case_reset(self, cr, uid, ids, context=None):
456 """Resets case as draft
458 res = super(hr_applicant, self).case_reset(cr, uid, ids, context)
459 self.write(cr, uid, ids, {'date_open': False, 'date_closed': False})
462 def set_priority(self, cr, uid, ids, priority, *args):
463 """Set applicant priority
465 return self.write(cr, uid, ids, {'priority' : priority})
467 def set_high_priority(self, cr, uid, ids, *args):
468 """Set applicant priority to high
470 return self.set_priority(cr, uid, ids, '1')
472 def set_normal_priority(self, cr, uid, ids, *args):
473 """Set applicant priority to normal
475 return self.set_priority(cr, uid, ids, '3')
477 def write(self, cr, uid, ids, vals, context=None):
478 if 'stage_id' in vals and vals['stage_id']:
479 stage = self.pool.get('hr.recruitment.stage').browse(cr, uid, vals['stage_id'], context=context)
480 self.message_append_note(cr, uid, ids, body=_("Stage changed to <b>%s</b>.") % stage.name, context=context)
481 return super(hr_applicant,self).write(cr, uid, ids, vals, context=context)
483 # -------------------------------------------------------
484 # OpenChatter methods and notifications
485 # -------------------------------------------------------
487 def message_get_subscribers(self, cr, uid, ids, get_ids=False, context=None):
488 user_ids = super(hr_applicant, self).message_get_subscribers(cr, uid, ids, True, context=context)
489 for obj in self.browse(cr, uid, ids, context=context):
490 if obj.user_id and not obj.user_id.id in user_ids:
491 self.message_subscribe(cr, uid, [obj.id], [obj.user_id.id], context=context)
492 return super(hr_applicant, self).message_get_subscribers(cr, uid, ids, get_ids, context=context)
494 def get_needaction_user_ids(self, cr, uid, ids, context=None):
495 result = dict.fromkeys(ids, [])
496 for obj in self.browse(cr, uid, ids, context=context):
497 if obj.state == 'draft' and obj.user_id:
498 result[obj.id] = [obj.user_id.id]
501 def case_get_note_msg_prefix(self, cr, uid, id, context=None):
504 def case_open_send_note(self, cr, uid, ids, context=None):
505 message = _("Applicant has been set <b>in progress</b>.")
506 return self.message_append_note(cr, uid, ids, body=message, context=context)
508 def case_close_send_note(self, cr, uid, ids, context=None):
511 for applicant in self.browse(cr, uid, ids, context=context):
513 message = _("Applicant has been <b>hired</b> and created as an employee.")
514 self.message_append_note(cr, uid, [applicant.id], body=message, context=context)
516 message = _("Applicant has been <b>hired</b>.")
517 self.message_append_note(cr, uid, [applicant.id], body=message, context=context)
520 def case_cancel_send_note(self, cr, uid, ids, context=None):
521 msg = 'Applicant <b>refused</b>.'
522 return self.message_append_note(cr, uid, ids, body=msg, context=context)
524 def case_reset_send_note(self, cr, uid, ids, context=None):
525 message =_("Applicant has been set as <b>new</b>.")
526 return self.message_append_note(cr, uid, ids, body=message, context=context)
528 def create_send_note(self, cr, uid, ids, context=None):
529 message = _("Applicant has been <b>created</b>.")
530 return self.message_append_note(cr, uid, ids, body=message, context=context)
534 class hr_job(osv.osv):
538 'survey_id': fields.many2one('survey', 'Interview Form', help="Choose an interview form for this job position and you will be able to print/answer this interview from all applicants who apply for this job"),
542 # vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4: