Skip to content

Commit 6b2e660

Browse files
committed
[ADD] estate: add new real estate module
Framework 101 tutorial - add estate module - add estate_account module for integration with invoicing
1 parent 781b590 commit 6b2e660

19 files changed

+590
-0
lines changed

estate/__init__.py

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
from . import models

estate/__manifest__.py

Lines changed: 18 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
{
2+
"name": "Real Estate",
3+
"description": "Real Estate Advertising",
4+
"author": "Odoo S.A.",
5+
"license": "LGPL-3",
6+
"depends": ["base"],
7+
"application": True,
8+
"data": [
9+
"security/ir.model.access.csv",
10+
11+
"views/estate_property_views.xml",
12+
"views/estate_property_offer_views.xml",
13+
"views/estate_property_type_views.xml",
14+
"views/estate_property_tag_views.xml",
15+
"views/estate_menus.xml",
16+
"views/res_users_views.xml",
17+
],
18+
}

estate/models/__init__.py

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
from . import estate_property_offer
2+
from . import estate_property_tag
3+
from . import estate_property_type
4+
from . import estate_property
5+
from . import res_users

estate/models/estate_property.py

Lines changed: 113 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,113 @@
1+
from dateutil.relativedelta import relativedelta
2+
from odoo import api, fields, models
3+
from odoo.exceptions import UserError, ValidationError
4+
from odoo.tools.float_utils import float_compare, float_is_zero
5+
6+
7+
class EstateProperty(models.Model):
8+
_name = "estate.property"
9+
_description = "Estate property"
10+
_order = "id desc"
11+
12+
name = fields.Char(string="Title", required=True)
13+
description = fields.Text()
14+
postcode = fields.Char()
15+
date_availability = fields.Date(string="Available From", default=lambda self: fields.Date.context_today(self) + relativedelta(months=3), copy=False)
16+
expected_price = fields.Float(required=True)
17+
selling_price = fields.Float(readonly=True, copy=False)
18+
bedrooms = fields.Integer(default=2)
19+
living_area = fields.Integer(string="Living Area (sqm)")
20+
facades = fields.Integer()
21+
garage = fields.Boolean()
22+
garden = fields.Boolean()
23+
garden_area = fields.Integer(string="Garden Area (sqm)")
24+
garden_orientation = fields.Selection(
25+
selection=[("north", "North"), ("south", "South"), ("east", "East"), ("west", "West")],
26+
)
27+
28+
# Reserved fields
29+
active = fields.Boolean(default=True)
30+
state = fields.Selection(
31+
required=True,
32+
selection=[("new", "New"), ("offer_received", "Offer Received"), ("offer_accepted", "Offer Accepted"), ("sold", "Sold"), ("cancelled", "Cancelled")],
33+
default="new",
34+
copy=False,
35+
string="Status",
36+
)
37+
38+
# Relations
39+
property_type_id = fields.Many2one("estate.property.type", string="Property Type")
40+
salesman_id = fields.Many2one("res.users", default=lambda self: self.env.user)
41+
buyer_id = fields.Many2one("res.partner", copy=False)
42+
tag_ids = fields.Many2many("estate.property.tag")
43+
offer_ids = fields.One2many("estate.property.offer", "property_id")
44+
45+
# Computed
46+
total_area = fields.Integer(compute="_compute_total_area", string="Total Area (sqm)")
47+
best_price = fields.Float(compute="_compute_best_price", string="Best Offer")
48+
49+
@api.depends("living_area", "garden_area")
50+
def _compute_total_area(self) -> None:
51+
for record in self:
52+
record.total_area = record.living_area + record.garden_area
53+
54+
@api.depends("offer_ids.price")
55+
def _compute_best_price(self) -> None:
56+
for record in self:
57+
record.best_price = max(record.offer_ids.mapped("price"), default=0)
58+
59+
# Methods that trigger on changes
60+
@api.onchange("garden")
61+
def _onchange_garden_defaults(self) -> None:
62+
if self.garden:
63+
self.garden_area = 10
64+
self.garden_orientation = "north"
65+
else:
66+
self.garden_area = None
67+
self.garden_orientation = None
68+
69+
@api.ondelete(at_uninstall=False)
70+
def _unlink_except_new_or_cancelled(self) -> None:
71+
if any(record.state not in {"new", "cancelled"} for record in self):
72+
raise UserError(self.env._("Cannot delete properties unless they are new or cancelled."))
73+
74+
# Public methods
75+
def action_set_sold(self) -> bool:
76+
for record in self:
77+
if record.state == "sold":
78+
continue
79+
80+
if record.state == "cancelled":
81+
raise UserError(record.env._("Cancelled properties cannot be sold."))
82+
83+
record.state = "sold"
84+
return True
85+
86+
def action_set_cancelled(self) -> bool:
87+
for record in self:
88+
if record.state == "cancelled":
89+
continue
90+
91+
if record.state == "sold":
92+
raise UserError(record.env._("Sold properties cannot be cancelled."))
93+
94+
record.state = "cancelled"
95+
return True
96+
97+
# Constraints
98+
_check_expected_price_strict_positive = models.Constraint(
99+
"CHECK(expected_price > 0)",
100+
"A property's expected price must be strictly greater than 0.",
101+
)
102+
103+
_check_selling_price_positive = models.Constraint(
104+
"CHECK(selling_price >= 0)",
105+
"A property's selling price must be equal to or greater than 0.",
106+
)
107+
108+
@api.constrains("expected_price", "selling_price")
109+
def _check_selling_price_percentage(self) -> None:
110+
for record in self:
111+
# Selling price is zero when no offer has been accepted
112+
if not float_is_zero(record.selling_price, precision_digits=2) and float_compare(record.selling_price, 0.9 * record.expected_price, precision_digits=2) < 0:
113+
raise ValidationError(record.env._("A property's selling price cannot be lower than 90 percent of its expected price."))
Lines changed: 72 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
from dateutil.relativedelta import relativedelta
2+
from odoo import api, fields, models
3+
from odoo.exceptions import UserError
4+
from odoo.tools.float_utils import float_compare
5+
6+
7+
class EstatePropertyOffer(models.Model):
8+
_name = "estate.property.offer"
9+
_description = "Estate property offer"
10+
_order = "price desc"
11+
12+
price = fields.Float()
13+
status = fields.Selection(
14+
selection=[("accepted", "Accepted"), ("refused", "Refused")],
15+
copy=False,
16+
)
17+
validity = fields.Integer(default=7, string="Validity (days)")
18+
19+
# Relations
20+
partner_id = fields.Many2one("res.partner", required=True)
21+
property_id = fields.Many2one("estate.property", required=True)
22+
property_type_id = fields.Many2one(related="property_id.property_type_id", store=True)
23+
24+
# Computed
25+
date_deadline = fields.Date(compute="_compute_deadline", inverse="_inverse_deadline", string="Deadline")
26+
27+
@api.depends("create_date", "validity")
28+
def _compute_deadline(self) -> None:
29+
for record in self:
30+
ref_date = fields.Date.context_today(self) if not record.create_date else record.create_date.date()
31+
record.date_deadline = ref_date + relativedelta(days=record.validity)
32+
33+
def _inverse_deadline(self) -> None:
34+
for record in self:
35+
ref_date = fields.Date.context_today(self) if not record.create_date else record.create_date.date()
36+
record.validity = (record.date_deadline - ref_date).days
37+
38+
# CRUD overrides
39+
@api.model
40+
def create(self, vals):
41+
for val in vals:
42+
estate_property = self.env["estate.property"].browse(val["property_id"])
43+
best_price = estate_property.best_price or 0.0
44+
if float_compare(val["price"], best_price, precision_digits=2) < 0:
45+
raise UserError(self.env._("A new offer must match or exceed the price of the current best offer."))
46+
estate_property.state = "offer_received"
47+
return super().create(vals)
48+
49+
# Public methods
50+
def action_accept(self) -> bool:
51+
# TODO double check validation
52+
for record in self:
53+
if record.property_id.offer_ids.filtered(lambda r: r.status == "accepted"):
54+
raise UserError(record.env._("Cannot accept this offer because another offer has already been accepted for the property."))
55+
56+
record.status = "accepted"
57+
record.property_id.selling_price = record.price
58+
record.property_id.buyer_id = record.partner_id
59+
record.property_id.state = "offer_accepted"
60+
return True
61+
62+
def action_refuse(self) -> bool:
63+
# TODO double check validation
64+
for record in self:
65+
record.status = "refused"
66+
return True
67+
68+
# Constraints
69+
_check_price_strict_positive = models.Constraint(
70+
"CHECK(price > 0)",
71+
"An offer's price must be strictly greater than 0.",
72+
)
Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
from odoo import fields, models
2+
3+
4+
class EstatePropertyTag(models.Model):
5+
_name = "estate.property.tag"
6+
_description = "Estate property tag"
7+
_order = "name"
8+
9+
name = fields.Char(required=True)
10+
color = fields.Integer()
11+
12+
# Constraints
13+
_uniq_name = models.Constraint(
14+
"UNIQUE(name)",
15+
"A property tag's name must be unique.",
16+
)
Lines changed: 28 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
from odoo import api, fields, models
2+
3+
4+
class EstatePropertyType(models.Model):
5+
_name = "estate.property.type"
6+
_description = "Estate property type"
7+
_order = "name"
8+
9+
name = fields.Char(required=True)
10+
sequence = fields.Integer(default=1)
11+
12+
# Relations
13+
property_ids = fields.One2many("estate.property", "property_type_id")
14+
offer_ids = fields.One2many("estate.property.offer", "property_type_id")
15+
16+
# Computed
17+
offer_count = fields.Integer(compute="_compute_offer_count")
18+
19+
@api.depends("offer_ids")
20+
def _compute_offer_count(self) -> None:
21+
for record in self:
22+
record.offer_count = len(record.offer_ids)
23+
24+
# Constraints
25+
_uniq_name = models.Constraint(
26+
"UNIQUE(name)",
27+
"A property type's name must be unique.",
28+
)

