From 068874e415e831cc8b873b01042fb10ebb09c68b Mon Sep 17 00:00:00 2001 From: "Mohit Ahir (moahi)" Date: Tue, 10 Mar 2026 18:18:24 +0530 Subject: [PATCH 1/8] [ADD] Estate: Chapter 2 - 3 Finished initial setup of real estate module - Done Basic structure of estate module - Done Estate Property Model Using ORM -Done --- estate/__init__.py | 1 + estate/__manifest__.py | 5 +++++ estate/models/__init__.py | 1 + estate/models/estate_property.py | 29 +++++++++++++++++++++++++++++ 4 files changed, 36 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 diff --git a/estate/__init__.py b/estate/__init__.py new file mode 100644 index 00000000000..9a7e03eded3 --- /dev/null +++ b/estate/__init__.py @@ -0,0 +1 @@ +from . import models \ No newline at end of file diff --git a/estate/__manifest__.py b/estate/__manifest__.py new file mode 100644 index 00000000000..203471d9d03 --- /dev/null +++ b/estate/__manifest__.py @@ -0,0 +1,5 @@ +{ + 'name': 'Estate', + 'depends': ['base'], + 'application': True, +} \ No newline at end of file diff --git a/estate/models/__init__.py b/estate/models/__init__.py new file mode 100644 index 00000000000..f4c8fd6db6d --- /dev/null +++ b/estate/models/__init__.py @@ -0,0 +1 @@ +from . import estate_property \ No newline at end of file diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py new file mode 100644 index 00000000000..19188894851 --- /dev/null +++ b/estate/models/estate_property.py @@ -0,0 +1,29 @@ +from odoo import models, fields + +class EstateProperty(models.Model): + _name = "estate.property" + _description = "Real Estate Property" + + name = fields.Char() + description = fields.Text() + postcode = fields.Char() + date_availability = fields.Date() + expected_price = fields.Float() + 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( + [ + ('north', 'North'), + ('south', 'South'), + ('east', 'East'), + ('west', 'West') + ] + ) + + \ No newline at end of file From 002667bed27d464eb60d9b94bed0eccc1f389615 Mon Sep 17 00:00:00 2001 From: "Mohit Ahir (moahi)" Date: Wed, 11 Mar 2026 18:15:43 +0530 Subject: [PATCH 2/8] [IMP] Error Solved and Chapter 4 done Solved Previous Issue Chapter 4 - done Security File Added --- estate/__init__.py | 2 +- estate/__manifest__.py | 13 ++++++++++++- estate/models/__init__.py | 2 +- estate/models/estate_property.py | 3 +-- estate/security/ir.model.access.csv | 2 ++ 5 files changed, 17 insertions(+), 5 deletions(-) create mode 100644 estate/security/ir.model.access.csv diff --git a/estate/__init__.py b/estate/__init__.py index 9a7e03eded3..0650744f6bc 100644 --- a/estate/__init__.py +++ b/estate/__init__.py @@ -1 +1 @@ -from . import models \ No newline at end of file +from . import models diff --git a/estate/__manifest__.py b/estate/__manifest__.py index 203471d9d03..84dcdddcc16 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -1,5 +1,16 @@ { 'name': 'Estate', + 'summary': 'Manage properties listing and offers', + 'description': "A simple module to manage real estate properties and track offers made by buyers.", + 'author': 'Mohit Ahir (moahi)', + 'website': '', + 'category': 'Estate', + 'version': '1.0', + 'license': 'LGPL-3', 'depends': ['base'], + 'data': [ + 'security/ir.model.access.csv', + ], + 'installable': True, 'application': True, -} \ No newline at end of file +} diff --git a/estate/models/__init__.py b/estate/models/__init__.py index f4c8fd6db6d..5e1963c9d2f 100644 --- a/estate/models/__init__.py +++ b/estate/models/__init__.py @@ -1 +1 @@ -from . import estate_property \ No newline at end of file +from . import estate_property diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 19188894851..2999095095c 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -1,5 +1,6 @@ from odoo import models, fields + class EstateProperty(models.Model): _name = "estate.property" _description = "Real Estate Property" @@ -25,5 +26,3 @@ class EstateProperty(models.Model): ('west', 'West') ] ) - - \ No newline at end of file diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv new file mode 100644 index 00000000000..0e11f47e58d --- /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_estate_property,access_estate_property,model_estate_property,base.group_user,1,1,1,1 \ No newline at end of file From c5101150b928403d16ce9f3a17f512aa09a15f56 Mon Sep 17 00:00:00 2001 From: "Mohit Ahir (moahi)" Date: Thu, 12 Mar 2026 18:40:58 +0530 Subject: [PATCH 3/8] [IMP] Chapter 5 Started Views folder created with XML data file Action and Menu added in UI Learned how actions and menus work --- estate/__manifest__.py | 2 ++ estate/views/estate_menus.xml | 18 ++++++++++++++++++ estate/views/estate_property_views.xml | 9 +++++++++ 3 files changed, 29 insertions(+) create mode 100644 estate/views/estate_menus.xml create mode 100644 estate/views/estate_property_views.xml diff --git a/estate/__manifest__.py b/estate/__manifest__.py index 84dcdddcc16..d1919128a0b 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -10,6 +10,8 @@ 'depends': ['base'], 'data': [ 'security/ir.model.access.csv', + 'views/estate_property_views.xml', + 'views/estate_menus.xml', ], 'installable': True, 'application': True, diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml new file mode 100644 index 00000000000..686e96c8b62 --- /dev/null +++ b/estate/views/estate_menus.xml @@ -0,0 +1,18 @@ + + + + + + + + + + + + diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml new file mode 100644 index 00000000000..911523488a5 --- /dev/null +++ b/estate/views/estate_property_views.xml @@ -0,0 +1,9 @@ + + + + Properties + estate.property + list,form + + + From 8a9ce27aa67eea640b89197204efd4aec58fe3bc Mon Sep 17 00:00:00 2001 From: "Mohit Ahir (moahi)" Date: Fri, 13 Mar 2026 20:59:32 +0530 Subject: [PATCH 4/8] [IMP] Chapter 5 Completed Remaining Exercises - Done Added Active fields with Default Value Added State field with their Values --- estate/models/estate_property.py | 25 ++++++++++++++++++------- 1 file changed, 18 insertions(+), 7 deletions(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 2999095095c..24655c6fe42 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -1,22 +1,23 @@ from odoo import models, fields +from datetime import timedelta class EstateProperty(models.Model): _name = "estate.property" _description = "Real Estate Property" - name = fields.Char() + name = fields.Char(string="Title", required=True) description = fields.Text() postcode = fields.Char() - date_availability = fields.Date() - expected_price = fields.Float() - selling_price = fields.Float() - bedrooms = fields.Integer() - living_area = fields.Integer() + date_availability = fields.Date(string="Available from", copy=False, default=lambda self: fields.Date.today() + timedelta(days=90)) + expected_price = fields.Float(required=True) + selling_price = fields.Float(readonly=True, copy=False) + bedrooms = fields.Integer(default=2) + living_area = fields.Integer(string="living Area (sqm)") facades = fields.Integer() garage = fields.Boolean() garden = fields.Boolean() - garden_area = fields.Integer() + garden_area = fields.Integer(string="Garden Area (sqm)") garden_orientation = fields.Selection( [ @@ -26,3 +27,13 @@ class EstateProperty(models.Model): ('west', 'West') ] ) + active = fields.Boolean(default=True) + state = fields.Selection( + [ + ('new', "New"), + ('offer_received', "Offer Received"), + ('offer_accepted', "Offer Accepted"), + ('sold', "Sold"), + ('cancelled', "Cancelled") + ], + string="Status", required=True, copy=False, default="new") From 559cc005c240905ce87fe47a7e3ff24206be8dfa Mon Sep 17 00:00:00 2001 From: "Mohit Ahir (moahi)" Date: Mon, 16 Mar 2026 19:02:53 +0530 Subject: [PATCH 5/8] [IMP] Chapter 6 Started 2 Exercise Solved Added List view to display records Added Form view to display one record in detail --- estate/models/estate_property.py | 2 +- estate/views/estate_property_views.xml | 60 ++++++++++++++++++++++++++ 2 files changed, 61 insertions(+), 1 deletion(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 24655c6fe42..91b5f42bdbe 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -11,7 +11,7 @@ class EstateProperty(models.Model): postcode = fields.Char() date_availability = fields.Date(string="Available from", copy=False, default=lambda self: fields.Date.today() + timedelta(days=90)) expected_price = fields.Float(required=True) - selling_price = fields.Float(readonly=True, copy=False) + selling_price = fields.Float(readonly=True, copy=True) bedrooms = fields.Integer(default=2) living_area = fields.Integer(string="living Area (sqm)") facades = fields.Integer() diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 911523488a5..e89ab978c4e 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -6,4 +6,64 @@ list,form + + estate.property.list + estate.property + + + + + + + + + + + + + + estate.property.form + estate.property + +
+ + +

+ +

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+ From 477e0a3bdbd938870da879a64406c41231fd34ac Mon Sep 17 00:00:00 2001 From: "Mohit Ahir (moahi)" Date: Tue, 17 Mar 2026 18:55:34 +0530 Subject: [PATCH 6/8] [IMP] Chapter 6 done Applied all requested changes Added Search view Added Domain by filter and Group by filter --- estate/models/estate_property.py | 13 +++++++------ estate/security/ir.model.access.csv | 2 +- estate/views/estate_property_views.xml | 21 ++++++++++++++++++++- 3 files changed, 28 insertions(+), 8 deletions(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 91b5f42bdbe..23c2519f8fe 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -1,9 +1,10 @@ from odoo import models, fields + from datetime import timedelta class EstateProperty(models.Model): - _name = "estate.property" + _name = 'estate.property' _description = "Real Estate Property" name = fields.Char(string="Title", required=True) @@ -21,10 +22,10 @@ class EstateProperty(models.Model): garden_orientation = fields.Selection( [ - ('north', 'North'), - ('south', 'South'), - ('east', 'East'), - ('west', 'West') + ('north', "North"), + ('south', "South"), + ('east', "East"), + ('west', "West") ] ) active = fields.Boolean(default=True) @@ -36,4 +37,4 @@ class EstateProperty(models.Model): ('sold', "Sold"), ('cancelled', "Cancelled") ], - string="Status", required=True, copy=False, default="new") + string="Status", required=True, copy=False, default='new') diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index 0e11f47e58d..32389642d4f 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_estate_property,access_estate_property,model_estate_property,base.group_user,1,1,1,1 \ No newline at end of file +access_estate_property,access_estate_property,model_estate_property,base.group_user,1,1,1,1 diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index e89ab978c4e..83d6f4631c6 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -20,12 +20,31 @@ + + + estate.property.search + estate.property + + + + + + + + + + + + + estate.property.form estate.property -
+

