619 lines
28 KiB
Python
619 lines
28 KiB
Python
# -*- coding: utf-8 -*-
|
|
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
|
|
|
from ast import literal_eval
|
|
from dateutil.relativedelta import relativedelta
|
|
import json
|
|
import werkzeug.urls
|
|
|
|
from pytz import utc, timezone
|
|
|
|
from odoo import api, fields, models, _
|
|
from odoo.addons.http_routing.models.ir_http import slug
|
|
from odoo.exceptions import ValidationError
|
|
from odoo.osv import expression
|
|
from odoo.tools.misc import get_lang, format_date
|
|
|
|
GOOGLE_CALENDAR_URL = 'https://www.google.com/calendar/render?'
|
|
|
|
|
|
class Event(models.Model):
|
|
_name = 'event.event'
|
|
_inherit = [
|
|
'event.event',
|
|
'website.seo.metadata',
|
|
'website.published.multi.mixin',
|
|
'website.cover_properties.mixin',
|
|
'website.searchable.mixin',
|
|
]
|
|
|
|
def _default_cover_properties(self):
|
|
res = super()._default_cover_properties()
|
|
res.update({
|
|
'background-image': "url('/website_event/static/src/img/event_cover_4.jpg')",
|
|
'opacity': '0.4',
|
|
'resize_class': 'cover_auto'
|
|
})
|
|
return res
|
|
|
|
def _default_question_ids(self):
|
|
return self.env['event.type']._default_question_ids()
|
|
|
|
# description
|
|
subtitle = fields.Char('Event Subtitle', translate=True)
|
|
# registration
|
|
is_participating = fields.Boolean("Is Participating", compute="_compute_is_participating")
|
|
# website
|
|
website_published = fields.Boolean(tracking=True)
|
|
website_menu = fields.Boolean(
|
|
string='Website Menu',
|
|
compute='_compute_website_menu', precompute=True, readonly=False, store=True,
|
|
help="Allows to display and manage event-specific menus on website.")
|
|
menu_id = fields.Many2one('website.menu', 'Event Menu', copy=False)
|
|
menu_register_cta = fields.Boolean(
|
|
'Extra Register Button', compute='_compute_menu_register_cta',
|
|
readonly=False, store=True)
|
|
# sub-menus management
|
|
introduction_menu = fields.Boolean(
|
|
"Introduction Menu", compute="_compute_website_menu_data",
|
|
readonly=False, store=True)
|
|
introduction_menu_ids = fields.One2many(
|
|
"website.event.menu", "event_id", string="Introduction Menus",
|
|
domain=[("menu_type", "=", "introduction")])
|
|
location_menu = fields.Boolean(
|
|
"Location Menu", compute="_compute_website_menu_data",
|
|
readonly=False, store=True)
|
|
location_menu_ids = fields.One2many(
|
|
"website.event.menu", "event_id", string="Location Menus",
|
|
domain=[("menu_type", "=", "location_menu")])
|
|
address_name = fields.Char(related='address_id.name')
|
|
register_menu = fields.Boolean(
|
|
"Register Menu", compute="_compute_website_menu_data",
|
|
readonly=False, store=True)
|
|
register_menu_ids = fields.One2many(
|
|
"website.event.menu", "event_id", string="Register Menus",
|
|
domain=[("menu_type", "=", "register")])
|
|
community_menu = fields.Boolean(
|
|
"Community Menu", compute="_compute_community_menu",
|
|
readonly=False, store=True,
|
|
help="Display community tab on website")
|
|
community_menu_ids = fields.One2many(
|
|
"website.event.menu", "event_id", string="Event Community Menus",
|
|
domain=[("menu_type", "=", "community")])
|
|
# live information
|
|
is_ongoing = fields.Boolean(
|
|
'Is Ongoing', compute='_compute_time_data', search='_search_is_ongoing',
|
|
help="Whether event has begun")
|
|
is_done = fields.Boolean(
|
|
'Is Done', compute='_compute_time_data')
|
|
start_today = fields.Boolean(
|
|
'Start Today', compute='_compute_time_data',
|
|
help="Whether event is going to start today if still not ongoing")
|
|
start_remaining = fields.Integer(
|
|
'Remaining before start', compute='_compute_time_data',
|
|
help="Remaining time before event starts (minutes)")
|
|
# questions
|
|
question_ids = fields.One2many(
|
|
'event.question', 'event_id', 'Questions', copy=True,
|
|
compute='_compute_question_ids', readonly=False, store=True)
|
|
general_question_ids = fields.One2many('event.question', 'event_id', 'General Questions',
|
|
domain=[('once_per_order', '=', True)])
|
|
specific_question_ids = fields.One2many('event.question', 'event_id', 'Specific Questions',
|
|
domain=[('once_per_order', '=', False)])
|
|
|
|
def _compute_is_participating(self):
|
|
"""Heuristic
|
|
|
|
* public, no visitor: not participating as we have no information;
|
|
* check only confirmed and attended registrations, a draft registration
|
|
does not make the attendee participating;
|
|
* public and visitor: check visitor is linked to a registration. As
|
|
visitors are merged on the top parent, current visitor check is
|
|
sufficient even for successive visits;
|
|
* logged, no visitor: check partner is linked to a registration. Do
|
|
not check the email as it is not really secure;
|
|
* logged as visitor: check partner or visitor are linked to a
|
|
registration;
|
|
"""
|
|
current_visitor = self.env['website.visitor']._get_visitor_from_request(force_create=False)
|
|
base_domain = [('event_id', 'in', self.ids), ('state', 'in', ['open', 'done'])]
|
|
if self.env.user._is_public() and not current_visitor:
|
|
events = self.env['event.event']
|
|
elif self.env.user._is_public():
|
|
events = self.env['event.registration'].sudo().search(
|
|
expression.AND([base_domain, [('visitor_id', '=', current_visitor.id)]])
|
|
).event_id
|
|
else:
|
|
if current_visitor:
|
|
domain = [
|
|
'|',
|
|
('partner_id', '=', self.env.user.partner_id.id),
|
|
('visitor_id', '=', current_visitor.id)
|
|
]
|
|
else:
|
|
domain = [('partner_id', '=', self.env.user.partner_id.id)]
|
|
events = self.env['event.registration'].sudo().search(
|
|
expression.AND([base_domain, domain])
|
|
).event_id
|
|
|
|
for event in self:
|
|
event.is_participating = event in events
|
|
|
|
@api.depends('event_type_id')
|
|
def _compute_website_menu(self):
|
|
""" Also ensure a value for website_menu as it is a trigger notably for
|
|
track related menus. """
|
|
for event in self:
|
|
if event.event_type_id and event.event_type_id != event._origin.event_type_id:
|
|
event.website_menu = event.event_type_id.website_menu
|
|
elif not event.website_menu:
|
|
event.website_menu = False
|
|
|
|
@api.depends("event_type_id", "website_menu", "community_menu")
|
|
def _compute_community_menu(self):
|
|
""" Set False in base module. Sub modules will add their own logic
|
|
(meet or track_quiz). """
|
|
for event in self:
|
|
event.community_menu = False
|
|
|
|
@api.depends("website_menu")
|
|
def _compute_website_menu_data(self):
|
|
""" Synchronize with website_menu at change and let people update them
|
|
at will afterwards. """
|
|
for event in self:
|
|
event.introduction_menu = event.website_menu
|
|
event.location_menu = event.website_menu
|
|
event.register_menu = event.website_menu
|
|
|
|
@api.depends("event_type_id", "website_menu")
|
|
def _compute_menu_register_cta(self):
|
|
""" At type onchange: synchronize. At website_menu update: synchronize. """
|
|
for event in self:
|
|
if event.event_type_id and event.event_type_id != event._origin.event_type_id:
|
|
event.menu_register_cta = event.event_type_id.menu_register_cta
|
|
elif event.website_menu and (event.website_menu != event._origin.website_menu or not event.menu_register_cta):
|
|
event.menu_register_cta = True
|
|
elif not event.website_menu:
|
|
event.menu_register_cta = False
|
|
|
|
@api.depends('date_begin', 'date_end')
|
|
def _compute_time_data(self):
|
|
""" Compute start and remaining time. Do everything in UTC as we compute only
|
|
time deltas here. """
|
|
now_utc = utc.localize(fields.Datetime.now().replace(microsecond=0))
|
|
for event in self:
|
|
date_begin_utc = utc.localize(event.date_begin, is_dst=False)
|
|
date_end_utc = utc.localize(event.date_end, is_dst=False)
|
|
event.is_ongoing = date_begin_utc <= now_utc <= date_end_utc
|
|
event.is_done = now_utc > date_end_utc
|
|
event.start_today = date_begin_utc.date() == now_utc.date()
|
|
if date_begin_utc >= now_utc:
|
|
td = date_begin_utc - now_utc
|
|
event.start_remaining = int(td.total_seconds() / 60)
|
|
else:
|
|
event.start_remaining = 0
|
|
|
|
@api.depends('name')
|
|
def _compute_website_url(self):
|
|
super(Event, self)._compute_website_url()
|
|
for event in self:
|
|
if event.id: # avoid to perform a slug on a not yet saved record in case of an onchange.
|
|
event.website_url = '/event/%s' % slug(event)
|
|
|
|
@api.depends('event_type_id')
|
|
def _compute_question_ids(self):
|
|
""" Update event questions from its event type. Depends are set only on
|
|
event_type_id itself to emulate an onchange. Changing event type content
|
|
itself should not trigger this method.
|
|
|
|
When synchronizing questions:
|
|
|
|
* lines with no registered answers are removed;
|
|
* type lines are added;
|
|
"""
|
|
if self._origin.question_ids:
|
|
# lines to keep: those with already given answers
|
|
questions_tokeep_ids = self.env['event.registration.answer'].search(
|
|
[('question_id', 'in', self._origin.question_ids.ids)]
|
|
).question_id.ids
|
|
else:
|
|
questions_tokeep_ids = []
|
|
for event in self:
|
|
if not event.event_type_id and not event.question_ids:
|
|
event.question_ids = self._default_question_ids()
|
|
continue
|
|
|
|
if questions_tokeep_ids:
|
|
questions_toremove = event._origin.question_ids.filtered(
|
|
lambda question: question.id not in questions_tokeep_ids)
|
|
command = [(3, question.id) for question in questions_toremove]
|
|
else:
|
|
command = [(5, 0)]
|
|
event.question_ids = command
|
|
|
|
# copy questions so changes in the event don't affect the event type
|
|
for question in event.event_type_id.question_ids:
|
|
event.question_ids += question.copy({'event_type_id': False})
|
|
|
|
# -------------------------------------------------------------------------
|
|
# CONSTRAINT METHODS
|
|
# -------------------------------------------------------------------------
|
|
|
|
@api.constrains('website_id')
|
|
def _check_website_id(self):
|
|
for event in self:
|
|
if event.website_id and event.website_id.company_id != event.company_id:
|
|
raise ValidationError(_("The website must be from the same company as the event."))
|
|
|
|
# ------------------------------------------------------------
|
|
# CRUD
|
|
# ------------------------------------------------------------
|
|
|
|
@api.model_create_multi
|
|
def create(self, vals_list):
|
|
events = super().create(vals_list)
|
|
events._update_website_menus()
|
|
return events
|
|
|
|
def write(self, vals):
|
|
menus_state_by_field = self._split_menus_state_by_field()
|
|
res = super(Event, self).write(vals)
|
|
menus_update_by_field = self._get_menus_update_by_field(menus_state_by_field, force_update=vals.keys())
|
|
self._update_website_menus(menus_update_by_field=menus_update_by_field)
|
|
return res
|
|
|
|
# ------------------------------------------------------------
|
|
# WEBSITE MENU MANAGEMENT
|
|
# ------------------------------------------------------------
|
|
|
|
def toggle_website_menu(self, val):
|
|
self.website_menu = val
|
|
|
|
def _get_menu_update_fields(self):
|
|
"""" Return a list of fields triggering a split of menu to activate /
|
|
menu to de-activate. Due to saas-13.3 improvement of menu management
|
|
this is done using side-methods to ease inheritance.
|
|
|
|
:return list: list of fields, each of which triggering a menu update
|
|
like website_menu, website_track, ... """
|
|
return ['community_menu', 'introduction_menu', 'location_menu', 'register_menu']
|
|
|
|
def _get_menu_type_field_matching(self):
|
|
return {
|
|
'community': 'community_menu',
|
|
'introduction': 'introduction_menu',
|
|
'location': 'location_menu',
|
|
'register': 'register_menu',
|
|
}
|
|
|
|
def _split_menus_state_by_field(self):
|
|
""" For each field linked to a menu, get the set of events having this
|
|
menu activated and de-activated. Purpose is to find those whose value
|
|
changed and update the underlying menus.
|
|
|
|
:return dict: key = name of field triggering a website menu update, get {
|
|
'activated': subset of self having its menu currently set to True
|
|
'deactivated': subset of self having its menu currently set to False
|
|
} """
|
|
menus_state_by_field = dict()
|
|
for fname in self._get_menu_update_fields():
|
|
activated = self.filtered(lambda event: event[fname])
|
|
menus_state_by_field[fname] = {
|
|
'activated': activated,
|
|
'deactivated': self - activated,
|
|
}
|
|
return menus_state_by_field
|
|
|
|
def _get_menus_update_by_field(self, menus_state_by_field, force_update=None):
|
|
""" For each field linked to a menu, get the set of events requiring
|
|
this menu to be activated or de-activated based on previous recorded
|
|
value.
|
|
|
|
:param menus_state_by_field: see ``_split_menus_state_by_field``;
|
|
:param force_update: list of field to which we force update of menus. This
|
|
is used notably when a direct write to a stored editable field messes with
|
|
its pre-computed value, notably in a transient mode (aka demo for example);
|
|
|
|
:return dict: key = name of field triggering a website menu update, get {
|
|
'activated': subset of self having its menu toggled to True
|
|
'deactivated': subset of self having its menu toggled to False
|
|
} """
|
|
menus_update_by_field = dict()
|
|
for fname in self._get_menu_update_fields():
|
|
if fname in force_update:
|
|
menus_update_by_field[fname] = self
|
|
else:
|
|
menus_update_by_field[fname] = self.env['event.event']
|
|
menus_update_by_field[fname] |= menus_state_by_field[fname]['activated'].filtered(lambda event: not event[fname])
|
|
menus_update_by_field[fname] |= menus_state_by_field[fname]['deactivated'].filtered(lambda event: event[fname])
|
|
return menus_update_by_field
|
|
|
|
def _get_website_menu_entries(self):
|
|
""" Method returning menu entries to display on the website view of the
|
|
event, possibly depending on some options in inheriting modules.
|
|
|
|
Each menu entry is a tuple containing :
|
|
* name: menu item name
|
|
* url: if set, url to a route (do not use xml_id in that case);
|
|
* xml_id: template linked to the page (do not use url in that case);
|
|
* sequence: specific sequence of menu entry to be set on the menu;
|
|
* menu_type: type of menu entry (used in inheriting modules to ease
|
|
menu management; not used in this module in 13.3 due to technical
|
|
limitations);
|
|
"""
|
|
self.ensure_one()
|
|
return [
|
|
(_('Introduction'), False, 'website_event.template_intro', 1, 'introduction'),
|
|
(_('Location'), False, 'website_event.template_location', 50, 'location'),
|
|
(_('Register'), '/event/%s/register' % slug(self), False, 100, 'register'),
|
|
(_('Community'), '/event/%s/community' % slug(self), False, 80, 'community'),
|
|
]
|
|
|
|
def _update_website_menus(self, menus_update_by_field=None):
|
|
""" Synchronize event configuration and its menu entries for frontend.
|
|
|
|
:param menus_update_by_field: see ``_get_menus_update_by_field``"""
|
|
for event in self:
|
|
if event.menu_id and not event.website_menu:
|
|
# do not rely on cascade, as it is done in SQL -> not calling override and
|
|
# letting some ir.ui.views in DB
|
|
(event.menu_id + event.menu_id.child_id).sudo().unlink()
|
|
elif event.website_menu and not event.menu_id:
|
|
root_menu = self.env['website.menu'].sudo().create({'name': event.name, 'website_id': event.website_id.id})
|
|
event.menu_id = root_menu
|
|
if event.menu_id and (not menus_update_by_field or event in menus_update_by_field.get('community_menu')):
|
|
event._update_website_menu_entry('community_menu', 'community_menu_ids', 'community')
|
|
if event.menu_id and (not menus_update_by_field or event in menus_update_by_field.get('introduction_menu')):
|
|
event._update_website_menu_entry('introduction_menu', 'introduction_menu_ids', 'introduction')
|
|
if event.menu_id and (not menus_update_by_field or event in menus_update_by_field.get('location_menu')):
|
|
event._update_website_menu_entry('location_menu', 'location_menu_ids', 'location')
|
|
if event.menu_id and (not menus_update_by_field or event in menus_update_by_field.get('register_menu')):
|
|
event._update_website_menu_entry('register_menu', 'register_menu_ids', 'register')
|
|
|
|
def _update_website_menu_entry(self, fname_bool, fname_o2m, fmenu_type):
|
|
""" Generic method to create menu entries based on a flag on event. This
|
|
method is a bit obscure, but is due to preparation of adding new menus
|
|
entries and pages for event in a stable version, leading to some constraints
|
|
while developing.
|
|
|
|
:param fname_bool: field name (e.g. website_track)
|
|
:param fname_o2m: o2m linking towards website.event.menu matching the
|
|
boolean fields (normally an entry of website.event.menu with type matching
|
|
the boolean field name)
|
|
:param method_name: method returning menu entries information: url, sequence, ...
|
|
"""
|
|
self.ensure_one()
|
|
new_menu = None
|
|
|
|
menu_data = [menu_info for menu_info in self._get_website_menu_entries()
|
|
if menu_info[4] == fmenu_type]
|
|
if self[fname_bool] and not self[fname_o2m]:
|
|
# menus not found but boolean True: get menus to create
|
|
for name, url, xml_id, menu_sequence, menu_type in menu_data:
|
|
new_menu = self._create_menu(menu_sequence, name, url, xml_id, menu_type)
|
|
elif not self[fname_bool]:
|
|
# will cascade delete to the website.event.menu
|
|
self[fname_o2m].mapped('menu_id').sudo().unlink()
|
|
|
|
return new_menu
|
|
|
|
def _create_menu(self, sequence, name, url, xml_id, menu_type):
|
|
""" Create a new menu for the current event.
|
|
|
|
If url: create a website menu. Menu leads directly to the URL that
|
|
should be a valid route.
|
|
|
|
If xml_id: create a new page using the qweb template given by its
|
|
xml_id. Take its url back thanks to new_page of website, then link
|
|
it to a menu. Template is duplicated and linked to a new url, meaning
|
|
each menu will have its own copy of the template. This is currently
|
|
limited to two menus: introduction and location.
|
|
|
|
:param menu_type: type of menu. Mainly used for inheritance purpose
|
|
allowing more fine-grain tuning of menus.
|
|
"""
|
|
self.check_access_rights('write')
|
|
view_id = False
|
|
if not url:
|
|
# add_menu=False, ispage=False -> simply create a new ir.ui.view with name
|
|
# and template
|
|
page_result = self.env['website'].sudo().new_page(
|
|
name=f'{name} {self.name}', template=xml_id,
|
|
add_menu=False, ispage=False)
|
|
url = f"/event/{slug(self)}/page{page_result['url']}" # url contains starting "/"
|
|
view_id = page_result['view_id']
|
|
|
|
website_menu = self.env['website.menu'].sudo().create({
|
|
'name': name,
|
|
'url': url,
|
|
'parent_id': self.menu_id.id,
|
|
'sequence': sequence,
|
|
'website_id': self.website_id.id,
|
|
})
|
|
self.env['website.event.menu'].create({
|
|
'menu_id': website_menu.id,
|
|
'event_id': self.id,
|
|
'menu_type': menu_type,
|
|
'view_id': view_id,
|
|
})
|
|
return website_menu
|
|
|
|
# ------------------------------------------------------------
|
|
# TOOLS
|
|
# ------------------------------------------------------------
|
|
|
|
def google_map_link(self, zoom=8):
|
|
""" Temporary method for stable """
|
|
return self._google_map_link(zoom=zoom)
|
|
|
|
def _google_map_link(self, zoom=8):
|
|
self.ensure_one()
|
|
if self.address_id:
|
|
return self.sudo().address_id.google_map_link(zoom=zoom)
|
|
return None
|
|
|
|
def _track_subtype(self, init_values):
|
|
self.ensure_one()
|
|
if init_values.keys() & {'is_published', 'website_published'}:
|
|
if self.is_published:
|
|
return self.env.ref('website_event.mt_event_published', raise_if_not_found=False)
|
|
return self.env.ref('website_event.mt_event_unpublished', raise_if_not_found=False)
|
|
return super(Event, self)._track_subtype(init_values)
|
|
|
|
def _get_event_resource_urls(self):
|
|
url_date_start = self.date_begin.astimezone(timezone(self.date_tz)).strftime('%Y%m%dT%H%M%S')
|
|
url_date_stop = self.date_end.astimezone(timezone(self.date_tz)).strftime('%Y%m%dT%H%M%S')
|
|
params = {
|
|
'action': 'TEMPLATE',
|
|
'text': self.name,
|
|
'dates': f'{url_date_start}/{url_date_stop}',
|
|
'ctz': self.date_tz,
|
|
'details': self.name,
|
|
}
|
|
if self.address_id:
|
|
params.update(location=self.address_inline)
|
|
encoded_params = werkzeug.urls.url_encode(params)
|
|
google_url = GOOGLE_CALENDAR_URL + encoded_params
|
|
iCal_url = f'/event/{self.id:d}/ics?{encoded_params}'
|
|
return {'google_url': google_url, 'iCal_url': iCal_url}
|
|
|
|
def _default_website_meta(self):
|
|
res = super(Event, self)._default_website_meta()
|
|
event_cover_properties = json.loads(self.cover_properties)
|
|
# background-image might contain single quotes eg `url('/my/url')`
|
|
res['default_opengraph']['og:image'] = res['default_twitter']['twitter:image'] = event_cover_properties.get('background-image', 'none')[4:-1].strip("'")
|
|
res['default_opengraph']['og:title'] = res['default_twitter']['twitter:title'] = self.name
|
|
res['default_opengraph']['og:description'] = res['default_twitter']['twitter:description'] = self.subtitle
|
|
res['default_twitter']['twitter:card'] = 'summary'
|
|
res['default_meta_description'] = self.subtitle
|
|
return res
|
|
|
|
def get_backend_menu_id(self):
|
|
return self.env.ref('event.event_main_menu').id
|
|
|
|
@api.model
|
|
def _search_build_dates(self):
|
|
today = fields.Datetime.today()
|
|
|
|
def sdn(date):
|
|
return fields.Datetime.to_string(date.replace(hour=23, minute=59, second=59))
|
|
|
|
def sd(date):
|
|
return fields.Datetime.to_string(date)
|
|
|
|
def get_month_filter_domain(filter_name, months_delta):
|
|
first_day_of_the_month = today.replace(day=1)
|
|
filter_string = _('This month') if months_delta == 0 \
|
|
else format_date(self.env, value=today + relativedelta(months=months_delta),
|
|
date_format='LLLL', lang_code=get_lang(self.env).code).capitalize()
|
|
return [filter_name, filter_string, [
|
|
("date_end", ">=", sd(first_day_of_the_month + relativedelta(months=months_delta))),
|
|
("date_begin", "<", sd(first_day_of_the_month + relativedelta(months=months_delta+1)))],
|
|
0]
|
|
|
|
return [
|
|
['upcoming', _('Upcoming Events'), [("date_end", ">", sd(today))], 0],
|
|
['today', _('Today'), [
|
|
("date_end", ">", sd(today)),
|
|
("date_begin", "<", sdn(today))],
|
|
0],
|
|
get_month_filter_domain('month', 0),
|
|
['old', _('Past Events'), [
|
|
("date_end", "<", sd(today))],
|
|
0],
|
|
['all', _('All Events'), [], 0]
|
|
]
|
|
|
|
@api.model
|
|
def _search_get_detail(self, website, order, options):
|
|
with_description = options['displayDescription']
|
|
with_date = options['displayDetail']
|
|
date = options.get('date', 'all')
|
|
country = options.get('country')
|
|
tags = options.get('tags')
|
|
event_type = options.get('type', 'all')
|
|
|
|
domain = [website.website_domain()]
|
|
if event_type != 'all':
|
|
domain.append([("event_type_id", "=", int(event_type))])
|
|
search_tags = self.env['event.tag']
|
|
if tags:
|
|
try:
|
|
tag_ids = literal_eval(tags)
|
|
except SyntaxError:
|
|
pass
|
|
else:
|
|
# perform a search to filter on existing / valid tags implicitely + apply rules on color
|
|
search_tags = self.env['event.tag'].search([('id', 'in', tag_ids)])
|
|
|
|
# Example: You filter on age: 10-12 and activity: football.
|
|
# Doing it this way allows to only get events who are tagged "age: 10-12" AND "activity: football".
|
|
# Add another tag "age: 12-15" to the search and it would fetch the ones who are tagged:
|
|
# ("age: 10-12" OR "age: 12-15") AND "activity: football
|
|
for tags in search_tags.grouped('category_id').values():
|
|
domain.append([('tag_ids', 'in', tags.ids)])
|
|
|
|
no_country_domain = domain.copy()
|
|
if country:
|
|
if country == 'online':
|
|
domain.append([("country_id", "=", False)])
|
|
elif country != 'all':
|
|
domain.append(['|', ("country_id", "=", int(country)), ("country_id", "=", False)])
|
|
|
|
no_date_domain = domain.copy()
|
|
dates = self._search_build_dates()
|
|
current_date = None
|
|
for date_details in dates:
|
|
if date == date_details[0]:
|
|
domain.append(date_details[2])
|
|
no_country_domain.append(date_details[2])
|
|
if date_details[0] != 'upcoming':
|
|
current_date = date_details[1]
|
|
|
|
search_fields = ['name']
|
|
fetch_fields = ['name', 'website_url', 'address_name']
|
|
mapping = {
|
|
'name': {'name': 'name', 'type': 'text', 'match': True},
|
|
'website_url': {'name': 'website_url', 'type': 'text', 'truncate': False},
|
|
'address_name': {'name': 'address_name', 'type': 'text', 'match': True},
|
|
}
|
|
if with_description:
|
|
search_fields.append('subtitle')
|
|
fetch_fields.append('subtitle')
|
|
mapping['description'] = {'name': 'subtitle', 'type': 'text', 'match': True}
|
|
if with_date:
|
|
mapping['detail'] = {'name': 'range', 'type': 'html'}
|
|
|
|
# Bypassing the access rigths of partner to search the address.
|
|
def search_in_address(env, search_term):
|
|
ret = env['event.event'].sudo()._search([
|
|
('address_search', 'ilike', search_term),
|
|
])
|
|
return [('id', 'in', ret)]
|
|
|
|
return {
|
|
'model': 'event.event',
|
|
'base_domain': domain,
|
|
'search_fields': search_fields,
|
|
'search_extra': search_in_address,
|
|
'fetch_fields': fetch_fields,
|
|
'mapping': mapping,
|
|
'icon': 'fa-ticket',
|
|
# for website_event main controller:
|
|
'dates': dates,
|
|
'current_date': current_date,
|
|
'search_tags': search_tags,
|
|
'no_date_domain': no_date_domain,
|
|
'no_country_domain': no_country_domain,
|
|
}
|
|
|
|
def _search_render_results(self, fetch_fields, mapping, icon, limit):
|
|
with_date = 'detail' in mapping
|
|
results_data = super()._search_render_results(fetch_fields, mapping, icon, limit)
|
|
if with_date:
|
|
for event, data in zip(self, results_data):
|
|
begin = self.env['ir.qweb.field.date'].record_to_html(event, 'date_begin', {})
|
|
end = self.env['ir.qweb.field.date'].record_to_html(event, 'date_end', {})
|
|
data['range'] = '%s🠖%s' % (begin, end) if begin != end else begin
|
|
return results_data
|