mirror of
https://github.com/OCA/pms.git
synced 2025-01-29 00:17:45 +02:00
[WIP] Wizard Node Reservation
This commit is contained in:
@@ -20,3 +20,22 @@ class HotelRoomType(models.Model):
|
|||||||
"""
|
"""
|
||||||
free_rooms = super().check_availability_room(dfrom, dto, room_type_id, notthis)
|
free_rooms = super().check_availability_room(dfrom, dto, room_type_id, notthis)
|
||||||
return free_rooms.ids
|
return free_rooms.ids
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def get_room_type_availability(self, dfrom, dto, room_type_id):
|
||||||
|
free_rooms = self.check_availability_room(dfrom, dto)
|
||||||
|
availability_real = self.env['hotel.room'].search_count([
|
||||||
|
('id', 'in', free_rooms.ids),
|
||||||
|
('room_type_id', '=', room_type_id),
|
||||||
|
])
|
||||||
|
availability_plan = self.env['hotel.room.type.availability'].search_read([
|
||||||
|
('date', '>=', dfrom),
|
||||||
|
('date', '<', dto),
|
||||||
|
('room_type_id', '=', room_type_id),
|
||||||
|
|
||||||
|
], ['avail']) or float('inf')
|
||||||
|
|
||||||
|
if isinstance(availability_plan, list):
|
||||||
|
availability_plan = min([r['avail'] for r in availability_plan])
|
||||||
|
|
||||||
|
return min(availability_real, availability_plan)
|
||||||
|
|||||||
@@ -22,107 +22,104 @@ class HotelNodeReservationWizard(models.TransientModel):
|
|||||||
_description = "Hotel Node Reservation Wizard"
|
_description = "Hotel Node Reservation Wizard"
|
||||||
|
|
||||||
@api.model
|
@api.model
|
||||||
def _get_default_node_id(self):
|
def _default_node_id(self):
|
||||||
return self._context.get('node_id') or None
|
return self._context.get('node_id') or None
|
||||||
|
|
||||||
@api.model
|
@api.model
|
||||||
def _get_default_checkin(self):
|
def _default_checkin(self):
|
||||||
pass
|
today = fields.Date.context_today(self.with_context())
|
||||||
|
return fields.Date.from_string(today).strftime(DEFAULT_SERVER_DATE_FORMAT)
|
||||||
|
|
||||||
@api.model
|
@api.model
|
||||||
def _get_default_checkout(self):
|
def _default_checkout(self):
|
||||||
pass
|
today = fields.Date.context_today(self.with_context())
|
||||||
|
return (fields.Date.from_string(today) + timedelta(days=1)).strftime(DEFAULT_SERVER_DATE_FORMAT)
|
||||||
node_id = fields.Many2one('project.project', 'Hotel', required=True,
|
|
||||||
default=_get_default_node_id)
|
|
||||||
|
|
||||||
|
node_id = fields.Many2one('project.project', 'Hotel', required=True, default=_default_node_id)
|
||||||
partner_id = fields.Many2one('res.partner', string="Customer", required=True)
|
partner_id = fields.Many2one('res.partner', string="Customer", required=True)
|
||||||
|
checkin = fields.Date('Check In', required=True, default=_default_checkin)
|
||||||
checkin = fields.Date('Check In', required=True,
|
checkout = fields.Date('Check Out', required=True, default=_default_checkout)
|
||||||
default=_get_default_checkin)
|
|
||||||
checkout = fields.Date('Check Out', required=True,
|
|
||||||
default=_get_default_checkout)
|
|
||||||
|
|
||||||
room_type_wizard_ids = fields.One2many('node.room.type.wizard', 'node_reservation_wizard_id',
|
room_type_wizard_ids = fields.One2many('node.room.type.wizard', 'node_reservation_wizard_id',
|
||||||
string="Room Types")
|
string="Room Types")
|
||||||
price_total = fields.Float(string='Total Price', default=250.0)
|
price_total = fields.Float(string='Total Price', compute='_compute_price_total')
|
||||||
|
|
||||||
|
@api.depends('room_type_wizard_ids.price_total')
|
||||||
|
def _compute_price_total(self):
|
||||||
|
_logger.info('_compute_price_total for wizard %s', self.id)
|
||||||
|
price_total = 0.0
|
||||||
|
for record in self.room_type_wizard_ids:
|
||||||
|
price_total += record.price_total
|
||||||
|
self.price_total = price_total
|
||||||
|
|
||||||
@api.onchange('node_id')
|
@api.onchange('node_id')
|
||||||
def _onchange_node_id(self):
|
def _onchange_node_id(self):
|
||||||
# self.ensure_one()
|
self.ensure_one()
|
||||||
if self.node_id:
|
if self.node_id:
|
||||||
_logger.info('onchange_node_id(self): %s', self)
|
_logger.info('_onchange_node_id(self): %s', self)
|
||||||
try:
|
# Save your credentials (session)
|
||||||
noderpc = odoorpc.ODOO(self.node_id.odoo_host, self.node_id.odoo_protocol, self.node_id.odoo_port)
|
|
||||||
noderpc.login(self.node_id.odoo_db, self.node_id.odoo_user, self.node_id.odoo_password)
|
|
||||||
|
|
||||||
today = fields.Date.context_today(self.with_context())
|
|
||||||
|
|
||||||
# TODO check hotel timezone
|
|
||||||
checkin = fields.Date.from_string(today).strftime(
|
|
||||||
DEFAULT_SERVER_DATE_FORMAT) if not self.checkin else fields.Date.from_string(self.checkin)
|
|
||||||
|
|
||||||
checkout = (fields.Date.from_string(today) + timedelta(days=1)).strftime(
|
|
||||||
DEFAULT_SERVER_DATE_FORMAT) if not self.checkout else fields.Date.from_string(self.checkout)
|
|
||||||
|
|
||||||
free_room_ids = noderpc.env['hotel.room.type'].check_availability_room_ids(checkin, checkout)
|
|
||||||
|
|
||||||
room_type_availability = {}
|
|
||||||
for room_type in self.node_id.room_type_ids:
|
|
||||||
availability_real = noderpc.env['hotel.room'].search_count([
|
|
||||||
('id', 'in', free_room_ids),
|
|
||||||
('room_type_id', '=', room_type.remote_room_type_id),
|
|
||||||
])
|
|
||||||
availability_plan = noderpc.env['hotel.room.type.availability'].search_read([
|
|
||||||
('date', '>=', checkin),
|
|
||||||
('date', '<', checkout),
|
|
||||||
('room_type_id', '=', room_type.remote_room_type_id),
|
|
||||||
|
|
||||||
], ['avail']) or float('inf')
|
|
||||||
|
|
||||||
if isinstance(availability_plan, list):
|
|
||||||
availability_plan = min([r['avail'] for r in availability_plan])
|
|
||||||
|
|
||||||
room_type_availability[room_type.id] = min(
|
|
||||||
availability_real, availability_plan)
|
|
||||||
|
|
||||||
cmds = self.node_id.room_type_ids.mapped(lambda room_type_id: (0, False, {
|
|
||||||
'room_type_id': room_type_id.id,
|
|
||||||
'checkin': checkin,
|
|
||||||
'checkout': checkout,
|
|
||||||
'room_type_availability': room_type_availability[room_type_id.id],
|
|
||||||
'node_reservation_wizard_id': self.id,
|
|
||||||
|
|
||||||
}))
|
|
||||||
self.update({
|
|
||||||
'checkin': checkin,
|
|
||||||
'checkout': checkout,
|
|
||||||
'room_type_wizard_ids': cmds,
|
|
||||||
})
|
|
||||||
noderpc.logout()
|
|
||||||
except (odoorpc.error.RPCError, odoorpc.error.InternalError, urllib.error.URLError) as err:
|
|
||||||
raise ValidationError(err)
|
|
||||||
|
|
||||||
@api.onchange('checkin', 'checkout')
|
@api.onchange('checkin', 'checkout')
|
||||||
def _onchange_dates(self):
|
def _onchange_dates(self):
|
||||||
|
self.ensure_one()
|
||||||
_logger.info('_onchange_dates(self): %s', self)
|
_logger.info('_onchange_dates(self): %s', self)
|
||||||
|
|
||||||
today = fields.Date.context_today(self.with_context())
|
|
||||||
|
|
||||||
# TODO check hotel timezone
|
# TODO check hotel timezone
|
||||||
self.checkin = fields.Date.from_string(today).strftime(
|
self.checkin = self._get_default_checkin() if not self.checkin \
|
||||||
DEFAULT_SERVER_DATE_FORMAT) if not self.checkin else fields.Date.from_string(self.checkin)
|
else fields.Date.from_string(self.checkin)
|
||||||
|
self.checkout = self._get_default_checkout() if not self.checkout \
|
||||||
self.checkout = (fields.Date.from_string(today) + timedelta(days=1)).strftime(
|
else fields.Date.from_string(self.checkout)
|
||||||
DEFAULT_SERVER_DATE_FORMAT) if not self.checkout else fields.Date.from_string(self.checkout)
|
|
||||||
|
|
||||||
if fields.Date.from_string(self.checkin) >= fields.Date.from_string(self.checkout):
|
if fields.Date.from_string(self.checkin) >= fields.Date.from_string(self.checkout):
|
||||||
self.checkout = (fields.Date.from_string(self.checkin) + timedelta(days=1)).strftime(
|
self.checkout = (fields.Date.from_string(self.checkin) + timedelta(days=1)).strftime(
|
||||||
DEFAULT_SERVER_DATE_FORMAT)
|
DEFAULT_SERVER_DATE_FORMAT)
|
||||||
|
|
||||||
for room_type in self.room_type_wizard_ids:
|
try:
|
||||||
room_type.checkin = self.checkin
|
noderpc = odoorpc.ODOO(self.node_id.odoo_host, self.node_id.odoo_protocol, self.node_id.odoo_port)
|
||||||
room_type.checkout = self.checkout
|
noderpc.login(self.node_id.odoo_db, self.node_id.odoo_user, self.node_id.odoo_password)
|
||||||
|
|
||||||
|
# free_room_ids = noderpc.env['hotel.room.type'].check_availability_room_ids(self.checkin, self.checkout)
|
||||||
|
room_type_availability = {}
|
||||||
|
# room_type_price_unit = {}
|
||||||
|
for room_type in self.node_id.room_type_ids:
|
||||||
|
room_type_availability[room_type.id] = \
|
||||||
|
noderpc.env['hotel.room.type'].get_room_type_availability(
|
||||||
|
self.checkin, self.checkout, room_type.remote_room_type_id)
|
||||||
|
# availability_real = noderpc.env['hotel.room'].search_count([
|
||||||
|
# ('id', 'in', free_room_ids),
|
||||||
|
# ('room_type_id', '=', room_type.remote_room_type_id),
|
||||||
|
# ])
|
||||||
|
# availability_plan = noderpc.env['hotel.room.type.availability'].search_read([
|
||||||
|
# ('date', '>=', self.checkin),
|
||||||
|
# ('date', '<', self.checkout),
|
||||||
|
# ('room_type_id', '=', room_type.remote_room_type_id),
|
||||||
|
#
|
||||||
|
# ], ['avail']) or float('inf')
|
||||||
|
#
|
||||||
|
# if isinstance(availability_plan, list):
|
||||||
|
# availability_plan = min([r['avail'] for r in availability_plan])
|
||||||
|
#
|
||||||
|
# room_type_availability[room_type.id] = min(
|
||||||
|
# availability_real, availability_plan)
|
||||||
|
|
||||||
|
# room_type_price_unit[room_type.id] = noderpc.env['hotel.room.type'].search_read([
|
||||||
|
# ('id', '=', room_type.remote_room_type_id),
|
||||||
|
# ], ['list_price'])[0]['list_price']
|
||||||
|
|
||||||
|
nights = (fields.Date.from_string(self.checkout) - fields.Date.from_string(self.checkin)).days
|
||||||
|
|
||||||
|
cmds = self.node_id.room_type_ids.mapped(lambda room_type_id: (0, False, {
|
||||||
|
'room_type_id': room_type_id.id,
|
||||||
|
'checkin': self.checkin,
|
||||||
|
'checkout': self.checkout,
|
||||||
|
'nights': nights,
|
||||||
|
'room_type_availability': room_type_availability[room_type_id.id],
|
||||||
|
# 'price_unit': room_type_price_unit[room_type_id.id],
|
||||||
|
'node_reservation_wizard_id': self.id,
|
||||||
|
}))
|
||||||
|
self.room_type_wizard_ids = cmds
|
||||||
|
|
||||||
|
noderpc.logout()
|
||||||
|
|
||||||
|
except (odoorpc.error.RPCError, odoorpc.error.InternalError, urllib.error.URLError) as err:
|
||||||
|
raise ValidationError(err)
|
||||||
|
|
||||||
@api.multi
|
@api.multi
|
||||||
def create_node_reservation(self):
|
def create_node_reservation(self):
|
||||||
@@ -143,6 +140,7 @@ class HotelNodeReservationWizard(models.TransientModel):
|
|||||||
'partner_id': remote_partner_id,
|
'partner_id': remote_partner_id,
|
||||||
'room_type_id': line.room_type_id.remote_room_type_id,
|
'room_type_id': line.room_type_id.remote_room_type_id,
|
||||||
}
|
}
|
||||||
|
# añadir descuento
|
||||||
reservation_line_ids = noderpc.env['hotel.reservation'].prepare_reservation_lines(
|
reservation_line_ids = noderpc.env['hotel.reservation'].prepare_reservation_lines(
|
||||||
line.checkin,
|
line.checkin,
|
||||||
(fields.Date.from_string(line.checkout) - fields.Date.from_string(line.checkin)).days,
|
(fields.Date.from_string(line.checkout) - fields.Date.from_string(line.checkin)).days,
|
||||||
@@ -173,16 +171,6 @@ class NodeRoomTypeWizard(models.TransientModel):
|
|||||||
_name = "node.room.type.wizard"
|
_name = "node.room.type.wizard"
|
||||||
_description = "Node Room Type Wizard"
|
_description = "Node Room Type Wizard"
|
||||||
|
|
||||||
def _default_checkin(self):
|
|
||||||
today = fields.Date.context_today(self.with_context())
|
|
||||||
return self.node_reservation_wizard_id.checkin or \
|
|
||||||
fields.Date.from_string(today).strftime(DEFAULT_SERVER_DATE_FORMAT)
|
|
||||||
|
|
||||||
def _default_checkout(self):
|
|
||||||
today = fields.Date.context_today(self.with_context())
|
|
||||||
return self.node_reservation_wizard_id.checkin or \
|
|
||||||
(fields.Date.from_string(today) + timedelta(days=1)).strftime(DEFAULT_SERVER_DATE_FORMAT)
|
|
||||||
|
|
||||||
node_reservation_wizard_id = fields.Many2one('hotel.node.reservation.wizard')
|
node_reservation_wizard_id = fields.Many2one('hotel.node.reservation.wizard')
|
||||||
node_id = fields.Many2one(related='node_reservation_wizard_id.node_id')
|
node_id = fields.Many2one(related='node_reservation_wizard_id.node_id')
|
||||||
|
|
||||||
@@ -191,28 +179,49 @@ class NodeRoomTypeWizard(models.TransientModel):
|
|||||||
room_type_availability = fields.Integer('Availability') #, compute="_compute_room_type_availability")
|
room_type_availability = fields.Integer('Availability') #, compute="_compute_room_type_availability")
|
||||||
room_qty = fields.Integer('Quantity', default=0)
|
room_qty = fields.Integer('Quantity', default=0)
|
||||||
|
|
||||||
checkin = fields.Date('Check In', required=True, default=_default_checkin)
|
checkin = fields.Date('Check In', required=True)
|
||||||
checkout = fields.Date('Check Out', required=True, default=_default_checkout)
|
checkout = fields.Date('Check Out', required=True)
|
||||||
|
nights = fields.Integer('Nights', readonly=True)
|
||||||
|
min_stay = fields.Integer('Min. Days', compute="_compute_restrictions", readonly=True)
|
||||||
|
|
||||||
price_unit = fields.Float(string='Room Price', required=True, default=0.0)
|
price_unit = fields.Float(string='Room Price', required=True, default=0.0)
|
||||||
discount = fields.Float(string='Discount (%)', default=0.0)
|
discount = fields.Float(string='Discount (%)', default=0.0)
|
||||||
price_total = fields.Float(string='Total Price', compute="_compute_price_total")
|
price_total = fields.Float(string='Total Price', compute='_compute_price_total')
|
||||||
|
|
||||||
# compute and search fields, in the same order that fields declaration
|
@api.depends('room_qty', 'price_unit', 'discount')
|
||||||
@api.onchange('checkin','checkout')
|
def _compute_price_total(self):
|
||||||
|
for room_type in self:
|
||||||
|
_logger.info('_compute_price_total for room type %s', room_type.room_type_id)
|
||||||
|
# noderpc = odoorpc.ODOO(self.node_id.odoo_host, self.node_id.odoo_protocol, self.node_id.odoo_port)
|
||||||
|
# noderpc.login(self.node_id.odoo_db, self.node_id.odoo_user, self.node_id.odoo_password)
|
||||||
|
# self.price_unit = noderpc.env['hotel.room.type'].search_read([
|
||||||
|
# ('id', '=', self.room_type_id.remote_room_type_id),
|
||||||
|
# ], ['list_price'])[0]['list_price']
|
||||||
|
# noderpc.logout()
|
||||||
|
|
||||||
|
room_type.price_total = (room_type.room_qty * room_type.price_unit * room_type.nights) * (1.0 - room_type.discount * 0.01)
|
||||||
|
# Unidades x precio unidad (el precio de unidad ya incluye el conjunto de días)
|
||||||
|
|
||||||
|
@api.depends('checkin', 'checkout')
|
||||||
|
def _compute_restrictions(self):
|
||||||
|
for room_type in self:
|
||||||
|
_logger.info('_compute_restrictions for room type %s', room_type.room_type_id)
|
||||||
|
|
||||||
|
@api.onchange('checkin', 'checkout')
|
||||||
def _onchange_dates(self):
|
def _onchange_dates(self):
|
||||||
_logger.info('_onchange_dates for room type %s', self.room_type_id)
|
_logger.info('_onchange_dates for room type %s', self.room_type_id)
|
||||||
wdb.set_trace()
|
# recompute price unit
|
||||||
# if self.checkin and self.checkout:
|
self.checkin = self._default_checkin() \
|
||||||
|
if not self.checkin else fields.Date.from_string(self.checkin)
|
||||||
|
self.checkout = self._default_checkout() \
|
||||||
|
if not self.checkout else fields.Date.from_string(self.checkout)
|
||||||
|
if fields.Date.from_string(self.checkin) >= fields.Date.from_string(self.checkout):
|
||||||
|
self.checkout = (fields.Date.from_string(self.checkin) + timedelta(days=1)).strftime(
|
||||||
|
DEFAULT_SERVER_DATE_FORMAT)
|
||||||
|
|
||||||
|
self.nights = (fields.Date.from_string(self.checkout) - fields.Date.from_string(self.checkin)).days
|
||||||
|
|
||||||
# Conectar con nodo para traer dispo(availability) y precio por habitación(price_unit)
|
# Conectar con nodo para traer dispo(availability) y precio por habitación(price_unit)
|
||||||
# availability: search de hotel.room.type.availability filtrando por room_type y date y escogiendo el min avail en el rango
|
# availability: search de hotel.room.type.availability filtrando por room_type y date y escogiendo el min avail en el rango
|
||||||
# preci_unit y json_days: usando prepare_reservation_lines
|
# preci_unit y json_days: usando prepare_reservation_lines
|
||||||
|
|
||||||
@api.depends('room_qty', 'price_unit', 'discount', 'checkin', 'checkout')
|
|
||||||
def _compute_price_total(self):
|
|
||||||
_logger.info('_compute_price_total')
|
|
||||||
for record in self:
|
|
||||||
record.price_total = (record.room_qty * record.price_unit) * (1.0 - record.discount * 0.01)
|
|
||||||
# Unidades x precio unidad (el precio de unidad ya incluye el conjunto de días)
|
|
||||||
|
|
||||||
|
|||||||
@@ -30,6 +30,8 @@
|
|||||||
<field name="room_qty"/>
|
<field name="room_qty"/>
|
||||||
<field name="checkin" widget="date" />
|
<field name="checkin" widget="date" />
|
||||||
<field name="checkout" widget="date" />
|
<field name="checkout" widget="date" />
|
||||||
|
<field name="nights"/>
|
||||||
|
<field name="min_stay"/>
|
||||||
<field name="price_unit" widget="monetary"/>
|
<field name="price_unit" widget="monetary"/>
|
||||||
<!--<field name="currency_id" invisible="1"/>-->
|
<!--<field name="currency_id" invisible="1"/>-->
|
||||||
<field name="discount"/>
|
<field name="discount"/>
|
||||||
|
|||||||
Reference in New Issue
Block a user