estate/models/res_users.py

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
from odoo import fields, models
2+
3+
4+
class User(models.Model):
5+
_inherit = "res.users"
6+
7+
property_ids = fields.One2many("estate.property", "salesman_id", domain=[("state", "in", ["new", "offer_received"])])
Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink
2+
access_estate_property,access_estate_property,model_estate_property,base.group_user,1,1,1,1
3+
access_estate_property_type,access_estate_property_type,model_estate_property_type,base.group_user,1,1,1,1
4+
access_estate_property_tag,access_estate_property_tag,model_estate_property_tag,base.group_user,1,1,1,1
5+
access_estate_property_offer,access_estate_property_offer,model_estate_property_offer,base.group_user,1,1,1,1

estate/views/estate_menus.xml

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
<?xml version="1.0"?>
2+
<odoo>
3+
<menuitem id="estate_menu_root" name="Real Estate">
4+
<menuitem id="estate_property_menu" name="Properties">
5+
<menuitem id="estate_property_menu_action" action="estate_property_action"/>
6+
</menuitem>
7+
<menuitem id="estate_settings_menu" name="Settings">
8+
<menuitem id="estate_property_type_menu" action="estate_property_type_action"/>
9+
<menuitem id="estate_property_tag_menu" action="estate_property_tag_action"/>
10+
</menuitem>
11+
</menuitem>
12+
</odoo>

0 commit comments

Comments
 (0)