mirror of
https://github.com/bringout/oca-ocb-sale.git
synced 2026-04-27 15:32:04 +02:00
Initial commit: Sale packages
This commit is contained in:
commit
14e3d26998
6469 changed files with 2479670 additions and 0 deletions
|
|
@ -0,0 +1,2 @@
|
|||
from . import event_edit_registration
|
||||
from . import event_configurator
|
||||
|
|
@ -0,0 +1,24 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||
|
||||
from odoo import _, api, models, fields
|
||||
from odoo.exceptions import ValidationError
|
||||
|
||||
|
||||
class EventConfigurator(models.TransientModel):
|
||||
_name = 'event.event.configurator'
|
||||
_description = 'Event Configurator'
|
||||
|
||||
product_id = fields.Many2one('product.product', string="Product", readonly=True)
|
||||
event_id = fields.Many2one('event.event', string="Event")
|
||||
event_ticket_id = fields.Many2one('event.event.ticket', string="Event Ticket")
|
||||
|
||||
@api.constrains('event_id', 'event_ticket_id')
|
||||
def check_event_id(self):
|
||||
error_messages = []
|
||||
for record in self:
|
||||
if record.event_id.id != record.event_ticket_id.event_id.id:
|
||||
error_messages.append(
|
||||
_('Invalid ticket choice "%(ticket_name)s" for event "%(event_name)s".'))
|
||||
if error_messages:
|
||||
raise ValidationError('\n'.join(error_messages))
|
||||
|
|
@ -0,0 +1,47 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<odoo>
|
||||
<record id="event_configurator_view_form" model="ir.ui.view">
|
||||
<field name="name">event.configurator.view.form</field>
|
||||
<field name="model">event.event.configurator</field>
|
||||
<field name="arch" type="xml">
|
||||
<form js_class="event_configurator_form">
|
||||
<group>
|
||||
<field
|
||||
name="event_id"
|
||||
domain="[
|
||||
('event_ticket_ids.product_id','=', product_id),
|
||||
('date_end','>=',time.strftime('%Y-%m-%d 00:00:00'))
|
||||
]"
|
||||
required="1"
|
||||
context="{'name_with_seats_availability': True}"
|
||||
options="{'no_open': True, 'no_create': True}"
|
||||
/>
|
||||
<field
|
||||
name="event_ticket_id"
|
||||
domain="[('event_id', '=', event_id), ('product_id', '=', product_id)]"
|
||||
attrs="{
|
||||
'invisible': [('event_id', '=', False)],
|
||||
'required': [('event_id', '!=', False)],
|
||||
}"
|
||||
context="{'name_with_seats_availability': True}"
|
||||
options="{'no_open': True, 'no_create': True}"
|
||||
/>
|
||||
<field name="product_id" invisible="1"/>
|
||||
</group>
|
||||
<footer>
|
||||
<button string="Ok" class="btn-primary o_event_sale_js_event_configurator_ok" special="save" data-hotkey="v"/>
|
||||
<button string="Cancel" class="btn-secondary" special="cancel" data-hotkey="z"/>
|
||||
</footer>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="event_configurator_action" model="ir.actions.act_window">
|
||||
<field name="name">Configure an event</field>
|
||||
<field name="res_model">event.event.configurator</field>
|
||||
<field name="view_mode">form</field>
|
||||
<field name="target">new</field>
|
||||
<field name="context">{'name_with_seats_availability': True}</field>
|
||||
<field name="view_id" ref="event_configurator_view_form"/>
|
||||
</record>
|
||||
</odoo>
|
||||
|
|
@ -0,0 +1,132 @@
|
|||
# -*- coding: utf-8 -*-
|
||||
|
||||
from collections import Counter, defaultdict
|
||||
|
||||
from odoo import models, fields, api
|
||||
from odoo.exceptions import ValidationError
|
||||
|
||||
|
||||
class RegistrationEditor(models.TransientModel):
|
||||
_name = "registration.editor"
|
||||
_description = 'Edit Attendee Details on Sales Confirmation'
|
||||
|
||||
sale_order_id = fields.Many2one('sale.order', 'Sales Order', required=True, ondelete='cascade')
|
||||
event_registration_ids = fields.One2many('registration.editor.line', 'editor_id', string='Registrations to Edit')
|
||||
seats_available_insufficient = fields.Boolean(
|
||||
'Not enough seats for all registrations', compute='_compute_seats_available_insufficient', readonly=True)
|
||||
|
||||
@api.depends('event_registration_ids')
|
||||
def _compute_seats_available_insufficient(self):
|
||||
for editor in self:
|
||||
editor.seats_available_insufficient = False
|
||||
|
||||
events_counts = Counter()
|
||||
event_tickets_counts = defaultdict(Counter)
|
||||
|
||||
for registration in editor.event_registration_ids:
|
||||
events_counts[registration.event_id] += 1
|
||||
event_tickets_counts[registration.event_id][registration.event_ticket_id] += 1
|
||||
|
||||
for event, nb_seats_event in events_counts.items():
|
||||
# Check nb of seats in each event for all registrations on sale order
|
||||
try:
|
||||
event._check_seats_availability(nb_seats_event)
|
||||
except ValidationError:
|
||||
editor.seats_available_insufficient = True
|
||||
break
|
||||
# Check nb of seats for each ticket of the event for all registrations on sale order
|
||||
for ticket, nb_seats_ticket in event_tickets_counts[event].items():
|
||||
try:
|
||||
ticket._check_seats_availability(nb_seats_ticket)
|
||||
except ValidationError:
|
||||
editor.seats_available_insufficient = True
|
||||
break
|
||||
if editor.seats_available_insufficient:
|
||||
break
|
||||
|
||||
@api.model
|
||||
def default_get(self, fields):
|
||||
res = super(RegistrationEditor, self).default_get(fields)
|
||||
if not res.get('sale_order_id'):
|
||||
sale_order_id = res.get('sale_order_id', self._context.get('active_id'))
|
||||
res['sale_order_id'] = sale_order_id
|
||||
sale_order = self.env['sale.order'].browse(res.get('sale_order_id'))
|
||||
registrations = self.env['event.registration'].search([
|
||||
('sale_order_id', '=', sale_order.id),
|
||||
('event_ticket_id', 'in', sale_order.mapped('order_line.event_ticket_id').ids),
|
||||
('state', '!=', 'cancel')])
|
||||
|
||||
attendee_list = []
|
||||
for so_line in [l for l in sale_order.order_line if l.event_ticket_id]:
|
||||
existing_registrations = [r for r in registrations if r.event_ticket_id == so_line.event_ticket_id and r.sale_order_line_id == so_line]
|
||||
for reg in existing_registrations:
|
||||
attendee_list.append([0, 0, {
|
||||
'event_id': reg.event_id.id,
|
||||
'event_ticket_id': reg.event_ticket_id.id,
|
||||
'registration_id': reg.id,
|
||||
'name': reg.name,
|
||||
'email': reg.email,
|
||||
'phone': reg.phone,
|
||||
'mobile': reg.mobile,
|
||||
'sale_order_line_id': so_line.id,
|
||||
}])
|
||||
for count in range(int(so_line.product_uom_qty) - len(existing_registrations)):
|
||||
attendee_list.append([0, 0, {
|
||||
'event_id': so_line.event_id.id,
|
||||
'event_ticket_id': so_line.event_ticket_id.id,
|
||||
'sale_order_line_id': so_line.id,
|
||||
'name': so_line.order_partner_id.name,
|
||||
'email': so_line.order_partner_id.email,
|
||||
'phone': so_line.order_partner_id.phone,
|
||||
'mobile': so_line.order_partner_id.mobile,
|
||||
}])
|
||||
res['event_registration_ids'] = attendee_list
|
||||
res = self._convert_to_write(res)
|
||||
return res
|
||||
|
||||
def action_make_registration(self):
|
||||
self.ensure_one()
|
||||
registrations_to_create = []
|
||||
for registration_line in self.event_registration_ids:
|
||||
values = registration_line.get_registration_data()
|
||||
if registration_line.registration_id:
|
||||
registration_line.registration_id.write(values)
|
||||
else:
|
||||
registrations_to_create.append(values)
|
||||
|
||||
self.env['event.registration'].create(registrations_to_create)
|
||||
self.sale_order_id.order_line._update_registrations(
|
||||
confirm=self.sale_order_id.amount_total == 0 and not self.seats_available_insufficient)
|
||||
|
||||
return {'type': 'ir.actions.act_window_close'}
|
||||
|
||||
|
||||
class RegistrationEditorLine(models.TransientModel):
|
||||
"""Event Registration"""
|
||||
_name = "registration.editor.line"
|
||||
_description = 'Edit Attendee Line on Sales Confirmation'
|
||||
_order = "id desc"
|
||||
|
||||
editor_id = fields.Many2one('registration.editor')
|
||||
sale_order_line_id = fields.Many2one('sale.order.line', string='Sales Order Line')
|
||||
event_id = fields.Many2one('event.event', string='Event', required=True)
|
||||
registration_id = fields.Many2one('event.registration', 'Original Registration')
|
||||
event_ticket_id = fields.Many2one('event.event.ticket', string='Event Ticket')
|
||||
email = fields.Char(string='Email')
|
||||
phone = fields.Char(string='Phone')
|
||||
mobile = fields.Char(string='Mobile')
|
||||
name = fields.Char(string='Name')
|
||||
|
||||
def get_registration_data(self):
|
||||
self.ensure_one()
|
||||
return {
|
||||
'event_id': self.event_id.id,
|
||||
'event_ticket_id': self.event_ticket_id.id,
|
||||
'partner_id': self.editor_id.sale_order_id.partner_id.id,
|
||||
'name': self.name or self.editor_id.sale_order_id.partner_id.name,
|
||||
'phone': self.phone or self.editor_id.sale_order_id.partner_id.phone,
|
||||
'mobile': self.mobile or self.editor_id.sale_order_id.partner_id.mobile,
|
||||
'email': self.email or self.editor_id.sale_order_id.partner_id.email,
|
||||
'sale_order_id': self.editor_id.sale_order_id.id,
|
||||
'sale_order_line_id': self.sale_order_line_id.id,
|
||||
}
|
||||
|
|
@ -0,0 +1,48 @@
|
|||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<odoo>
|
||||
|
||||
<record id="view_event_registration_editor_form" model="ir.ui.view">
|
||||
<field name="name">registration.editor.form</field>
|
||||
<field name="model">registration.editor</field>
|
||||
<field name="arch" type="xml">
|
||||
<form string="Registration">
|
||||
<field name="seats_available_insufficient" invisible="1"/>
|
||||
<div class="alert alert-warning m-0" role="alert" attrs="{'invisible': [('seats_available_insufficient', '=', False)]}">
|
||||
<p class="my-0">
|
||||
<span>Not enough seats available. All registrations were created as "Unconfirmed" and can be updated later on.</span>
|
||||
</p>
|
||||
</div>
|
||||
<sheet>
|
||||
<p>Before updating the linked registrations of <field name="sale_order_id" readonly="1" class="oe_inline"/>
|
||||
please give attendee details.</p>
|
||||
<field name="event_registration_ids">
|
||||
<tree string="Registration" editable="top" create="false" delete="false">
|
||||
<field name="event_id" readonly='1' force_save="1"/>
|
||||
<field name="registration_id" readonly='1' force_save="1"/>
|
||||
<field name="event_ticket_id" domain="[('event_id', '=', event_id)]" readonly='1' force_save="1"/>
|
||||
<field name="name"/>
|
||||
<field name="email"/>
|
||||
<field name="mobile" class="o_force_ltr"/>
|
||||
<field name="phone" class="o_force_ltr"/>
|
||||
<field name="sale_order_line_id" invisible="1"/>
|
||||
</tree>
|
||||
</field>
|
||||
</sheet>
|
||||
<footer>
|
||||
<button string="Create/Update registrations" name="action_make_registration" type="object" class="btn-primary" data-hotkey="q"/>
|
||||
<button string="Skip" class="btn-secondary" special="cancel" data-hotkey="z"/>
|
||||
</footer>
|
||||
</form>
|
||||
</field>
|
||||
</record>
|
||||
|
||||
<record id="action_sale_order_event_registration" model="ir.actions.act_window">
|
||||
<field name="name">Event Registrations</field>
|
||||
<field name="type">ir.actions.act_window</field>
|
||||
<field name="res_model">registration.editor</field>
|
||||
<field name="view_mode">form</field>
|
||||
<field name="view_id" ref="view_event_registration_editor_form"/>
|
||||
<field name="target">new</field>
|
||||
<field name="context">{}</field>
|
||||
</record>
|
||||
</odoo>
|
||||
Loading…
Add table
Add a link
Reference in a new issue