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 + + + + + + + +