From 23ee740d3effacf38fff9c6eb6f12cf41f8b92f1 Mon Sep 17 00:00:00 2001 From: "Mohit Ahir (moahi)" Date: Thu, 19 Mar 2026 18:42:59 +0530 Subject: [PATCH 7/8] [IMP] Chapter 7 Started Completed Exercise 1 ,2 and 3 Added New models, fields, Action and Menus Completed Many2one and Many2many relations --- estate/__manifest__.py | 2 ++ estate/models/__init__.py | 2 ++ estate/models/estate_property.py | 7 ++++- estate/models/estate_property_tags.py | 8 +++++ estate/models/estate_property_type.py | 8 +++++ estate/security/ir.model.access.csv | 2 ++ estate/views/estate_menus.xml | 18 ++++++++++- estate/views/estate_property_tags_views.xml | 23 ++++++++++++++ estate/views/estate_property_type_views.xml | 33 +++++++++++++++++++++ estate/views/estate_property_views.xml | 14 +++++++-- 10 files changed, 113 insertions(+), 4 deletions(-) create mode 100644 estate/models/estate_property_tags.py create mode 100644 estate/models/estate_property_type.py create mode 100644 estate/views/estate_property_tags_views.xml create mode 100644 estate/views/estate_property_type_views.xml diff --git a/estate/__manifest__.py b/estate/__manifest__.py index d1919128a0b..01abd01612f 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -11,6 +11,8 @@ 'data': [ 'security/ir.model.access.csv', 'views/estate_property_views.xml', + 'views/estate_property_type_views.xml', + 'views/estate_property_tags_views.xml', 'views/estate_menus.xml', ], 'installable': True, diff --git a/estate/models/__init__.py b/estate/models/__init__.py index 5e1963c9d2f..337c9c6647d 100644 --- a/estate/models/__init__.py +++ b/estate/models/__init__.py @@ -1 +1,3 @@ from . import estate_property +from . import estate_property_type +from . import estate_property_tags diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 23c2519f8fe..ec6488826d2 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -5,7 +5,7 @@ class EstateProperty(models.Model): _name = 'estate.property' - _description = "Real Estate Property" + _description = 'Real Estate Property' name = fields.Char(string="Title", required=True) description = fields.Text() @@ -38,3 +38,8 @@ class EstateProperty(models.Model): ('cancelled', "Cancelled") ], string="Status", 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) + seller_id = fields.Many2one('res.users', string="Seller", default=lambda self: self.env.user) + tag_id = fields.Many2many('estate.property.tags', string="Property tags") diff --git a/estate/models/estate_property_tags.py b/estate/models/estate_property_tags.py new file mode 100644 index 00000000000..ea1ab90a783 --- /dev/null +++ b/estate/models/estate_property_tags.py @@ -0,0 +1,8 @@ +from odoo import models, fields + + +class Estatetags(models.Model): + _name = 'estate.property.tags' + _description = 'Estate 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..093a7aa65d4 --- /dev/null +++ b/estate/models/estate_property_type.py @@ -0,0 +1,8 @@ +from odoo import models, fields + + +class EstatePropertyType(models.Model): + _name = 'estate.property.type' + _description = 'Property Type' + + name = fields.Char(required=True) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index 32389642d4f..083f6a71130 100644 --- a/estate/security/ir.model.access.csv +++ b/estate/security/ir.model.access.csv @@ -1,2 +1,4 @@ id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink access_estate_property,access_estate_property,model_estate_property,base.group_user,1,1,1,1 +access_estate_property_type,access_estate_property_type,model_estate_property_type,base.group_user,1,1,1,1 +access_estate_property_tags,access_estate_property_tags,model_estate_property_tags,base.group_user,1,1,1,1 diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml index 686e96c8b62..ea24b683de7 100644 --- a/estate/views/estate_menus.xml +++ b/estate/views/estate_menus.xml @@ -9,10 +9,26 @@ name="Advertisements" parent="estate_menu_root"/> - + + + + + + + + + diff --git a/estate/views/estate_property_tags_views.xml b/estate/views/estate_property_tags_views.xml new file mode 100644 index 00000000000..82be2b34d55 --- /dev/null +++ b/estate/views/estate_property_tags_views.xml @@ -0,0 +1,23 @@ + + + + Property Tags + estate.property.tags + list,form + + + + estate.property.tags.form + estate.property.tags + + + + + + + + + + + + diff --git a/estate/views/estate_property_type_views.xml b/estate/views/estate_property_type_views.xml new file mode 100644 index 00000000000..f98f20d5f4d --- /dev/null +++ b/estate/views/estate_property_type_views.xml @@ -0,0 +1,33 @@ + + + + Property Types + estate.property.type + list,form + + + + estate.property.type.list + estate.property.type + + + + + + + + + estate.property.type.form + estate.property.type + +
+ + + + + +
+
+
+ +
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 83d6f4631c6..4d32ba69456 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -20,7 +20,7 @@ - + estate.property.search estate.property @@ -48,13 +48,15 @@

