From 6a48a788ee97b894c72f8adfffd8966352ab393c Mon Sep 17 00:00:00 2001 From: Rana Date: Wed, 22 Apr 2026 14:02:20 +0200 Subject: [PATCH 01/11] [ADD] estate: Creating new estate module with estate property model and access rights A new estate module was created. Property model was created as well and contains some fields. Access rights were added. --- estate/__init__.py | 1 + estate/__manifest__.py | 11 +++++++++++ estate/models/__init__.py | 1 + estate/models/estate_property.py | 20 ++++++++++++++++++++ estate/security/ir.model.access.csv | 2 ++ 5 files changed, 35 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/security/ir.model.access.csv diff --git a/estate/__init__.py b/estate/__init__.py new file mode 100644 index 00000000000..0650744f6bc --- /dev/null +++ b/estate/__init__.py @@ -0,0 +1 @@ +from . import models diff --git a/estate/__manifest__.py b/estate/__manifest__.py new file mode 100644 index 00000000000..3b5e0e9a467 --- /dev/null +++ b/estate/__manifest__.py @@ -0,0 +1,11 @@ +{ + 'name': 'Real Estate', + 'depends': [ + 'base' + ], + 'data': [ + 'security/ir.model.access.csv' + ], + 'application': True, + 'license': 'AGPL-3' +} diff --git a/estate/models/__init__.py b/estate/models/__init__.py new file mode 100644 index 00000000000..5e1963c9d2f --- /dev/null +++ b/estate/models/__init__.py @@ -0,0 +1 @@ +from . import estate_property diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py new file mode 100644 index 00000000000..b35073924a8 --- /dev/null +++ b/estate/models/estate_property.py @@ -0,0 +1,20 @@ +from odoo import fields, models + + +class Property(models.Model): + _name = "estate.property" + _description = "Properties of the estate" + + name = fields.Char(required=True) + description = fields.Text() + postcode = fields.Char() + date_availability = fields.Date() + expected_price = fields.Float(required=True) + selling_price = fields.Float() + bedrooms = fields.Integer() + living_area = fields.Integer() + facades = fields.Integer() + garage = fields.Boolean() + garden = fields.Boolean() + garden_area = fields.Integer() + garden_orientation = fields.Selection(selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')]) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv new file mode 100644 index 00000000000..d84739f906f --- /dev/null +++ b/estate/security/ir.model.access.csv @@ -0,0 +1,2 @@ +id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink +access_property_model,access_property_model,model_estate_property,base.group_user,1,1,1,1 \ No newline at end of file From b02fdc050550645ea1385b59c243e2f7501ee133 Mon Sep 17 00:00:00 2001 From: "David Van Droogenbroeck (DROD)" Date: Thu, 23 Apr 2026 10:17:45 +0200 Subject: [PATCH 02/11] [FIX] estate: remove unlink permission for base users Base users were able to unlink records although they're just plebs. --- estate/security/ir.model.access.csv | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index d84739f906f..d523f65e039 100644 --- a/estate/security/ir.model.access.csv +++ b/estate/security/ir.model.access.csv @@ -1,2 +1,2 @@ id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink -access_property_model,access_property_model,model_estate_property,base.group_user,1,1,1,1 \ No newline at end of file +access_property_model,access_property_model,model_estate_property,base.group_user,1,1,1,0 \ No newline at end of file From 9d4d467129a4af7ae42cd3b164edcf09e34e9b4f Mon Sep 17 00:00:00 2001 From: Rana Date: Thu, 23 Apr 2026 12:29:53 +0200 Subject: [PATCH 03/11] [ADD] estate: Added actions, menues, and new fields Added an xml file for actions. Also added the 3 levels of menu items and linked them to actions. Finally, some new fields were added like active and state. Some new attributes were added to new and existing fields. --- FETCH_HEAD | 0 estate/__manifest__.py | 8 ++++++-- estate/models/estate_property.py | 10 ++++++---- estate/security/ir.model.access.csv | 2 +- estate/views/estate_menus.xml | 8 ++++++++ estate/views/estate_property_views.xml | 8 ++++++++ 6 files changed, 29 insertions(+), 7 deletions(-) create mode 100644 FETCH_HEAD create mode 100644 estate/views/estate_menus.xml create mode 100644 estate/views/estate_property_views.xml diff --git a/FETCH_HEAD b/FETCH_HEAD new file mode 100644 index 00000000000..e69de29bb2d diff --git a/estate/__manifest__.py b/estate/__manifest__.py index 3b5e0e9a467..f1b96d6a992 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -1,11 +1,15 @@ { 'name': 'Real Estate', + 'author': "Odoo", 'depends': [ 'base' ], 'data': [ - 'security/ir.model.access.csv' + 'security/ir.model.access.csv', + 'views/estate_property_views.xml', + 'views/estate_menus.xml', ], 'application': True, - 'license': 'AGPL-3' + 'installable': True, + 'license': 'AGPL-3', } diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index b35073924a8..7c9b589d5ef 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -3,18 +3,20 @@ class Property(models.Model): _name = "estate.property" - _description = "Properties of the estate" + _description = "Properties" name = fields.Char(required=True) description = fields.Text() postcode = fields.Char() - date_availability = fields.Date() + date_availability = fields.Date(copy=False, default=fields.Date.add(fields.Date.today(), months=3)) expected_price = fields.Float(required=True) - selling_price = fields.Float() - bedrooms = fields.Integer() + selling_price = fields.Float(readonly=True, copy=False) + bedrooms = fields.Integer(default=2) living_area = fields.Integer() facades = fields.Integer() garage = fields.Boolean() garden = fields.Boolean() garden_area = fields.Integer() garden_orientation = fields.Selection(selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')]) + active = fields.Boolean(default=True) + state = fields.Selection(selection=[('new', 'New'), ('offer_received', 'Offer Received'), ('offer_accepted', 'Offer Accepted'), ('sold', 'Sold'), ('cancelled', 'Cancelled')], required=True, copy=False, default='new') diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index d523f65e039..a8d7d43b717 100644 --- a/estate/security/ir.model.access.csv +++ b/estate/security/ir.model.access.csv @@ -1,2 +1,2 @@ id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink -access_property_model,access_property_model,model_estate_property,base.group_user,1,1,1,0 \ No newline at end of file +access_property_model,access_property_model,model_estate_property,base.group_user,1,1,1,0 diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml new file mode 100644 index 00000000000..b8b027e632b --- /dev/null +++ b/estate/views/estate_menus.xml @@ -0,0 +1,8 @@ + + + + + + + + diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml new file mode 100644 index 00000000000..ee5096e07a8 --- /dev/null +++ b/estate/views/estate_property_views.xml @@ -0,0 +1,8 @@ + + + + Property action + estate.property + list,form + + From 07e5a3b3d826d14874456d1b10e769aac4873f57 Mon Sep 17 00:00:00 2001 From: Rana Date: Fri, 24 Apr 2026 11:26:39 +0200 Subject: [PATCH 04/11] [ADD] estate: Chapter 6 List view, form view, and search with filter were added in this Chapter --- estate/views/estate_menus.xml | 4 +- estate/views/estate_property_views.xml | 73 +++++++++++++++++++++++++- 2 files changed, 74 insertions(+), 3 deletions(-) diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml index b8b027e632b..6d2d144b6f2 100644 --- a/estate/views/estate_menus.xml +++ b/estate/views/estate_menus.xml @@ -1,7 +1,7 @@ - - + + diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index ee5096e07a8..3e0414bc3d0 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -1,8 +1,79 @@ - Property action + Properties estate.property list,form + + + estate.property.list + estate.property + + + + + + + + + + + + + + + estate.property.form + estate.property + +
+ +

+ +

+ + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+ + + estate.property.search + estate.property + + + + + + + + + + + + +
From 7876339256b781e7b09d4344e95585a682a14887 Mon Sep 17 00:00:00 2001 From: Rana Date: Mon, 27 Apr 2026 09:45:02 +0200 Subject: [PATCH 05/11] [ADD] estate: Chapter 7 Many2one fields and One2many fields were added to represent types, tags, and offers --- estate/models/__init__.py | 3 ++ estate/models/estate_property.py | 5 ++ estate/models/estate_property_offer.py | 11 +++++ estate/models/estate_property_tag.py | 8 ++++ estate/models/estate_property_type.py | 8 ++++ estate/security/ir.model.access.csv | 3 ++ estate/views/estate_menus.xml | 5 ++ estate/views/estate_property_views.xml | 63 ++++++++++++++++++++++++++ 8 files changed, 106 insertions(+) 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 diff --git a/estate/models/__init__.py b/estate/models/__init__.py index 5e1963c9d2f..2f1821a39c1 100644 --- a/estate/models/__init__.py +++ b/estate/models/__init__.py @@ -1 +1,4 @@ from . import estate_property +from . import estate_property_type +from . import estate_property_tag +from . import estate_property_offer diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 7c9b589d5ef..63a49f853e5 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -20,3 +20,8 @@ class Property(models.Model): garden_orientation = fields.Selection(selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')]) active = fields.Boolean(default=True) state = fields.Selection(selection=[('new', 'New'), ('offer_received', 'Offer Received'), ('offer_accepted', 'Offer Accepted'), ('sold', 'Sold'), ('cancelled', 'Cancelled')], required=True, copy=False, default='new') + property_type_id = fields.Many2one("estate.property.type", string="property Type") + buyer_id = fields.Many2one("res.partner", string="Buyer", copy=False) + sales_person_id = fields.Many2one("res.users", string="Salesperson", default=lambda self: self.env.user) + tag_ids = fields.Many2many("estate.property.tag", string="Tags") + offer_ids = fields.One2many("estate.property.offer", "property_id", string="Offers") diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py new file mode 100644 index 00000000000..f0bb1214aa2 --- /dev/null +++ b/estate/models/estate_property_offer.py @@ -0,0 +1,11 @@ +from odoo import fields, models + + +class PropertyOffer(models.Model): + _name = "estate.property.offer" + _description = "Property Offers" + + price = fields.Float() + status = fields.Selection(selection=[('accepted', 'Accepted'), ('refused', 'Refused')], copy=False) + partner_id = fields.Many2one("res.partner", string="Buyer", required=True) + property_id = fields.Many2one("estate.property", string="Property", required=True) diff --git a/estate/models/estate_property_tag.py b/estate/models/estate_property_tag.py new file mode 100644 index 00000000000..30af78f7084 --- /dev/null +++ b/estate/models/estate_property_tag.py @@ -0,0 +1,8 @@ +from odoo import fields, models + + +class PropertyTag(models.Model): + _name = "estate.property.tag" + _description = "Property Tags" + + name = fields.Char(required=True) diff --git a/estate/models/estate_property_type.py b/estate/models/estate_property_type.py new file mode 100644 index 00000000000..c149610b54a --- /dev/null +++ b/estate/models/estate_property_type.py @@ -0,0 +1,8 @@ +from odoo import fields, models + + +class PropertyType(models.Model): + _name = "estate.property.type" + _description = "Property Types" + + name = fields.Char(required=True) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index a8d7d43b717..93d4136ebb1 100644 --- a/estate/security/ir.model.access.csv +++ b/estate/security/ir.model.access.csv @@ -1,2 +1,5 @@ id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink access_property_model,access_property_model,model_estate_property,base.group_user,1,1,1,0 +access_property_type_model,access_property_type_model,model_estate_property_type,base.group_user,1,1,1,0 +access_property_tag_model,access_property_tag_model,model_estate_property_tag,base.group_user,1,1,1,0 +access_property_offer_model,access_property_offer_model,model_estate_property_offer,base.group_user,1,1,1,0 diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml index 6d2d144b6f2..192ee2d502b 100644 --- a/estate/views/estate_menus.xml +++ b/estate/views/estate_menus.xml @@ -4,5 +4,10 @@ + + + + +
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 3e0414bc3d0..7a0f582b334 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -6,6 +6,18 @@ list,form + + Property Types + estate.property.type + list,form + + + + Property Tags + estate.property.tag + list,form + + estate.property.list estate.property @@ -22,6 +34,18 @@ + + estate.property.offer.list + estate.property.offer + + + + + + + + + estate.property.form estate.property @@ -31,8 +55,12 @@

+ + + + @@ -54,12 +82,47 @@ + + + + + + + + + + +
+ + estate.property.offer.form + estate.property.offer + +
+ +

+ +

+ + + + + + + + + +
+
+
+
+ + estate.property.search estate.property From b5fe287676f84d477221f4d2abc2f04a9ce1aecd Mon Sep 17 00:00:00 2001 From: Rana Date: Mon, 27 Apr 2026 15:43:33 +0200 Subject: [PATCH 06/11] [ADD] estate: Chapter 8 Added computed fields, inverse methods, and onchange methods --- estate/models/estate_property.py | 24 +++++++++++++++++++++++- estate/models/estate_property_offer.py | 15 ++++++++++++++- estate/views/estate_property_views.xml | 13 +++++++------ 3 files changed, 44 insertions(+), 8 deletions(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 63a49f853e5..63a4c4ccc18 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -1,4 +1,4 @@ -from odoo import fields, models +from odoo import api, fields, models class Property(models.Model): @@ -25,3 +25,25 @@ class Property(models.Model): sales_person_id = fields.Many2one("res.users", string="Salesperson", default=lambda self: self.env.user) tag_ids = fields.Many2many("estate.property.tag", string="Tags") offer_ids = fields.One2many("estate.property.offer", "property_id", string="Offers") + total_area = fields.Integer(compute="_compute_living_area") + best_price = fields.Float(compute="_compute_best_price") + + @api.depends("living_area", "garden_area") + def _compute_living_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.mapped("offer_ids.price")) + + @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 = 0 + property.garden_orientation = None diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py index f0bb1214aa2..5dc40a8c0f2 100644 --- a/estate/models/estate_property_offer.py +++ b/estate/models/estate_property_offer.py @@ -1,4 +1,4 @@ -from odoo import fields, models +from odoo import api, fields, models class PropertyOffer(models.Model): @@ -9,3 +9,16 @@ class PropertyOffer(models.Model): status = fields.Selection(selection=[('accepted', 'Accepted'), ('refused', 'Refused')], copy=False) partner_id = fields.Many2one("res.partner", string="Buyer", required=True) property_id = fields.Many2one("estate.property", string="Property", required=True) + validity = fields.Integer(default=7) + date_deadline = fields.Date(compute="_compute_date_deadline", inverse="_inverse_date_deadline") + + @api.depends("validity", "create_date") + def _compute_date_deadline(self): + for offer in self: + start_date = offer.create_date or fields.Date.today() + offer.date_deadline = fields.Date.add(start_date, days=offer.validity) + + def _inverse_date_deadline(self): + for offer in self: + start_date = fields.Date.to_date(offer.create_date) or fields.Date.today() + offer.validity = (offer.date_deadline - start_date).days diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 7a0f582b334..34366798fe8 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -41,6 +41,8 @@ + + @@ -66,6 +68,7 @@ + @@ -80,6 +83,7 @@ + @@ -105,15 +109,12 @@
-

- -

- - - + + + From 88c7d3946317f01639c5fcf07c127880b1763fce Mon Sep 17 00:00:00 2001 From: Rana Date: Tue, 28 Apr 2026 10:24:59 +0200 Subject: [PATCH 07/11] [ADD] estate: Chapter 9 Added action buttons --- estate/models/estate_property.py | 18 +++++++++++++++++- estate/models/estate_property_offer.py | 13 +++++++++++++ estate/views/estate_property_views.xml | 7 +++++++ 3 files changed, 37 insertions(+), 1 deletion(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 63a4c4ccc18..1ef02d50996 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -1,4 +1,4 @@ -from odoo import api, fields, models +from odoo import exceptions, api, fields, models class Property(models.Model): @@ -47,3 +47,19 @@ def _onchange_garden(self): else: property.garden_area = 0 property.garden_orientation = None + + def action_cancel(self): + for property in self: + if property.state == 'sold': + raise exceptions.UserError("Cannot cancel a sold peoperty") + else: + property.state = 'cancelled' + return True + + def action_sold(self): + for property in self: + if property.state == 'cancelled': + raise exceptions.UserError("Cannot sell a cancelled peoperty") + else: + property.state = 'sold' + return True diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py index 5dc40a8c0f2..7e0ff98b1a2 100644 --- a/estate/models/estate_property_offer.py +++ b/estate/models/estate_property_offer.py @@ -22,3 +22,16 @@ def _inverse_date_deadline(self): for offer in self: start_date = fields.Date.to_date(offer.create_date) or fields.Date.today() offer.validity = (offer.date_deadline - start_date).days + + def action_accept(self): + for offer in self: + offer.status = 'accepted' + offer.property_id.selling_price = offer.price + offer.property_id.buyer_id = offer.partner_id + return True + + def action_refuse(self): + for offer in self: + offer.status = 'refused' + offer.property_id.selling_price = offer.price + return True diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 34366798fe8..b1af8315016 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -43,6 +43,8 @@ +