547 lines
27 KiB
Python
547 lines
27 KiB
Python
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
|
|
|
from collections import OrderedDict
|
|
from operator import itemgetter
|
|
from markupsafe import Markup
|
|
|
|
from odoo import conf, http, _
|
|
from odoo.exceptions import AccessError, MissingError
|
|
from odoo.http import request
|
|
from odoo.addons.portal.controllers.portal import CustomerPortal, pager as portal_pager
|
|
from odoo.tools import groupby as groupbyelem
|
|
|
|
from odoo.osv.expression import OR, AND
|
|
|
|
|
|
class ProjectCustomerPortal(CustomerPortal):
|
|
|
|
def _prepare_home_portal_values(self, counters):
|
|
values = super()._prepare_home_portal_values(counters)
|
|
if 'project_count' in counters:
|
|
values['project_count'] = request.env['project.project'].search_count([]) \
|
|
if request.env['project.project'].check_access_rights('read', raise_exception=False) else 0
|
|
if 'task_count' in counters:
|
|
values['task_count'] = request.env['project.task'].search_count([('project_id', '!=', False)]) \
|
|
if request.env['project.task'].check_access_rights('read', raise_exception=False) else 0
|
|
return values
|
|
|
|
# ------------------------------------------------------------
|
|
# My Project
|
|
# ------------------------------------------------------------
|
|
def _project_get_page_view_values(self, project, access_token, page=1, date_begin=None, date_end=None, sortby=None, search=None, search_in='content', groupby=None, **kwargs):
|
|
# default filter by value
|
|
domain = [('project_id', '=', project.id)]
|
|
# pager
|
|
url = "/my/projects/%s" % project.id
|
|
values = self._prepare_tasks_values(page, date_begin, date_end, sortby, search, search_in, groupby, url, domain, su=bool(access_token), project=project)
|
|
# adding the access_token to the pager's url args,
|
|
# so we are not prompted for loging when switching pages
|
|
# if access_token is None, the arg is not present in the URL
|
|
values['pager']['url_args']['access_token'] = access_token
|
|
pager = portal_pager(**values['pager'])
|
|
|
|
values.update(
|
|
grouped_tasks=values['grouped_tasks'](pager['offset']),
|
|
page_name='project',
|
|
pager=pager,
|
|
project=project,
|
|
task_url=f'projects/{project.id}/task',
|
|
preview_object=project,
|
|
)
|
|
|
|
if not groupby:
|
|
values['groupby'] = 'project' if self._display_project_groupby(project) else 'none'
|
|
|
|
return self._get_page_view_values(project, access_token, values, 'my_projects_history', False, **kwargs)
|
|
|
|
def _prepare_project_domain(self):
|
|
return []
|
|
|
|
def _prepare_searchbar_sortings(self):
|
|
return {
|
|
'date': {'label': _('Newest'), 'order': 'create_date desc'},
|
|
'name': {'label': _('Name'), 'order': 'name'},
|
|
}
|
|
|
|
@http.route(['/my/projects', '/my/projects/page/<int:page>'], type='http', auth="user", website=True)
|
|
def portal_my_projects(self, page=1, date_begin=None, date_end=None, sortby=None, **kw):
|
|
values = self._prepare_portal_layout_values()
|
|
Project = request.env['project.project']
|
|
domain = self._prepare_project_domain()
|
|
|
|
searchbar_sortings = self._prepare_searchbar_sortings()
|
|
if not sortby:
|
|
sortby = 'date'
|
|
order = searchbar_sortings[sortby]['order']
|
|
|
|
if date_begin and date_end:
|
|
domain += [('create_date', '>', date_begin), ('create_date', '<=', date_end)]
|
|
|
|
# projects count
|
|
project_count = Project.search_count(domain)
|
|
# pager
|
|
pager = portal_pager(
|
|
url="/my/projects",
|
|
url_args={'date_begin': date_begin, 'date_end': date_end, 'sortby': sortby},
|
|
total=project_count,
|
|
page=page,
|
|
step=self._items_per_page
|
|
)
|
|
|
|
# content according to pager and archive selected
|
|
projects = Project.search(domain, order=order, limit=self._items_per_page, offset=pager['offset'])
|
|
request.session['my_projects_history'] = projects.ids[:100]
|
|
|
|
values.update({
|
|
'date': date_begin,
|
|
'date_end': date_end,
|
|
'projects': projects,
|
|
'page_name': 'project',
|
|
'default_url': '/my/projects',
|
|
'pager': pager,
|
|
'searchbar_sortings': searchbar_sortings,
|
|
'sortby': sortby
|
|
})
|
|
return request.render("project.portal_my_projects", values)
|
|
|
|
@http.route(['/my/project/<int:project_id>',
|
|
'/my/project/<int:project_id>/page/<int:page>',
|
|
'/my/project/<int:project_id>/task/<int:task_id>',
|
|
'/my/project/<int:project_id>/project_sharing'], type='http', auth="public")
|
|
def portal_project_routes_outdated(self, **kwargs):
|
|
""" Redirect the outdated routes to the new routes. """
|
|
return request.redirect(request.httprequest.full_path.replace('/my/project/', '/my/projects/'))
|
|
|
|
@http.route(['/my/task',
|
|
'/my/task/page/<int:page>',
|
|
'/my/task/<int:task_id>'], type='http', auth='public')
|
|
def portal_my_task_routes_outdated(self, **kwargs):
|
|
""" Redirect the outdated routes to the new routes. """
|
|
return request.redirect(request.httprequest.full_path.replace('/my/task', '/my/tasks'))
|
|
|
|
@http.route(['/my/projects/<int:project_id>', '/my/projects/<int:project_id>/page/<int:page>'], type='http', auth="public", website=True)
|
|
def portal_my_project(self, project_id=None, access_token=None, page=1, date_begin=None, date_end=None, sortby=None, search=None, search_in='content', groupby=None, task_id=None, **kw):
|
|
try:
|
|
project_sudo = self._document_check_access('project.project', project_id, access_token)
|
|
except (AccessError, MissingError):
|
|
return request.redirect('/my')
|
|
if project_sudo.collaborator_count and project_sudo.with_user(request.env.user)._check_project_sharing_access():
|
|
values = {'project_id': project_id}
|
|
if task_id:
|
|
values['task_id'] = task_id
|
|
return request.render("project.project_sharing_portal", values)
|
|
project_sudo = project_sudo if access_token else project_sudo.with_user(request.env.user)
|
|
if not groupby:
|
|
groupby = 'stage'
|
|
values = self._project_get_page_view_values(project_sudo, access_token, page, date_begin, date_end, sortby, search, search_in, groupby, **kw)
|
|
return request.render("project.portal_my_project", values)
|
|
|
|
def _get_project_sharing_company(self, project):
|
|
return project.company_id or request.env.user.company_id
|
|
|
|
def _prepare_project_sharing_session_info(self, project, task=None):
|
|
session_info = request.env['ir.http'].session_info()
|
|
user_context = dict(request.env.context) if request.session.uid else {}
|
|
mods = conf.server_wide_modules or []
|
|
if request.env.lang:
|
|
lang = request.env.lang
|
|
session_info['user_context']['lang'] = lang
|
|
# Update Cache
|
|
user_context['lang'] = lang
|
|
lang = user_context.get("lang")
|
|
translation_hash = request.env['ir.http'].get_web_translations_hash(mods, lang)
|
|
cache_hashes = {
|
|
"translations": translation_hash,
|
|
}
|
|
|
|
project_company = self._get_project_sharing_company(project)
|
|
|
|
session_info.update(
|
|
cache_hashes=cache_hashes,
|
|
action_name=project.action_project_sharing(),
|
|
project_id=project.id,
|
|
user_companies={
|
|
'current_company': project_company.id,
|
|
'allowed_companies': {
|
|
project_company.id: {
|
|
'id': project_company.id,
|
|
'name': project_company.name,
|
|
},
|
|
},
|
|
},
|
|
# FIXME: See if we prefer to give only the currency that the portal user just need to see the correct information in project sharing
|
|
currencies=request.env['ir.http'].get_currencies(),
|
|
)
|
|
if task:
|
|
session_info['open_task_action'] = task.action_project_sharing_open_task()
|
|
return session_info
|
|
|
|
@http.route("/my/projects/<int:project_id>/project_sharing", type="http", auth="user", methods=['GET'])
|
|
def render_project_backend_view(self, project_id, task_id=None):
|
|
project = request.env['project.project'].sudo().browse(project_id)
|
|
if not project.exists() or not project.with_user(request.env.user)._check_project_sharing_access():
|
|
return request.not_found()
|
|
task = task_id and request.env['project.task'].browse(int(task_id))
|
|
return request.render(
|
|
'project.project_sharing_embed',
|
|
{'session_info': self._prepare_project_sharing_session_info(project, task)},
|
|
)
|
|
|
|
@http.route('/my/projects/<int:project_id>/task/<int:task_id>', type='http', auth='public', website=True)
|
|
def portal_my_project_task(self, project_id=None, task_id=None, access_token=None, **kw):
|
|
try:
|
|
project_sudo = self._document_check_access('project.project', project_id, access_token)
|
|
except (AccessError, MissingError):
|
|
return request.redirect('/my')
|
|
Task = request.env['project.task']
|
|
if access_token:
|
|
Task = Task.sudo()
|
|
task_sudo = Task.search([('project_id', '=', project_id), ('id', '=', task_id)], limit=1).sudo()
|
|
task_sudo.attachment_ids.generate_access_token()
|
|
values = self._task_get_page_view_values(task_sudo, access_token, project=project_sudo, **kw)
|
|
values['project'] = project_sudo
|
|
return request.render("project.portal_my_task", values)
|
|
|
|
@http.route('/my/projects/<int:project_id>/task/<int:task_id>/subtasks', type='http', auth='user', methods=['GET'], website=True)
|
|
def portal_my_project_subtasks(self, project_id, task_id, page=1, date_begin=None, date_end=None, sortby=None, filterby=None, search=None, search_in='content', groupby=None, **kw):
|
|
try:
|
|
project_sudo = self._document_check_access('project.project', project_id)
|
|
task_sudo = request.env['project.task'].search([('project_id', '=', project_id), ('id', '=', task_id)]).sudo()
|
|
task_domain = [('id', 'child_of', task_id), ('id', '!=', task_id)]
|
|
searchbar_filters = self._get_my_tasks_searchbar_filters([('id', '=', task_sudo.project_id.id)], task_domain)
|
|
|
|
if not filterby:
|
|
filterby = 'all'
|
|
domain = searchbar_filters.get(filterby, searchbar_filters.get('all'))['domain']
|
|
|
|
values = self._prepare_tasks_values(page, date_begin, date_end, sortby, search, search_in, groupby, url=f'/my/projects/{project_id}/task/{task_id}/subtasks', domain=AND([task_domain, domain]))
|
|
values['page_name'] = 'project_subtasks'
|
|
|
|
# pager
|
|
pager_vals = values['pager']
|
|
pager_vals['url_args'].update(filterby=filterby)
|
|
pager = portal_pager(**pager_vals)
|
|
|
|
values.update({
|
|
'project': project_sudo,
|
|
'task': task_sudo,
|
|
'grouped_tasks': values['grouped_tasks'](pager['offset']),
|
|
'pager': pager,
|
|
'searchbar_filters': OrderedDict(sorted(searchbar_filters.items())),
|
|
'filterby': filterby,
|
|
})
|
|
return request.render("project.portal_my_tasks", values)
|
|
except (AccessError, MissingError):
|
|
return request.not_found()
|
|
|
|
# ------------------------------------------------------------
|
|
# My Task
|
|
# ------------------------------------------------------------
|
|
def _task_get_page_view_values(self, task, access_token, **kwargs):
|
|
project = kwargs.get('project')
|
|
if project:
|
|
project_accessible = True
|
|
page_name = 'project_task'
|
|
history = 'my_project_tasks_history'
|
|
else:
|
|
page_name = 'task'
|
|
history = 'my_tasks_history'
|
|
try:
|
|
project_accessible = bool(task.project_id.id and self._document_check_access('project.project', task.project_id.id))
|
|
except (AccessError, MissingError):
|
|
project_accessible = False
|
|
values = {
|
|
'page_name': page_name,
|
|
'task': task,
|
|
'user': request.env.user,
|
|
'project_accessible': project_accessible,
|
|
'task_link_section': [],
|
|
'preview_object': task,
|
|
}
|
|
|
|
values = self._get_page_view_values(task, access_token, values, history, False, **kwargs)
|
|
if project:
|
|
values['project_id'] = project.id
|
|
history = request.session.get('my_project_tasks_history', [])
|
|
try:
|
|
current_task_index = history.index(task.id)
|
|
except ValueError:
|
|
return values
|
|
|
|
total_task = len(history)
|
|
task_url = f"{task.project_id.access_url}/task/%s?model=project.project&res_id={values['user'].id}&access_token={access_token}"
|
|
|
|
values['prev_record'] = current_task_index != 0 and task_url % history[current_task_index - 1]
|
|
values['next_record'] = current_task_index < total_task - 1 and task_url % history[current_task_index + 1]
|
|
|
|
return values
|
|
|
|
def _task_get_searchbar_sortings(self, milestones_allowed, project=False):
|
|
values = {
|
|
'date': {'label': _('Newest'), 'order': 'create_date desc', 'sequence': 1},
|
|
'name': {'label': _('Title'), 'order': 'name', 'sequence': 2},
|
|
'users': {'label': _('Assignees'), 'order': 'user_ids', 'sequence': 4},
|
|
'stage': {'label': _('Stage'), 'order': 'stage_id, project_id', 'sequence': 5},
|
|
'status': {'label': _('Status'), 'order': 'state', 'sequence': 6},
|
|
'priority': {'label': _('Priority'), 'order': 'priority desc', 'sequence': 8},
|
|
'date_deadline': {'label': _('Deadline'), 'order': 'date_deadline asc', 'sequence': 9},
|
|
'update': {'label': _('Last Stage Update'), 'order': 'date_last_stage_update desc', 'sequence': 11},
|
|
}
|
|
if not project:
|
|
values['project'] = {'label': _('Project'), 'order': 'project_id, stage_id', 'sequence': 3}
|
|
if milestones_allowed:
|
|
values['milestone'] = {'label': _('Milestone'), 'order': 'milestone_id', 'sequence': 7}
|
|
return values
|
|
|
|
# Meant to be overridden in documents_project
|
|
def _display_project_groupby(self, project):
|
|
return not project
|
|
|
|
def _task_get_searchbar_groupby(self, milestones_allowed, project=False):
|
|
values = {
|
|
'none': {'input': 'none', 'label': _('None'), 'order': 1},
|
|
'stage': {'input': 'stage', 'label': _('Stage'), 'order': 4},
|
|
'status': {'input': 'status', 'label': _('Status'), 'order': 5},
|
|
'priority': {'input': 'priority', 'label': _('Priority'), 'order': 7},
|
|
'customer': {'input': 'customer', 'label': _('Customer'), 'order': 10},
|
|
}
|
|
if self._display_project_groupby(project):
|
|
values['project'] = {'input': 'project', 'label': _('Project'), 'order': 2}
|
|
if milestones_allowed:
|
|
values['milestone'] = {'input': 'milestone', 'label': _('Milestone'), 'order': 6}
|
|
return dict(sorted(values.items(), key=lambda item: item[1]["order"]))
|
|
|
|
def _task_get_groupby_mapping(self):
|
|
return {
|
|
'project': 'project_id',
|
|
'stage': 'stage_id',
|
|
'customer': 'partner_id',
|
|
'milestone': 'milestone_id',
|
|
'priority': 'priority',
|
|
'status': 'state',
|
|
}
|
|
|
|
def _task_get_order(self, order, groupby):
|
|
groupby_mapping = self._task_get_groupby_mapping()
|
|
field_name = groupby_mapping.get(groupby, '')
|
|
if not field_name:
|
|
return order
|
|
return '%s, %s' % (field_name, order)
|
|
|
|
def _task_get_searchbar_inputs(self, milestones_allowed, project=False):
|
|
values = {
|
|
'all': {'input': 'all', 'label': _('Search in All'), 'order': 1},
|
|
'content': {'input': 'content', 'label': Markup(_('Search <span class="nolabel"> (in Content)</span>')), 'order': 1},
|
|
'ref': {'input': 'ref', 'label': _('Search in Ref'), 'order': 1},
|
|
'users': {'input': 'users', 'label': _('Search in Assignees'), 'order': 3},
|
|
'stage': {'input': 'stage', 'label': _('Search in Stages'), 'order': 4},
|
|
'status': {'input': 'status', 'label': _('Search in Status'), 'order': 5},
|
|
'priority': {'input': 'priority', 'label': _('Search in Priority'), 'order': 7},
|
|
'customer': {'input': 'customer', 'label': _('Search in Customer'), 'order': 10},
|
|
'message': {'input': 'message', 'label': _('Search in Messages'), 'order': 11},
|
|
}
|
|
if not project:
|
|
values['project'] = {'input': 'project', 'label': _('Search in Project'), 'order': 2}
|
|
if milestones_allowed:
|
|
values['milestone'] = {'input': 'milestone', 'label': _('Search in Milestone'), 'order': 6}
|
|
|
|
return dict(sorted(values.items(), key=lambda item: item[1]["order"]))
|
|
|
|
def _task_get_search_domain(self, search_in, search):
|
|
search_domain = []
|
|
if search_in in ('content', 'all'):
|
|
search_domain.append([('name', 'ilike', search)])
|
|
search_domain.append([('description', 'ilike', search)])
|
|
if search_in in ('customer', 'all'):
|
|
search_domain.append([('partner_id', 'ilike', search)])
|
|
if search_in in ('message', 'all'):
|
|
search_domain.append([('message_ids.body', 'ilike', search)])
|
|
if search_in in ('stage', 'all'):
|
|
search_domain.append([('stage_id', 'ilike', search)])
|
|
if search_in in ('project', 'all'):
|
|
search_domain.append([('project_id', 'ilike', search)])
|
|
if search_in in ('ref', 'all'):
|
|
search_domain.append([('id', 'ilike', search)])
|
|
if search_in in ('milestone', 'all'):
|
|
search_domain.append([('milestone_id', 'ilike', search)])
|
|
if search_in in ('users', 'all'):
|
|
user_ids = request.env['res.users'].sudo().search([('name', 'ilike', search)])
|
|
search_domain.append([('user_ids', 'in', user_ids.ids)])
|
|
if search_in in ('priority', 'all'):
|
|
search_domain.append([('priority', 'ilike', search == 'normal' and '0' or '1')])
|
|
if search_in in ('status', 'all'):
|
|
state_dict = dict(map(reversed, request.env['project.task']._fields['state']._description_selection(request.env)))
|
|
search_domain.append([('state', 'ilike', state_dict.get(search, search))])
|
|
return OR(search_domain)
|
|
|
|
def _prepare_tasks_values(self, page, date_begin, date_end, sortby, search, search_in, groupby, url="/my/tasks", domain=None, su=False, project=False):
|
|
values = self._prepare_portal_layout_values()
|
|
|
|
Task = request.env['project.task']
|
|
milestone_domain = AND([domain, [('allow_milestones', '=', 'True')]])
|
|
milestones_allowed = Task.sudo().search_count(milestone_domain, limit=1) == 1
|
|
searchbar_sortings = dict(sorted(self._task_get_searchbar_sortings(milestones_allowed, project).items(),
|
|
key=lambda item: item[1]["sequence"]))
|
|
searchbar_inputs = self._task_get_searchbar_inputs(milestones_allowed, project)
|
|
searchbar_groupby = self._task_get_searchbar_groupby(milestones_allowed, project)
|
|
|
|
if not domain:
|
|
domain = []
|
|
if not su and Task.check_access_rights('read'):
|
|
domain = AND([domain, request.env['ir.rule']._compute_domain(Task._name, 'read')])
|
|
Task_sudo = Task.sudo()
|
|
|
|
# default sort by value
|
|
if not sortby or (sortby == 'milestone' and not milestones_allowed):
|
|
sortby = 'date'
|
|
order = searchbar_sortings[sortby]['order']
|
|
|
|
# default group by value
|
|
if not groupby or (groupby == 'milestone' and not milestones_allowed):
|
|
groupby = 'project'
|
|
|
|
if date_begin and date_end:
|
|
domain += [('create_date', '>', date_begin), ('create_date', '<=', date_end)]
|
|
|
|
# search reset if needed
|
|
if not milestones_allowed and search_in == 'milestone':
|
|
search_in = 'all'
|
|
# search
|
|
if search and search_in:
|
|
domain += self._task_get_search_domain(search_in, search)
|
|
|
|
# content according to pager and archive selected
|
|
order = self._task_get_order(order, groupby)
|
|
|
|
def get_grouped_tasks(pager_offset):
|
|
tasks = Task_sudo.search(domain, order=order, limit=self._items_per_page, offset=pager_offset)
|
|
request.session['my_project_tasks_history' if url.startswith('/my/projects') else 'my_tasks_history'] = tasks.ids[:100]
|
|
|
|
tasks_project_allow_milestone = tasks.filtered(lambda t: t.allow_milestones)
|
|
tasks_no_milestone = tasks - tasks_project_allow_milestone
|
|
|
|
groupby_mapping = self._task_get_groupby_mapping()
|
|
group = groupby_mapping.get(groupby)
|
|
if group:
|
|
if group == 'milestone_id':
|
|
grouped_tasks = [Task_sudo.concat(*g) for k, g in groupbyelem(tasks_project_allow_milestone, itemgetter(group))]
|
|
|
|
if not grouped_tasks:
|
|
if tasks_no_milestone:
|
|
grouped_tasks = [tasks_no_milestone]
|
|
else:
|
|
if grouped_tasks[len(grouped_tasks) - 1][0].milestone_id and tasks_no_milestone:
|
|
grouped_tasks.append(tasks_no_milestone)
|
|
else:
|
|
grouped_tasks[len(grouped_tasks) - 1] |= tasks_no_milestone
|
|
|
|
else:
|
|
grouped_tasks = [Task_sudo.concat(*g) for k, g in groupbyelem(tasks, itemgetter(group))]
|
|
else:
|
|
grouped_tasks = [tasks] if tasks else []
|
|
|
|
|
|
task_states = dict(Task_sudo._fields['state']._description_selection(request.env))
|
|
if sortby == 'status':
|
|
if groupby == 'none' and grouped_tasks:
|
|
grouped_tasks[0] = grouped_tasks[0].sorted(lambda tasks: task_states.get(tasks.state))
|
|
else:
|
|
grouped_tasks.sort(key=lambda tasks: task_states.get(tasks[0].state))
|
|
return grouped_tasks
|
|
|
|
values.update({
|
|
'date': date_begin,
|
|
'date_end': date_end,
|
|
'grouped_tasks': get_grouped_tasks,
|
|
'allow_milestone': milestones_allowed,
|
|
'page_name': 'task',
|
|
'default_url': url,
|
|
'task_url': 'tasks',
|
|
'pager': {
|
|
"url": url,
|
|
"url_args": {'date_begin': date_begin, 'date_end': date_end, 'sortby': sortby, 'groupby': groupby, 'search_in': search_in, 'search': search},
|
|
"total": Task_sudo.search_count(domain),
|
|
"page": page,
|
|
"step": self._items_per_page
|
|
},
|
|
'searchbar_sortings': searchbar_sortings,
|
|
'searchbar_groupby': searchbar_groupby,
|
|
'searchbar_inputs': searchbar_inputs,
|
|
'search_in': search_in,
|
|
'search': search,
|
|
'sortby': sortby,
|
|
'groupby': groupby,
|
|
})
|
|
return values
|
|
|
|
def _get_my_tasks_searchbar_filters(self, project_domain=None, task_domain=None):
|
|
searchbar_filters = {
|
|
'all': {'label': _('All'), 'domain': [('project_id', '!=', False)]},
|
|
}
|
|
|
|
# extends filterby criteria with project the customer has access to
|
|
projects = request.env['project.project'].search(project_domain or [])
|
|
for project in projects:
|
|
searchbar_filters.update({
|
|
str(project.id): {'label': project.name, 'domain': [('project_id', '=', project.id)]}
|
|
})
|
|
|
|
# extends filterby criteria with project (criteria name is the project id)
|
|
# Note: portal users can't view projects they don't follow
|
|
project_groups = request.env['project.task']._read_group(AND([[('project_id', 'not in', projects.ids)], task_domain or []]),
|
|
['project_id'])
|
|
for [project] in project_groups:
|
|
proj_name = project.sudo().display_name if project else _('Others')
|
|
searchbar_filters.update({
|
|
str(project.id): {'label': proj_name, 'domain': [('project_id', '=', project.id)]}
|
|
})
|
|
return searchbar_filters
|
|
|
|
@http.route(['/my/tasks', '/my/tasks/page/<int:page>'], type='http', auth="user", website=True)
|
|
def portal_my_tasks(self, page=1, date_begin=None, date_end=None, sortby=None, filterby=None, search=None, search_in='content', groupby=None, **kw):
|
|
searchbar_filters = self._get_my_tasks_searchbar_filters()
|
|
|
|
if not filterby:
|
|
filterby = 'all'
|
|
domain = searchbar_filters.get(filterby, searchbar_filters.get('all'))['domain']
|
|
|
|
values = self._prepare_tasks_values(page, date_begin, date_end, sortby, search, search_in, groupby, domain=domain)
|
|
|
|
# pager
|
|
pager_vals = values['pager']
|
|
pager_vals['url_args'].update(filterby=filterby)
|
|
pager = portal_pager(**pager_vals)
|
|
|
|
values.update({
|
|
'grouped_tasks': values['grouped_tasks'](pager['offset']),
|
|
'pager': pager,
|
|
'searchbar_filters': OrderedDict(sorted(searchbar_filters.items())),
|
|
'filterby': filterby,
|
|
})
|
|
return request.render("project.portal_my_tasks", values)
|
|
|
|
def _show_task_report(self, task_sudo, report_type, download):
|
|
# This method is to be overriden to report timesheets if the module is installed.
|
|
# The route should not be called if at least hr_timesheet is not installed
|
|
raise MissingError(_('There is nothing to report.'))
|
|
|
|
@http.route(['/my/tasks/<int:task_id>'], type='http', auth="public", website=True)
|
|
def portal_my_task(self, task_id, report_type=None, access_token=None, project_sharing=False, **kw):
|
|
try:
|
|
task_sudo = self._document_check_access('project.task', task_id, access_token)
|
|
except (AccessError, MissingError):
|
|
return request.redirect('/my')
|
|
|
|
if report_type in ('pdf', 'html', 'text'):
|
|
return self._show_task_report(task_sudo, report_type, download=kw.get('download'))
|
|
|
|
# ensure attachment are accessible with access token inside template
|
|
for attachment in task_sudo.attachment_ids:
|
|
attachment.generate_access_token()
|
|
if project_sharing is True:
|
|
# Then the user arrives to the stat button shown in form view of project.task and the portal user can see only 1 task
|
|
# so the history should be reset.
|
|
request.session['my_tasks_history'] = task_sudo.ids
|
|
values = self._task_get_page_view_values(task_sudo, access_token, **kw)
|
|
return request.render("project.portal_my_task", values)
|