From fd32a73cf9783f0883c66401e887db4c9400596b Mon Sep 17 00:00:00 2001 From: sreedevk Date: Tue, 19 May 2026 14:33:52 +0000 Subject: [PATCH] [ADD] estate: estate module added for real estate businesses [ADD] estate: added the module setup files, the estate_property model & basic default views [LINT] estate: ruff linting & formatting [FIX] estate: added all rights to estate_property model [ADD] estate: estate_property model field level restrictions & properties added [LINT] estate: ruff linting & formatting [ADD] estate: custom list view for estate_property [REF] estate: follow naming convention for view records [REM] estate: remove test model [ADD] estate: added search view with custom fields [ADD] estate: add filters for available properties & group by postcode [ADD] estate: add property type models & views [ADD] estate: property type relationship to property added [ADD] estate: buyer & seller fields added to estate_property [ADD] estate: property tag added [LINT] estate: linting & formatting [ADD] estate: estate property offer added [ADD] estate: best price & total area computed fields added [ADD] estate: property offer date_deadline & validity fields added [ADD] estate: property onchange method for garden based attributes [ADD] estate: offer refuse & accept operations added [ADD] estate: sold & cancelled buttons added to properties [ADD] estate: accepting offer now sets buyer & selling price of property [ADD] estate: property constraints [LINT] satisfy the linting and formatting overlords [FIX] estate: api.constrains was misspelt as api.constraint [FIX] estate: missing precision_digits in float_is_zero call [FIX] estate: inverse deadline computation type issue fixed [ADD] estate: property_ids view for property_type form [LINT] estate: sacrifice to the linter [ADD] estate: property status now displayed as a statusbar widget [ADD] estate: deterministic ordering of records [ADD] estate: manual sequencing of types enabled using a handle widget [ADD] estate: tag color fields added to property form & prevent type creation [ADD] estate: sold and cancel buttons now only show up when appropriate [ADD] estate: default available filter applied --- estate/__init__.py | 3 + estate/__manifest__.py | 16 ++++ estate/models/__init__.py | 13 +++ estate/models/estate_property.py | 117 +++++++++++++++++++++++++ estate/models/estate_property_offer.py | 66 ++++++++++++++ estate/models/estate_property_tag.py | 15 ++++ estate/models/estate_property_type.py | 18 ++++ estate/security/ir.model.access.csv | 5 ++ estate/views/estate_menus.xml | 21 +++++ estate/views/estate_property.xml | 115 ++++++++++++++++++++++++ estate/views/estate_property_offer.xml | 35 ++++++++ estate/views/estate_property_tag.xml | 8 ++ estate/views/estate_property_type.xml | 40 +++++++++ 13 files changed, 472 insertions(+) create mode 100644 estate/__init__.py create mode 100644 estate/__manifest__.py create mode 100644 estate/models/__init__.py create mode 100644 estate/models/estate_property.py create mode 100644 estate/models/estate_property_offer.py create mode 100644 estate/models/estate_property_tag.py create mode 100644 estate/models/estate_property_type.py create mode 100644 estate/security/ir.model.access.csv create mode 100644 estate/views/estate_menus.xml create mode 100644 estate/views/estate_property.xml create mode 100644 estate/views/estate_property_offer.xml create mode 100644 estate/views/estate_property_tag.xml create mode 100644 estate/views/estate_property_type.xml diff --git a/estate/__init__.py b/estate/__init__.py new file mode 100644 index 00000000000..0d4bc745927 --- /dev/null +++ b/estate/__init__.py @@ -0,0 +1,3 @@ +from . import models + +__all__ = ["models"] diff --git a/estate/__manifest__.py b/estate/__manifest__.py new file mode 100644 index 00000000000..cee581c4c74 --- /dev/null +++ b/estate/__manifest__.py @@ -0,0 +1,16 @@ +{ + "name": "Estate", + "depends": ["base"], + "author": "Sreedev Kodichath ", + "license": "LGPL-3", + "data": [ + # datafiles + "security/ir.model.access.csv", + # views + "views/estate_property.xml", + "views/estate_property_type.xml", + "views/estate_property_tag.xml", + "views/estate_property_offer.xml", + "views/estate_menus.xml", + ], +} diff --git a/estate/models/__init__.py b/estate/models/__init__.py new file mode 100644 index 00000000000..a687607959d --- /dev/null +++ b/estate/models/__init__.py @@ -0,0 +1,13 @@ +from . import ( + estate_property, + estate_property_offer, + estate_property_tag, + estate_property_type, +) + +__all__ = [ + "estate_property", + "estate_property_offer", + "estate_property_tag", + "estate_property_type", +] diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py new file mode 100644 index 00000000000..843a8e5b72e --- /dev/null +++ b/estate/models/estate_property.py @@ -0,0 +1,117 @@ +from odoo.tools.float_utils import float_compare, float_is_zero +from odoo import api, models, fields, _ +from dateutil.relativedelta import relativedelta +from odoo.exceptions import UserError, ValidationError + + +class EstateProperty(models.Model): + _name = "estate.property" + _description = "Estate Property" + _order = "id desc" + + active = fields.Boolean(default=True) + state = fields.Selection( + string="State", + copy=False, + default="new", + required=True, + selection=[ + ("new", "New"), + ("offer_recd", "Offer Received"), + ("offer_accepted", "Offer Accepted"), + ("sold", "Sold"), + ("cancelled", "Cancelled"), + ], + ) + name = fields.Char(required=True) + description = fields.Text() + postcode = fields.Char() + date_availability = fields.Date( + default=(fields.Date.today() + relativedelta(months=3)), copy=False + ) + seller_id = fields.Many2one( + "res.users", string="Sales Person", default=lambda self: self.env.user + ) + buyer_id = fields.Many2one("res.partner", string="Buyer") + + expected_price = fields.Float(required=True) + _check_expected_price = models.Constraint( + "CHECK(expected_price > 0)", "Expected price should be positive." + ) + selling_price = fields.Float(readonly=True, copy=False) + _check_selling_price = models.Constraint( + "CHECK(selling_price > 0)", "Selling price must be positive." + ) + + best_price = fields.Float(compute="_compute_best_price") + property_tag_ids = fields.Many2many("estate.property.tag", string="Property Tags") + bedrooms = fields.Integer(default=2) + living_area = fields.Integer() + facades = fields.Integer() + garage = fields.Boolean() + garden = fields.Boolean() + garden_area = fields.Integer("Property Garden Area") + total_area = fields.Integer(compute="_compute_total_area") + property_type_id = fields.Many2one("estate.property.type", string="Property Type") + offer_ids = fields.One2many("estate.property.offer", "property_id", string="Offers") + garden_orientation = fields.Selection( + string="Garden Orientation", + selection=[ + ("north", "North"), + ("south", "South"), + ("east", "East"), + ("west", "West"), + ], + help="Select garden orientation", + ) + + @api.depends("living_area", "garden_area") + def _compute_total_area(self): + for property in self: + property.total_area = property.living_area + property.garden_area + + @api.depends("offer_ids.price") + def _compute_best_price(self): + for property in self: + property.best_price = max(property.offer_ids.mapped("price"), default=0) + + @api.onchange("garden") + def _onchange_garden(self): + for property in self: + if property.garden: + property.garden_area = 10 + property.garden_orientation = "north" + else: + property.garden_area = None + property.garden_orientation = None + + def action_cancel_property(self): + for property in self: + if property.state == "sold": + raise UserError(_("Sold properties cannot be cancelled.")) + else: + property.state = "cancelled" + return True + + def action_sold_property(self): + for property in self: + if property.state == "cancelled": + raise UserError(_("Cancelled properties cannot be sold.")) + else: + property.state = "sold" + return True + + @api.constrains("selling_price") + def _check_selling_price(self): + for property in self: + if not float_is_zero(property.selling_price, precision_digits=2) and ( + float_compare( + property.selling_price, + 0.9 * property.expected_price, + precision_digits=2, + ) + == -1 + ): + raise ValidationError( + _("Selling price is not atleast 90% of expected price.") + ) diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py new file mode 100644 index 00000000000..13f9abc58c9 --- /dev/null +++ b/estate/models/estate_property_offer.py @@ -0,0 +1,66 @@ +from odoo import api, models, fields, _ +from dateutil.relativedelta import relativedelta +from odoo.exceptions import UserError + + +class EstatePropertyOffer(models.Model): + _name = "estate.property.offer" + _description = "Offer made on properties" + _order = "price desc" + + price = fields.Float() + _check_price = models.Constraint( + "CHECK(price > 0)", "Offer price must be positive." + ) + + status = fields.Selection( + string="status", + selection=[ + ("accepted", "Accepted"), + ("refused", "Refused"), + ], + help="Offer Status", + ) + + partner_id = fields.Many2one("res.partner", required=True) + property_id = fields.Many2one("estate.property", required=True) + validity = fields.Integer(default=7) + date_deadline = fields.Date( + compute="_compute_date_deadline", inverse="_compute_validity" + ) + + @api.depends("validity", "create_date") + def _compute_date_deadline(self): + for offer in self: + offer.date_deadline = ( + offer.create_date or fields.Date.today() + ) + relativedelta(days=offer.validity) + + def action_accept_offer(self): + for offer in self: + is_not_actionable = any( + pstat == "accepted" + for pstat in offer.property_id.offer_ids.mapped("status") + ) + if is_not_actionable: + raise UserError( + _("An offer has already been accepted for this property.") + ) + else: + offer.status = "accepted" + offer.property_id.buyer_id = offer.partner_id + offer.property_id.selling_price = offer.price + offer.property_id.state = "sold" + return True + + def action_reject_offer(self): + for offer in self: + offer.status = "refused" + return True + + @api.depends("create_date", "date_deadline") + def _compute_validity(self): + for offer in self: + offer.validity = ( + offer.date_deadline - (offer.create_date or fields.Date.today()).date() + ).days diff --git a/estate/models/estate_property_tag.py b/estate/models/estate_property_tag.py new file mode 100644 index 00000000000..ab9a7c78bb4 --- /dev/null +++ b/estate/models/estate_property_tag.py @@ -0,0 +1,15 @@ +from odoo import models, fields + + +class EstatePropertyTag(models.Model): + _name = "estate.property.tag" + _description = "Tag representing an attribute's presence in a property" + _order = "name" + + active = fields.Boolean(default=True) + color = fields.Integer(string="Color") + name = fields.Char(required=True) + _name_uniq = models.Constraint( + "unique (name)", + "Each tag name must be unique.", + ) diff --git a/estate/models/estate_property_type.py b/estate/models/estate_property_type.py new file mode 100644 index 00000000000..887f9b69d1b --- /dev/null +++ b/estate/models/estate_property_type.py @@ -0,0 +1,18 @@ +from odoo import models, fields + + +class EstatePropertyType(models.Model): + _name = "estate.property.type" + _description = "type of the estate property" + _order = "name" + + active = fields.Boolean(default=True) + name = fields.Char(required=True) + sequence = fields.Integer(string="Sequence", default=1, help="Used for ordering") + _name_uniq = models.Constraint( + "unique (name)", + "Each property type name must be unique.", + ) + property_ids = fields.One2many( + "estate.property", "property_type_id", string="Properties" + ) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv new file mode 100644 index 00000000000..a90b03a38c6 --- /dev/null +++ b/estate/security/ir.model.access.csv @@ -0,0 +1,5 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +estate.access_estate_property,access_estate_property,model_estate_property,base.group_user,1,1,1,1 +estate.access_estate_property_type,access_estate_property_type,model_estate_property_type,base.group_user,1,1,1,1 +estate.access_estate_property_tag,access_estate_property_tag,model_estate_property_tag,base.group_user,1,1,1,1 +estate.access_estate_property_offer,access_estate_property_offer,model_estate_property_offer,base.group_user,1,1,1,1 diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml new file mode 100644 index 00000000000..ea5a53a5491 --- /dev/null +++ b/estate/views/estate_menus.xml @@ -0,0 +1,21 @@ + + + + + + + + + + + + diff --git a/estate/views/estate_property.xml b/estate/views/estate_property.xml new file mode 100644 index 00000000000..3be4cbc7ffd --- /dev/null +++ b/estate/views/estate_property.xml @@ -0,0 +1,115 @@ + + + + estate.property.list + estate.property + + + + + + + + + + + + + + + estate.property.form + estate.property + +
+
+ +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+ + + estate.property.search + estate.property + + + + + + + + + + + + + + + + + + + Estate Properties + estate.property + list,form + {'search_default_available': True} + +
diff --git a/estate/views/estate_property_offer.xml b/estate/views/estate_property_offer.xml new file mode 100644 index 00000000000..4deeeed431a --- /dev/null +++ b/estate/views/estate_property_offer.xml @@ -0,0 +1,35 @@ + + + + + estate.property.offer.list + estate.property.offer + + + + + + + +