mirror of
https://github.com/OCA/pms.git
synced 2025-01-29 00:17:45 +02:00
1038 lines
37 KiB
Python
1038 lines
37 KiB
Python
# Copyright 2017 Dario Lodeiros
|
|
# Copyright 2018 Alexandre Diaz
|
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
|
|
|
import json
|
|
from datetime import datetime
|
|
|
|
from dateutil.relativedelta import relativedelta
|
|
|
|
from odoo import _, api, fields, models
|
|
from odoo.exceptions import UserError, ValidationError
|
|
from odoo.tools import DEFAULT_SERVER_DATE_FORMAT
|
|
from odoo.tools.safe_eval import safe_eval
|
|
|
|
|
|
class PmsCheckinPartner(models.Model):
|
|
_name = "pms.checkin.partner"
|
|
_description = "Partner Checkins"
|
|
_inherit = ["mail.thread", "mail.activity.mixin", "portal.mixin"]
|
|
_rec_name = "identifier"
|
|
|
|
identifier = fields.Char(
|
|
string="Identifier",
|
|
help="Checkin Partner Id",
|
|
readonly=True,
|
|
index=True,
|
|
default=lambda self: _("New"),
|
|
)
|
|
partner_id = fields.Many2one(
|
|
string="Partner",
|
|
help="Partner associated with checkin partner",
|
|
readonly=False,
|
|
index=True,
|
|
store=True,
|
|
comodel_name="res.partner",
|
|
domain="[('is_company', '=', False)]",
|
|
compute="_compute_partner_id",
|
|
)
|
|
reservation_id = fields.Many2one(
|
|
string="Reservation",
|
|
help="Reservation to which checkin partners belong",
|
|
index=True,
|
|
comodel_name="pms.reservation",
|
|
)
|
|
folio_id = fields.Many2one(
|
|
string="Folio",
|
|
help="Folio to which reservation of checkin partner belongs",
|
|
store=True,
|
|
index=True,
|
|
comodel_name="pms.folio",
|
|
related="reservation_id.folio_id",
|
|
)
|
|
pms_property_id = fields.Many2one(
|
|
string="Property",
|
|
help="Property to which the folio associated belongs",
|
|
readonly=True,
|
|
store=True,
|
|
index=True,
|
|
comodel_name="pms.property",
|
|
related="reservation_id.pms_property_id",
|
|
)
|
|
name = fields.Char(
|
|
string="Name", help="Checkin partner name", related="partner_id.name"
|
|
)
|
|
email = fields.Char(
|
|
string="E-mail",
|
|
help="Checkin Partner Email",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_email",
|
|
)
|
|
mobile = fields.Char(
|
|
string="Mobile",
|
|
help="Checkin Partner Mobile",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_mobile",
|
|
)
|
|
phone = fields.Char(
|
|
string="Phone",
|
|
help="Checkin Partner Phone",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_phone",
|
|
)
|
|
image_128 = fields.Image(
|
|
string="Image",
|
|
help="Checkin Partner Image, it corresponds with Partner Image associated",
|
|
related="partner_id.image_128",
|
|
)
|
|
segmentation_ids = fields.Many2many(
|
|
string="Segmentation",
|
|
help="Segmentation tags to classify checkin partners",
|
|
readonly=True,
|
|
related="reservation_id.segmentation_ids",
|
|
domain="[('is_used_in_checkin', '=', True)]",
|
|
)
|
|
checkin = fields.Date(
|
|
string="Checkin",
|
|
help="Checkin date",
|
|
store=True,
|
|
related="reservation_id.checkin",
|
|
depends=["reservation_id.checkin"],
|
|
)
|
|
checkout = fields.Date(
|
|
string="Checkout",
|
|
help="Checkout date",
|
|
store=True,
|
|
related="reservation_id.checkout",
|
|
depends=["reservation_id.checkout"],
|
|
)
|
|
arrival = fields.Datetime("Enter", help="Checkin partner arrival date and time")
|
|
departure = fields.Datetime(
|
|
string="Exit", help="Checkin partner departure date and time"
|
|
)
|
|
state = fields.Selection(
|
|
string="Status",
|
|
help="Status of the checkin partner regarding the reservation",
|
|
readonly=True,
|
|
store=True,
|
|
selection=[
|
|
("dummy", "Unkown Guest"),
|
|
("draft", "Incomplete data"),
|
|
("precheckin", "Pending arrival"),
|
|
("onboard", "On Board"),
|
|
("done", "Out"),
|
|
("cancel", "Cancelled"),
|
|
],
|
|
compute="_compute_state",
|
|
)
|
|
|
|
gender = fields.Selection(
|
|
string="Gender",
|
|
help="host gender",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_gender",
|
|
selection=[("male", "Male"), ("female", "Female"), ("other", "Other")],
|
|
)
|
|
nationality_id = fields.Many2one(
|
|
string="Nationality",
|
|
help="host nationality",
|
|
readonly=False,
|
|
store=True,
|
|
index=True,
|
|
compute="_compute_nationality_id",
|
|
comodel_name="res.country",
|
|
)
|
|
residence_street = fields.Char(
|
|
string="Street",
|
|
help="Street of the guest's residence",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_residence_street",
|
|
)
|
|
residence_street2 = fields.Char(
|
|
string="Street2",
|
|
help="Second street of the guest's residence",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_residence_street2",
|
|
)
|
|
residence_zip = fields.Char(
|
|
string="Zip",
|
|
help="Zip of the guest's residence",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_residence_zip",
|
|
change_default=True,
|
|
)
|
|
residence_city = fields.Char(
|
|
string="City",
|
|
help="City of the guest's residence",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_residence_city",
|
|
)
|
|
residence_country_id = fields.Many2one(
|
|
string="Country of residence",
|
|
help="Country of the guest's residence",
|
|
readonly=False,
|
|
store=True,
|
|
index=True,
|
|
compute="_compute_residence_country_id",
|
|
comodel_name="res.country",
|
|
)
|
|
residence_state_id = fields.Many2one(
|
|
string="State of residence",
|
|
help="State of the guest's residence",
|
|
readonly=False,
|
|
store=True,
|
|
index=True,
|
|
compute="_compute_residence_state_id",
|
|
comodel_name="res.country.state",
|
|
)
|
|
|
|
firstname = fields.Char(
|
|
string="First Name",
|
|
help="host firstname",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_firstname",
|
|
)
|
|
lastname = fields.Char(
|
|
string="Last Name",
|
|
help="host lastname",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_lastname",
|
|
)
|
|
lastname2 = fields.Char(
|
|
string="Second Last Name",
|
|
help="host second lastname",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_lastname2",
|
|
)
|
|
birthdate_date = fields.Date(
|
|
string="Birthdate",
|
|
help="host birthdate",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_birth_date",
|
|
)
|
|
document_number = fields.Char(
|
|
string="Document Number",
|
|
help="Host document number",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_document_number",
|
|
)
|
|
document_type = fields.Many2one(
|
|
string="Document Type",
|
|
help="Select a valid document type",
|
|
readonly=False,
|
|
store=True,
|
|
index=True,
|
|
comodel_name="res.partner.id_category",
|
|
compute="_compute_document_type",
|
|
domain="['|', ('country_ids', '=', False), ('country_ids', 'in', document_country_id)]",
|
|
)
|
|
document_expedition_date = fields.Date(
|
|
string="Expedition Date",
|
|
help="Date on which document_type was issued",
|
|
readonly=False,
|
|
store=True,
|
|
compute="_compute_document_expedition_date",
|
|
)
|
|
|
|
document_id = fields.Many2one(
|
|
string="Document",
|
|
help="Technical field",
|
|
readonly=False,
|
|
store=True,
|
|
index=True,
|
|
comodel_name="res.partner.id_number",
|
|
compute="_compute_document_id",
|
|
ondelete="restrict",
|
|
)
|
|
|
|
document_country_id = fields.Many2one(
|
|
string="Document Country",
|
|
help="Country of the document",
|
|
comodel_name="res.country",
|
|
compute="_compute_document_country_id",
|
|
store=True,
|
|
readonly=False,
|
|
)
|
|
|
|
partner_incongruences = fields.Char(
|
|
string="partner_incongruences",
|
|
help="indicates that some partner fields \
|
|
on the checkin do not correspond to that of \
|
|
the associated partner",
|
|
compute="_compute_partner_incongruences",
|
|
)
|
|
|
|
possible_existing_customer_ids = fields.One2many(
|
|
string="Possible existing customer",
|
|
compute="_compute_possible_existing_customer_ids",
|
|
comodel_name="res.partner",
|
|
inverse_name="checkin_partner_possible_customer_id",
|
|
)
|
|
|
|
partner_relationship = fields.Char(
|
|
string="Partner relationship", help="Family relationship between travelers"
|
|
)
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_document_number(self):
|
|
for record in self:
|
|
if not record.document_number and record.partner_id.id_numbers:
|
|
last_update_document = record.partner_id.id_numbers.filtered(
|
|
lambda x: x.write_date
|
|
== max(record.partner_id.id_numbers.mapped("write_date"))
|
|
)
|
|
if last_update_document and last_update_document[0].name:
|
|
record.document_number = last_update_document[0].name
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_document_type(self):
|
|
for record in self:
|
|
if not record.document_type and record.partner_id.id_numbers:
|
|
last_update_document = record.partner_id.id_numbers.filtered(
|
|
lambda x: x.write_date
|
|
== max(record.partner_id.id_numbers.mapped("write_date"))
|
|
)
|
|
if last_update_document and last_update_document[0].category_id:
|
|
record.document_type = last_update_document[0].category_id
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_document_expedition_date(self):
|
|
for record in self:
|
|
if not record.document_expedition_date and record.partner_id.id_numbers:
|
|
last_update_document = record.partner_id.id_numbers.filtered(
|
|
lambda x: x.write_date
|
|
== max(record.partner_id.id_numbers.mapped("write_date"))
|
|
)
|
|
if last_update_document and last_update_document[0].valid_from:
|
|
record.document_expedition_date = last_update_document[0].valid_from
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_document_country_id(self):
|
|
for record in self:
|
|
if not record.document_country_id and record.partner_id.id_numbers:
|
|
last_update_document = record.partner_id.id_numbers.filtered(
|
|
lambda x: x.write_date
|
|
== max(record.partner_id.id_numbers.mapped("write_date"))
|
|
)
|
|
if last_update_document and last_update_document[0].country_id:
|
|
record.document_country_id = last_update_document[0].country_id
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_firstname(self):
|
|
for record in self:
|
|
if not record.firstname and record.partner_id.firstname:
|
|
record.firstname = record.partner_id.firstname
|
|
elif not record.firstname:
|
|
record.firstname = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_lastname(self):
|
|
for record in self:
|
|
if not record.lastname and record.partner_id.lastname:
|
|
record.lastname = record.partner_id.lastname
|
|
elif not record.lastname:
|
|
record.lastname = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_lastname2(self):
|
|
for record in self:
|
|
if not record.lastname2 and record.partner_id.lastname2:
|
|
record.lastname2 = record.partner_id.lastname2
|
|
elif not record.lastname2:
|
|
record.lastname2 = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_birth_date(self):
|
|
for record in self:
|
|
if not record.birthdate_date and record.partner_id.birthdate_date:
|
|
record.birthdate_date = record.partner_id.birthdate_date
|
|
elif not record.birthdate_date:
|
|
record.birthdate_date = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_gender(self):
|
|
for record in self:
|
|
if not record.gender and record.partner_id.gender:
|
|
record.gender = record.partner_id.gender
|
|
elif not record.gender:
|
|
record.gender = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_nationality_id(self):
|
|
for record in self:
|
|
if not record.nationality_id and record.partner_id.nationality_id:
|
|
record.nationality_id = record.partner_id.nationality_id
|
|
elif not record.nationality_id:
|
|
record.nationality_id = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_residence_street(self):
|
|
for record in self:
|
|
if not record.residence_street and record.partner_id.residence_street:
|
|
record.residence_street = record.partner_id.residence_street
|
|
elif not record.residence_street:
|
|
record.residence_street = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_residence_street2(self):
|
|
for record in self:
|
|
if not record.residence_street2 and record.partner_id.residence_street2:
|
|
record.residence_street2 = record.partner_id.residence_street2
|
|
elif not record.residence_street2:
|
|
record.residence_street2 = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_residence_zip(self):
|
|
for record in self:
|
|
if not record.residence_zip and record.partner_id.residence_zip:
|
|
record.residence_zip = record.partner_id.residence_zip
|
|
elif not record.residence_zip:
|
|
record.residence_zip = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_residence_city(self):
|
|
for record in self:
|
|
if not record.residence_city and record.partner_id.residence_city:
|
|
record.residence_city = record.partner_id.residence_city
|
|
elif not record.residence_city:
|
|
record.residence_city = False
|
|
|
|
@api.depends("partner_id", "nationality_id")
|
|
def _compute_residence_country_id(self):
|
|
for record in self:
|
|
if (
|
|
not record.residence_country_id
|
|
and record.partner_id.residence_country_id
|
|
):
|
|
record.residence_country_id = record.partner_id.residence_country_id
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_residence_state_id(self):
|
|
for record in self:
|
|
if not record.residence_state_id and record.partner_id.residence_state_id:
|
|
record.residence_state_id = record.partner_id.residence_state_id
|
|
elif not record.residence_state_id:
|
|
record.residence_state_id = False
|
|
|
|
@api.depends(lambda self: self._checkin_manual_fields(depends=True))
|
|
def _compute_state(self):
|
|
for record in self:
|
|
if not record.state:
|
|
record.state = "dummy"
|
|
if record.reservation_id.state == "cancel":
|
|
record.state = "cancel"
|
|
elif record.state in ("dummy", "draft", "precheckin", "cancel"):
|
|
if all(
|
|
not getattr(record, field)
|
|
for field in record._checkin_manual_fields()
|
|
):
|
|
record.state = "dummy"
|
|
elif any(
|
|
not getattr(record, field)
|
|
for field in record._checkin_mandatory_fields(
|
|
country=record.document_country_id
|
|
)
|
|
):
|
|
record.state = "draft"
|
|
else:
|
|
record.state = "precheckin"
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_name(self):
|
|
for record in self:
|
|
if not record.name or record.partner_id.name:
|
|
record.name = record.partner_id.name
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_email(self):
|
|
for record in self:
|
|
if not record.email and record.partner_id.email:
|
|
record.email = record.partner_id.email
|
|
elif not record.email:
|
|
record.email = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_mobile(self):
|
|
for record in self:
|
|
if not record.mobile and record.partner_id.mobile:
|
|
record.mobile = record.partner_id.mobile
|
|
elif not record.mobile:
|
|
record.mobile = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_phone(self):
|
|
for record in self:
|
|
if not record.phone and record.partner_id.phone:
|
|
record.phone = record.partner_id.phone
|
|
elif not record.phone:
|
|
record.phone = False
|
|
|
|
@api.depends("partner_id")
|
|
def _compute_document_id(self):
|
|
for record in self:
|
|
if record.partner_id:
|
|
if (
|
|
not record.document_id
|
|
and record.document_number
|
|
and record.document_type
|
|
):
|
|
id_number_id = (
|
|
self.sudo()
|
|
.env["res.partner.id_number"]
|
|
.search(
|
|
[
|
|
("partner_id", "=", record.partner_id.id),
|
|
("name", "=", record.document_number),
|
|
("category_id", "=", record.document_type.id),
|
|
]
|
|
)
|
|
)
|
|
if not id_number_id:
|
|
id_number_id = self.env["res.partner.id_number"].create(
|
|
{
|
|
"partner_id": record.partner_id.id,
|
|
"name": record.document_number,
|
|
"category_id": record.document_type.id,
|
|
"valid_from": record.document_expedition_date,
|
|
}
|
|
)
|
|
|
|
record.document_id = id_number_id
|
|
else:
|
|
record.document_id = False
|
|
|
|
@api.depends(
|
|
"document_number",
|
|
"document_type",
|
|
"firstname",
|
|
"lastname",
|
|
"lastname2",
|
|
)
|
|
def _compute_partner_id(self):
|
|
for record in self:
|
|
if not record.partner_id:
|
|
if record.document_number and record.document_type:
|
|
partner = self._get_partner_by_document(
|
|
record.document_number, record.document_type
|
|
)
|
|
if not partner:
|
|
if record.firstname or record.lastname or record.lastname2:
|
|
partner_values = {
|
|
"firstname": record.firstname,
|
|
"lastname": record.lastname,
|
|
"lastname2": record.lastname2,
|
|
"gender": record.gender,
|
|
"birthdate_date": record.birthdate_date,
|
|
"nationality_id": record.nationality_id.id,
|
|
}
|
|
partner = (
|
|
self.env["res.partner"]
|
|
.with_context(avoid_document_restriction=True)
|
|
.create(partner_values)
|
|
)
|
|
record.partner_id = partner
|
|
|
|
@api.model
|
|
def _get_partner_by_document(self, document_number, document_type):
|
|
number = (
|
|
self.sudo()
|
|
.env["res.partner.id_number"]
|
|
.search(
|
|
[
|
|
("name", "=", document_number),
|
|
("category_id", "=", document_type.id),
|
|
]
|
|
)
|
|
)
|
|
return (
|
|
self.sudo().env["res.partner"].search([("id", "=", number.partner_id.id)])
|
|
)
|
|
|
|
@api.depends("email", "mobile")
|
|
def _compute_possible_existing_customer_ids(self):
|
|
for record in self:
|
|
possible_customer = self.env[
|
|
"pms.folio"
|
|
]._apply_possible_existing_customer_ids(record.email, record.mobile)
|
|
if possible_customer:
|
|
record.possible_existing_customer_ids = possible_customer
|
|
else:
|
|
record.possible_existing_customer_ids = False
|
|
|
|
@api.depends(
|
|
"firstname",
|
|
"lastname",
|
|
"lastname2",
|
|
"gender",
|
|
"birthdate_date",
|
|
"nationality_id",
|
|
"email",
|
|
"mobile",
|
|
"partner_id",
|
|
)
|
|
def _compute_partner_incongruences(self):
|
|
for record in self:
|
|
incongruous_fields = False
|
|
if record.partner_id:
|
|
for field in record._checkin_partner_fields():
|
|
if (
|
|
record.partner_id[field]
|
|
and record.partner_id[field] != record[field]
|
|
):
|
|
if not incongruous_fields:
|
|
incongruous_fields = record._fields[field].string
|
|
else:
|
|
incongruous_fields += ", " + record._fields[field].string
|
|
if incongruous_fields:
|
|
record.partner_incongruences = (
|
|
incongruous_fields + " field/s don't correspond to saved host"
|
|
)
|
|
else:
|
|
record.partner_incongruences = False
|
|
else:
|
|
record.partner_incongruences = False
|
|
|
|
def _compute_access_url(self):
|
|
super(PmsCheckinPartner, self)._compute_access_url()
|
|
for checkin in self:
|
|
checkin.access_url = "/my/folios/%s/reservations/%s/checkins/%s" % (
|
|
checkin.folio_id.id,
|
|
checkin.reservation_id.id,
|
|
checkin.id,
|
|
)
|
|
|
|
# Constraints and onchanges
|
|
|
|
@api.constrains("departure", "arrival")
|
|
def _check_departure(self):
|
|
for record in self:
|
|
if record.departure and record.arrival > record.departure:
|
|
raise ValidationError(
|
|
_("Departure date (%s) is prior to arrival on %s")
|
|
% (record.departure, record.arrival)
|
|
)
|
|
|
|
@api.constrains("partner_id")
|
|
def _check_partner_id(self):
|
|
for record in self:
|
|
if record.partner_id:
|
|
indoor_partner_ids = record.reservation_id.checkin_partner_ids.filtered(
|
|
lambda r: r.id != record.id
|
|
).mapped("partner_id.id")
|
|
if indoor_partner_ids.count(record.partner_id.id) > 1:
|
|
record.partner_id = None
|
|
raise ValidationError(
|
|
_("This guest is already registered in the room")
|
|
)
|
|
|
|
# REVIEW: Redesign email & mobile control (res.partner? other module in OCA?)
|
|
# @api.constrains("email")
|
|
# def check_email_pattern(self):
|
|
# for record in self:
|
|
# if record.email:
|
|
# if not re.search(
|
|
# r"^[a-zA-Z0-9]([a-zA-z0-9\-\_]*[\.]?[a-zA-Z0-9\-\_]+)*"
|
|
# r"@([a-zA-z0-9\-]+([\.][a-zA-Z0-9\-\_]+)?\.[a-zA-Z0-9]+)+$",
|
|
# record.email,
|
|
# ):
|
|
# raise ValidationError(_("'%s' is not a valid email", record.email))
|
|
|
|
# @api.constrains("mobile")
|
|
# def check_phone_pattern(self):
|
|
|
|
# for record in self:
|
|
# if record.mobile:
|
|
|
|
# if not re.search(
|
|
# r"^(\d{3}[\-\s]?\d{2}[\-\s]?\d{2}[\-\s]?\d{2}[\-\s]?|"
|
|
# r"\d{3}[\-\s]?\d{3}[\-\s]?\d{3})$",
|
|
# str(record.mobile),
|
|
# ):
|
|
# raise ValidationError(_("'%s' is not a valid phone", record.mobile))
|
|
|
|
@api.constrains("document_number")
|
|
def check_document_number(self):
|
|
for record in self:
|
|
if record.partner_id:
|
|
for number in record.partner_id.id_numbers:
|
|
if record.document_type == number.category_id:
|
|
if record.document_number != number.name:
|
|
raise ValidationError(_("Document_type has already exists"))
|
|
|
|
def _validation_eval_context(self, id_number):
|
|
self.ensure_one()
|
|
return {"self": self, "id_number": id_number}
|
|
|
|
@api.constrains("document_number", "document_type")
|
|
def validate_id_number(self):
|
|
"""Validate the given ID number
|
|
The method raises an odoo.exceptions.ValidationError if the eval of
|
|
python validation code fails
|
|
"""
|
|
for record in self:
|
|
if record.document_number and record.document_type:
|
|
id_number = self.env["res.partner.id_number"].new(
|
|
{
|
|
"name": record.document_number,
|
|
"category_id": record.document_type,
|
|
}
|
|
)
|
|
if (
|
|
self.env.context.get("id_no_validate")
|
|
or not record.document_type.validation_code
|
|
):
|
|
return
|
|
eval_context = record._validation_eval_context(id_number)
|
|
try:
|
|
safe_eval(
|
|
record.document_type.validation_code,
|
|
eval_context,
|
|
mode="exec",
|
|
nocopy=True,
|
|
)
|
|
except Exception as e:
|
|
raise UserError(
|
|
_(
|
|
"Error when evaluating the id_category validation code:"
|
|
":\n %s \n(%s)"
|
|
)
|
|
% (self.name, e)
|
|
)
|
|
if eval_context.get("failed", False):
|
|
raise ValidationError(
|
|
_("%s is not a valid %s identifier")
|
|
% (record.document_number, record.document_type.name)
|
|
)
|
|
|
|
@api.constrains("document_country_id", "document_type")
|
|
def _check_document_country_id_document_type_consistence(self):
|
|
for record in self:
|
|
if record.document_country_id and record.document_type:
|
|
if (
|
|
record.document_type.country_ids
|
|
and record.document_country_id
|
|
not in record.document_type.country_ids
|
|
):
|
|
raise ValidationError(
|
|
_("Document type and country of document do not match")
|
|
)
|
|
|
|
@api.model
|
|
def create(self, vals):
|
|
# The checkin records are created automatically from adult depends
|
|
# if you try to create one manually, we update one unassigned checkin
|
|
reservation_id = vals.get("reservation_id")
|
|
if reservation_id:
|
|
reservation = self.env["pms.reservation"].browse(reservation_id)
|
|
else:
|
|
raise ValidationError(
|
|
_("Is mandatory indicate the reservation on the checkin")
|
|
)
|
|
# If a checkin is manually created, we need make sure that
|
|
# the reservation adults are computed
|
|
if not reservation.checkin_partner_ids:
|
|
reservation.flush()
|
|
dummy_checkins = reservation.checkin_partner_ids.filtered(
|
|
lambda c: c.state == "dummy"
|
|
)
|
|
if len(reservation.checkin_partner_ids) < reservation.adults:
|
|
return super(PmsCheckinPartner, self).create(vals)
|
|
if len(dummy_checkins) > 0:
|
|
dummy_checkins[0].write(vals)
|
|
return dummy_checkins[0]
|
|
raise ValidationError(
|
|
_("Is not possible to create the proposed check-in in this reservation")
|
|
)
|
|
|
|
def unlink(self):
|
|
reservations = self.mapped("reservation_id")
|
|
res = super().unlink()
|
|
reservations._compute_checkin_partner_ids()
|
|
return res
|
|
|
|
@api.model
|
|
def _checkin_manual_fields(self, country=False, depends=False):
|
|
manual_fields = [
|
|
"name",
|
|
"partner_id",
|
|
"email",
|
|
"mobile",
|
|
"phone",
|
|
"gender",
|
|
"firstname",
|
|
"lastname",
|
|
"lastname2",
|
|
"birthdate_date",
|
|
"document_number",
|
|
"document_expedition_date",
|
|
"nationality_id",
|
|
"residence_street",
|
|
"residence_street2",
|
|
"residence_zip",
|
|
"residence_city",
|
|
"residence_country_id",
|
|
"residence_state_id",
|
|
]
|
|
# api.depends need "reservation_id.state" in the lambda function
|
|
if depends:
|
|
manual_fields.append("reservation_id.state")
|
|
return manual_fields
|
|
|
|
@api.model
|
|
def _checkin_mandatory_fields(self, country=False, depends=False):
|
|
mandatory_fields = [
|
|
"name",
|
|
]
|
|
# api.depends need "reservation_id.state" in the lambda function
|
|
if depends:
|
|
mandatory_fields.extend(["reservation_id.state", "name"])
|
|
|
|
return mandatory_fields
|
|
|
|
@api.model
|
|
def _checkin_partner_fields(self):
|
|
checkin_fields = [
|
|
"firstname",
|
|
"lastname",
|
|
"lastname2",
|
|
"mobile",
|
|
"email",
|
|
"gender",
|
|
"nationality_id",
|
|
"birthdate_date",
|
|
]
|
|
return checkin_fields
|
|
|
|
@api.model
|
|
def import_room_list_json(self, roomlist_json):
|
|
roomlist_json = json.loads(roomlist_json)
|
|
for checkin_dict in roomlist_json:
|
|
identifier = checkin_dict["identifier"]
|
|
reservation_id = checkin_dict["reservation_id"]
|
|
checkin = (
|
|
self.sudo()
|
|
.env["pms.checkin.partner"]
|
|
.search([("identifier", "=", identifier)])
|
|
)
|
|
reservation = self.env["pms.reservation"].browse(reservation_id)
|
|
if not checkin:
|
|
raise ValidationError(
|
|
_("%s not found in checkins (%s)"), identifier, reservation.name
|
|
)
|
|
checkin_vals = {}
|
|
for key, value in checkin_dict.items():
|
|
if key in ("reservation_id", "folio_id", "identifier"):
|
|
continue
|
|
checkin_vals[key] = value
|
|
checkin.write(checkin_vals)
|
|
|
|
@api.model
|
|
def calculate_doc_type_expedition_date_from_validity_date(
|
|
self, doc_type, doc_date, birthdate
|
|
):
|
|
today = fields.datetime.today()
|
|
datetime_doc_date = datetime.strptime(doc_date, DEFAULT_SERVER_DATE_FORMAT)
|
|
if datetime_doc_date < today:
|
|
return datetime_doc_date
|
|
datetime_birthdate = datetime.strptime(birthdate, DEFAULT_SERVER_DATE_FORMAT)
|
|
age = today.year - datetime_birthdate.year
|
|
|
|
document_expedition_date = False
|
|
if doc_type.code == "D" or doc_type.code == "P":
|
|
if age < 30:
|
|
document_expedition_date = datetime_doc_date - relativedelta(years=5)
|
|
else:
|
|
document_expedition_date = datetime_doc_date - relativedelta(years=10)
|
|
if doc_type.code == "C":
|
|
if age < 70:
|
|
document_expedition_date = datetime_doc_date - relativedelta(years=10)
|
|
return document_expedition_date
|
|
|
|
def action_on_board(self):
|
|
for record in self:
|
|
if record.reservation_id.checkin > fields.Date.today():
|
|
raise ValidationError(_("It is not yet checkin day!"))
|
|
if record.reservation_id.checkout < fields.Date.today():
|
|
raise ValidationError(_("Its too late to checkin"))
|
|
|
|
if any(
|
|
not getattr(record, field) for field in self._checkin_mandatory_fields()
|
|
):
|
|
raise ValidationError(_("Personal data is missing for check-in"))
|
|
vals = {
|
|
"state": "onboard",
|
|
"arrival": fields.Datetime.now(),
|
|
}
|
|
record.update(vals)
|
|
record.reservation_id.state = "onboard"
|
|
record.identifier = (
|
|
record.reservation_id.pms_property_id.checkin_sequence_id._next_do()
|
|
)
|
|
|
|
def action_done(self):
|
|
for record in self.filtered(lambda c: c.state == "onboard"):
|
|
vals = {
|
|
"state": "done",
|
|
"departure": fields.Datetime.now(),
|
|
}
|
|
record.update(vals)
|
|
return True
|
|
|
|
def action_undo_onboard(self):
|
|
for record in self.filtered(lambda c: c.state == "onboard"):
|
|
vals = {
|
|
"state": "precheckin",
|
|
"arrival": False,
|
|
}
|
|
record.update(vals)
|
|
return True
|
|
|
|
def open_partner(self):
|
|
"""Utility method used to add an "View Customer" button in checkin partner views"""
|
|
self.ensure_one()
|
|
partner_form_id = self.env.ref("pms.view_partner_data_form").id
|
|
return {
|
|
"type": "ir.actions.act_window",
|
|
"res_model": "res.partner",
|
|
"view_mode": "form",
|
|
"views": [(partner_form_id, "form")],
|
|
"res_id": self.partner_id.id,
|
|
"target": "new",
|
|
"flags": {"form": {"action_buttons": True}},
|
|
}
|
|
|
|
def open_wizard_several_partners(self):
|
|
ctx = dict(
|
|
checkin_partner_id=self.id,
|
|
possible_existing_customer_ids=self.possible_existing_customer_ids.ids,
|
|
)
|
|
return {
|
|
"view_type": "form",
|
|
"view_mode": "form",
|
|
"name": "Several Customers",
|
|
"res_model": "pms.several.partners.wizard",
|
|
"target": "new",
|
|
"type": "ir.actions.act_window",
|
|
"context": ctx,
|
|
}
|
|
|
|
def _save_data_from_portal(self, values):
|
|
checkin_partner = values.get("checkin_partner", "")
|
|
values.pop("checkin_partner")
|
|
values.pop("folio_access_token")
|
|
if values.get("nationality"):
|
|
values.update({"nationality_id": int(values.get("nationality_id"))})
|
|
|
|
doc_type = (
|
|
self.sudo()
|
|
.env["res.partner.id_category"]
|
|
.browse(int(values.get("document_type")))
|
|
)
|
|
if values.get("document_type"):
|
|
values.update({"document_type": int(values.get("document_type"))})
|
|
if values.get("residence_state_id"):
|
|
values.update({"residence_state_id": int(values.get("residence_state_id"))})
|
|
if values.get("residence_country_id"):
|
|
values.update(
|
|
{"residence_country_id": int(values.get("residence_country_id"))}
|
|
)
|
|
|
|
if values.get("document_expedition_date"):
|
|
values.update(
|
|
{
|
|
"document_expedition_date": datetime.strptime(
|
|
values.get("document_expedition_date"), "%d/%m/%Y"
|
|
).strftime("%Y-%m-%d"),
|
|
"birthdate_date": datetime.strptime(
|
|
values.get("birthdate_date"), "%d/%m/%Y"
|
|
).strftime("%Y-%m-%d"),
|
|
}
|
|
)
|
|
doc_date = values.get("document_expedition_date")
|
|
birthdate = values.get("birthdate_date")
|
|
document_expedition_date = (
|
|
self.calculate_doc_type_expedition_date_from_validity_date(
|
|
doc_type, doc_date, birthdate
|
|
)
|
|
)
|
|
values.update(
|
|
{
|
|
"document_expedition_date": document_expedition_date,
|
|
}
|
|
)
|
|
checkin_partner.sudo().write(values)
|
|
|
|
def send_portal_invitation_email(self, invitation_firstname=None, email=None):
|
|
template = self.sudo().env.ref(
|
|
"pms.precheckin_invitation_email", raise_if_not_found=False
|
|
)
|
|
subject = template._render_field(
|
|
"subject", [6, 0, self.id], compute_lang=True, post_process=True
|
|
)[self.id]
|
|
body = template._render_field(
|
|
"body_html", [6, 0, self.id], compute_lang=True, post_process=True
|
|
)[self.id]
|
|
invitation_mail = (
|
|
self.env["mail.mail"]
|
|
.sudo()
|
|
.create(
|
|
{
|
|
"subject": subject,
|
|
"body_html": body,
|
|
"email_from": self.pms_property_id.partner_id.email,
|
|
"email_to": email,
|
|
}
|
|
)
|
|
)
|
|
|
|
invitation_mail.send()
|
|
|
|
def send_exit_email(self, template_id):
|
|
template = self.env["mail.template"].browse(template_id)
|
|
if self.email:
|
|
template.send_mail(
|
|
self.id,
|
|
force_send=True,
|
|
raise_exception=False,
|
|
email_values={"email_to": self.email, "auto_delete": False},
|
|
)
|
|
body = template._render_field(
|
|
"body_html", [6, 0, self.id], compute_lang=True, post_process=True
|
|
)[self.id]
|
|
self.reservation_id.message_post(body=body)
|
|
|
|
if self.reservation_id.to_send_exit_mail:
|
|
emails = self.reservation_id.checkin_partner_ids.mapped("email")
|
|
if (
|
|
self.reservation_id.partner_id
|
|
and self.reservation_id.partner_id.email
|
|
and self.reservation_id.partner_id.email not in emails
|
|
):
|
|
template.send_mail(
|
|
self.partner_id.id,
|
|
force_send=True,
|
|
raise_exception=False,
|
|
email_values={
|
|
"email_to": self.reservation_id.email,
|
|
"auto_delete": False,
|
|
},
|
|
)
|
|
body = template._render_field(
|
|
"body_html", [6, 0, self.id], compute_lang=True, post_process=True
|
|
)[self.id]
|
|
self.reservation_id.message_post(body=body)
|
|
self.reservation_id.to_send_exit_mail = False
|