- +

+ + @@ -75,6 +77,14 @@ + + + + + + + + From 44870df7df2266b40e3df6d436dab6428875a738 Mon Sep 17 00:00:00 2001 From: "Mohit Ahir (moahi)" Date: Fri, 20 Mar 2026 18:50:11 +0530 Subject: [PATCH 8/8] [FIX] Chapter 7 Completed Completed last Exercise Added New model with their Form and list View Completed One2many relations --- estate/__manifest__.py | 1 + estate/models/__init__.py | 1 + estate/models/estate_property.py | 3 +- estate/models/estate_property_offer.py | 17 +++++++++++ estate/security/ir.model.access.csv | 1 + estate/views/estate_property_offer_views.xml | 32 ++++++++++++++++++++ estate/views/estate_property_views.xml | 4 +++ 7 files changed, 58 insertions(+), 1 deletion(-) create mode 100644 estate/models/estate_property_offer.py create mode 100644 estate/views/estate_property_offer_views.xml diff --git a/estate/__manifest__.py b/estate/__manifest__.py index 01abd01612f..1fc47e346b2 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -13,6 +13,7 @@ 'views/estate_property_views.xml', 'views/estate_property_type_views.xml', 'views/estate_property_tags_views.xml', + 'views/estate_property_offer_views.xml', 'views/estate_menus.xml', ], 'installable': True, diff --git a/estate/models/__init__.py b/estate/models/__init__.py index 337c9c6647d..f6ceb1825e3 100644 --- a/estate/models/__init__.py +++ b/estate/models/__init__.py @@ -1,3 +1,4 @@ from . import estate_property from . import estate_property_type from . import estate_property_tags +from . import estate_property_offer diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index ec6488826d2..ddb1dae392e 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -38,8 +38,9 @@ class EstateProperty(models.Model): ('cancelled', "Cancelled") ], string="Status", 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) seller_id = fields.Many2one('res.users', string="Seller", default=lambda self: self.env.user) tag_id = fields.Many2many('estate.property.tags', string="Property tags") + offer_ids = fields.One2many('estate.property.offer', 'property_id') diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py new file mode 100644 index 00000000000..fffbb4cfbaf --- /dev/null +++ b/estate/models/estate_property_offer.py @@ -0,0 +1,17 @@ +from odoo import models, fields + + +class EstatePropertyOffer(models.Model): + _name = 'estate.property.offer' + _description = 'Estate Property offer' + + price = fields.Float() + status = fields.Selection( + [ + ('Accepted', "Accepted"), + ('Refused', "Refused") + ] + , copy=False) + + partner_id = fields.Many2one('res.partner', required=True) + property_id = fields.Many2one('estate.property', required=True) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index 083f6a71130..7e28e608af5 100644 --- a/estate/security/ir.model.access.csv +++ b/estate/security/ir.model.access.csv @@ -2,3 +2,4 @@ id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink access_estate_property,access_estate_property,model_estate_property,base.group_user,1,1,1,1 access_estate_property_type,access_estate_property_type,model_estate_property_type,base.group_user,1,1,1,1 access_estate_property_tags,access_estate_property_tags,model_estate_property_tags,base.group_user,1,1,1,1 +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_property_offer_views.xml b/estate/views/estate_property_offer_views.xml new file mode 100644 index 00000000000..b6137ff8810 --- /dev/null +++ b/estate/views/estate_property_offer_views.xml @@ -0,0 +1,32 @@ + + + + estate.property.offer.list + estate.property.offer + + + + + + + + + + + estate.property.offer.form + estate.property.offer + +
+ + + + + + + + +
+
+
+ +
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 4d32ba69456..f31419a0429 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -88,6 +88,10 @@
+ + + +