From 98a475ef9b33c06c79231175dd607086d82f8c47 Mon Sep 17 00:00:00 2001 From: Hazem Ibrahim Date: Mon, 19 Jan 2026 14:17:57 +0100 Subject: [PATCH 1/6] ibrha - first commit --- website_airproof/__init__.py | 1 + 1 file changed, 1 insertion(+) diff --git a/website_airproof/__init__.py b/website_airproof/__init__.py index e69de29bb2d..860068760ad 100644 --- a/website_airproof/__init__.py +++ b/website_airproof/__init__.py @@ -0,0 +1 @@ +# first technical commit - ibrha \ No newline at end of file From a598836fa56af372d0d55f8ca0e265f035a1e85b Mon Sep 17 00:00:00 2001 From: Hazem Ibrahim Date: Tue, 20 Jan 2026 15:54:50 +0100 Subject: [PATCH 2/6] [ADD] estate: add real estate module This module provides managing real estate advertisements. It introduces the \`estate.property\` model with fields for: - Property descriptions (name, description, postcode, etc.) - Pricing (expected and selling price) - Physical characteristics (bedrooms, living area, facades) - Garden information (area and orientation) --- estate/__init__.py | 1 + estate/__manifest__.py | 16 ++++++++++++++++ estate/models/__init__.py | 1 + estate/models/estate_property.py | 23 +++++++++++++++++++++++ 4 files changed, 41 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..32e57c43c96 --- /dev/null +++ b/estate/__manifest__.py @@ -0,0 +1,16 @@ +{ + 'name': 'Real Estate', + 'category': 'tutorials', + 'summary': 'Real estate module to sell/rent houses and offices', + 'description': """ + immoweb alias xd + """, + 'website': 'https://www.odoo.com/app/real_estate', + 'depends': [ + 'base_setup', + 'base' + ], + 'installable': True, + 'application': True, + 'author': 'ibrha', +} \ 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..f5bde562fef --- /dev/null +++ b/estate/models/estate_property.py @@ -0,0 +1,23 @@ +from odoo import models, fields + +class Property(models.Model): + _name = "estate.property" + _description = "Estate Property app like immoweb" + + name = fields.Char(string="Title", required=True) + description = fields.Text(string="Description") + postcode = fields.Char(string="Postcode") + date_availability = fields.Date(string="Available From") + expected_price = fields.Float(string="Expected Price", required=True) + selling_price = fields.Float(string="Selling Price") + bedrooms = fields.Integer(string="Bedrooms") + living_area = fields.Integer(string="Living Area") + facades = fields.Integer(string="Facades") + garage = fields.Boolean(string="Garage") + garden = fields.Boolean(string="Garden") + garden_area = fields.Integer(string="Garden Area") + + garden_orientation = fields.Selection( + string='Garden Orientation', + selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')] + ) From de23281839057dcc4645583706cb7cac04931c5e Mon Sep 17 00:00:00 2001 From: Hazem Ibrahim Date: Wed, 21 Jan 2026 10:56:13 +0100 Subject: [PATCH 3/6] [IMP] estate: added access rights for estate property model --- estate/__manifest__.py | 4 ++++ estate/security/ir.model.access.csv | 2 ++ website_airproof/__init__.py | 1 - 3 files changed, 6 insertions(+), 1 deletion(-) create mode 100644 estate/security/ir.model.access.csv diff --git a/estate/__manifest__.py b/estate/__manifest__.py index 32e57c43c96..6ef31e26c1b 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -10,7 +10,11 @@ 'base_setup', 'base' ], + 'data': [ + 'security/ir.model.access.csv', + ], 'installable': True, 'application': True, 'author': 'ibrha', + 'license': 'LGPL-3', } \ 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..32389642d4f --- /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 diff --git a/website_airproof/__init__.py b/website_airproof/__init__.py index 860068760ad..e69de29bb2d 100644 --- a/website_airproof/__init__.py +++ b/website_airproof/__init__.py @@ -1 +0,0 @@ -# first technical commit - ibrha \ No newline at end of file From 6f915cdbac728986493bc3661d2ddf9a49af2752 Mon Sep 17 00:00:00 2001 From: Hazem Ibrahim Date: Wed, 21 Jan 2026 13:40:59 +0100 Subject: [PATCH 4/6] [IMP] estate: add views, menus and property fields adds the initial UI for the estate module: - Add action and menus in XML files. - Add new fields and modify the existing fields. - Update manifest to include the new XML files. --- estate/__manifest__.py | 2 ++ estate/models/estate_property.py | 26 +++++++++++++++++++++----- estate/views/estate_menus.xml | 7 +++++++ estate/views/estate_property_views.xml | 8 ++++++++ 4 files changed, 38 insertions(+), 5 deletions(-) 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 6ef31e26c1b..983a3a7198e 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -12,6 +12,8 @@ ], 'data': [ 'security/ir.model.access.csv', + 'views/estate_property_views.xml', + 'views/estate_menus.xml', ], 'installable': True, 'application': True, diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index f5bde562fef..0f074f1a401 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -1,3 +1,4 @@ +from dateutil.relativedelta import relativedelta from odoo import models, fields class Property(models.Model): @@ -7,17 +8,32 @@ class Property(models.Model): name = fields.Char(string="Title", required=True) description = fields.Text(string="Description") postcode = fields.Char(string="Postcode") - date_availability = fields.Date(string="Available From") + date_availability = fields.Date(string="Available From", default=fields.Date.today() + relativedelta(months=3), copy=False) expected_price = fields.Float(string="Expected Price", required=True) - selling_price = fields.Float(string="Selling Price") - bedrooms = fields.Integer(string="Bedrooms") + selling_price = fields.Float(string="Selling Price", readonly=True, copy=False) + bedrooms = fields.Integer(string="Bedrooms", default=2) living_area = fields.Integer(string="Living Area") facades = fields.Integer(string="Facades") garage = fields.Boolean(string="Garage") garden = fields.Boolean(string="Garden") garden_area = fields.Integer(string="Garden Area") - + active = fields.Boolean(string="Active", default=True) garden_orientation = fields.Selection( string='Garden Orientation', - selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')] + selection=[ + ('north', 'North'), + ('south', 'South'), + ('east', 'East'), + ('west', 'West') + ] ) + state = fields.Selection( + string='Status', + selection=[ + ('new', 'New'), + ('offer_received', 'Offer Received'), + ('offer_accepted', 'Offer Accepted'), + ('sold', 'Sold'), + ('cancelled', 'Cancelled') + ] + ) \ No newline at end of file diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml new file mode 100644 index 00000000000..85d1cd0cead --- /dev/null +++ b/estate/views/estate_menus.xml @@ -0,0 +1,7 @@ + + + + + + + \ No newline at end of file diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml new file mode 100644 index 00000000000..1d2a3aaa4cd --- /dev/null +++ b/estate/views/estate_property_views.xml @@ -0,0 +1,8 @@ + + + + Properties + estate.property + list,form + + From e9dffd5f4607873da2e0fa7984a6d624435e86ad Mon Sep 17 00:00:00 2001 From: Hazem Ibrahim Date: Thu, 22 Jan 2026 11:46:55 +0100 Subject: [PATCH 5/6] [IMP] Estate: Add list, form, search views Adds multiple ui views (list, form, search) 1) adding missing EOLs 2) updating fields in estate.property model --- estate/__init__.py | 2 +- estate/__manifest__.py | 2 +- estate/models/__init__.py | 2 +- estate/models/estate_property.py | 28 ++++----- estate/security/ir.model.access.csv | 1 + estate/views/estate_menus.xml | 4 +- estate/views/estate_property_views.xml | 79 ++++++++++++++++++++++++++ 7 files changed, 100 insertions(+), 18 deletions(-) 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 983a3a7198e..746e60e45ec 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -19,4 +19,4 @@ 'application': True, 'author': 'ibrha', 'license': 'LGPL-3', -} \ 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 0f074f1a401..387f6d76c8c 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -1,6 +1,7 @@ from dateutil.relativedelta import relativedelta from odoo import models, fields + class Property(models.Model): _name = "estate.property" _description = "Estate Property app like immoweb" @@ -19,21 +20,22 @@ class Property(models.Model): garden_area = fields.Integer(string="Garden Area") active = fields.Boolean(string="Active", default=True) garden_orientation = fields.Selection( - string='Garden Orientation', + string="Garden Orientation", selection=[ - ('north', 'North'), - ('south', 'South'), - ('east', 'East'), - ('west', 'West') + ("north", "North"), + ("south", "South"), + ("east", "East"), + ("west", "West") ] ) state = fields.Selection( - string='Status', + string="Status", selection=[ - ('new', 'New'), - ('offer_received', 'Offer Received'), - ('offer_accepted', 'Offer Accepted'), - ('sold', 'Sold'), - ('cancelled', 'Cancelled') - ] - ) \ No newline at end of file + ("new", "New"), + ("offer_received", "Offer Received"), + ("offer_accepted", "Offer Accepted"), + ("sold", "Sold"), + ("cancelled", "Cancelled") + ], + default="new" + ) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index 32389642d4f..d48cf7145c0 100644 --- a/estate/security/ir.model.access.csv +++ b/estate/security/ir.model.access.csv @@ -1,2 +1,3 @@ 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 + diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml index 85d1cd0cead..4b1d9709a03 100644 --- a/estate/views/estate_menus.xml +++ b/estate/views/estate_menus.xml @@ -1,7 +1,7 @@ - + - \ No newline at end of file + diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 1d2a3aaa4cd..307812e8f80 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -1,5 +1,84 @@ + + estate.property.view.list + estate.property + + + + + + + + + + + + + + + estate.property.view.form + estate.property + +
+ +

+ +

+ + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+ + + estate.property.view.search + estate.property + + + + + + + + + + + + + + + + Properties estate.property From be45f5d0b89312a79bb02f6591df86b9ff73aafe Mon Sep 17 00:00:00 2001 From: Hazem Ibrahim Date: Thu, 22 Jan 2026 16:50:55 +0100 Subject: [PATCH 6/6] [IMP] estate: add property offers, tags, types and relational fields - Define 'estate.property.offer' and 'estate.property.tag' and 'estate.property.type' models. - Add One2many relationship for offers and Many2many for tags, types on properties. - Update property views with a notebook containing 'Offers' and 'Other Info' tabs. - Add 'salesman' and 'buyer' fields to properties other info's section. --- estate/__manifest__.py | 2 ++ estate/models/__init__.py | 2 +- estate/models/estate_property.py | 11 +++++-- estate/models/estate_property_offer.py | 18 ++++++++++++ estate/models/estate_property_tag.py | 8 ++++++ estate/models/estate_property_type.py | 9 ++++++ estate/security/ir.model.access.csv | 4 ++- estate/views/estate_menus.xml | 4 +++ estate/views/estate_property_tag_views.xml | 32 +++++++++++++++++++++ estate/views/estate_property_type_views.xml | 32 +++++++++++++++++++++ estate/views/estate_property_views.xml | 23 ++++++++++++++- 11 files changed, 139 insertions(+), 6 deletions(-) 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/views/estate_property_tag_views.xml create mode 100644 estate/views/estate_property_type_views.xml diff --git a/estate/__manifest__.py b/estate/__manifest__.py index 746e60e45ec..771d7a8cd8b 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -12,6 +12,8 @@ ], 'data': [ 'security/ir.model.access.csv', + 'views/estate_property_type_views.xml', + 'views/estate_property_tag_views.xml', 'views/estate_property_views.xml', 'views/estate_menus.xml', ], diff --git a/estate/models/__init__.py b/estate/models/__init__.py index 5e1963c9d2f..93a6bd86abd 100644 --- a/estate/models/__init__.py +++ b/estate/models/__init__.py @@ -1 +1 @@ -from . import estate_property +from . import estate_property, estate_property_type, estate_property_tag, estate_property_offer diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 387f6d76c8c..3742fc3b6b4 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -9,6 +9,11 @@ class Property(models.Model): name = fields.Char(string="Title", required=True) description = fields.Text(string="Description") postcode = fields.Char(string="Postcode") + property_type_id = fields.Many2one("estate.property.type", string="Property Type") + offer_ids = fields.One2many("estate.property.offer", string="Offers", inverse_name="property_id") + tag_ids = fields.Many2many("estate.property.tag", string="Tags") + salesman = fields.Many2one("res.users", string ="Salesman", default=lambda self: self.env.uid) + buyer = fields.Many2one("res.partner", string="Buyer", copy=False) date_availability = fields.Date(string="Available From", default=fields.Date.today() + relativedelta(months=3), copy=False) expected_price = fields.Float(string="Expected Price", required=True) selling_price = fields.Float(string="Selling Price", readonly=True, copy=False) @@ -22,7 +27,7 @@ class Property(models.Model): garden_orientation = fields.Selection( string="Garden Orientation", selection=[ - ("north", "North"), + ("north", "North"), ("south", "South"), ("east", "East"), ("west", "West") @@ -31,10 +36,10 @@ class Property(models.Model): state = fields.Selection( string="Status", selection=[ - ("new", "New"), + ("new", "New"), ("offer_received", "Offer Received"), ("offer_accepted", "Offer Accepted"), - ("sold", "Sold"), + ("sold", "Sold"), ("cancelled", "Cancelled") ], default="new" diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py new file mode 100644 index 00000000000..dac61ce9ff3 --- /dev/null +++ b/estate/models/estate_property_offer.py @@ -0,0 +1,18 @@ +from odoo import models, fields + + +class PropertyOffer(models.Model): + _name = "estate.property.offer" + _description = "Estate Property Offer" + + price = fields.Float(string="Price", required=True) + status = fields.Selection( + string="Status", + selection=[ + ("accepted", "Accepted"), + ("refused", "Refused") + ], + copy=False + ) + property_id = fields.Many2one("estate.property", string="Property", required=True) + partner_id = fields.Many2one("res.partner", string="Partner", required=True) diff --git a/estate/models/estate_property_tag.py b/estate/models/estate_property_tag.py new file mode 100644 index 00000000000..47b67b49477 --- /dev/null +++ b/estate/models/estate_property_tag.py @@ -0,0 +1,8 @@ +from odoo import models, fields + + +class PropertyTag(models.Model): + _name = "estate.property.tag" + _description = "Estate Property Tag" + + name = fields.Char(string="Name", required=True) diff --git a/estate/models/estate_property_type.py b/estate/models/estate_property_type.py new file mode 100644 index 00000000000..6ad7355a54e --- /dev/null +++ b/estate/models/estate_property_type.py @@ -0,0 +1,9 @@ +from odoo import models, fields + + +class PropertyType(models.Model): + _name = "estate.property.type" + _description = "Estate Property Type" + + name = fields.Char(string="Name", required=True) + diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index d48cf7145c0..89f97c50842 100644 --- a/estate/security/ir.model.access.csv +++ b/estate/security/ir.model.access.csv @@ -1,3 +1,5 @@ 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_tag,access_estate_property_tag,model_estate_property_tag,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_menus.xml b/estate/views/estate_menus.xml index 4b1d9709a03..5da82161412 100644 --- a/estate/views/estate_menus.xml +++ b/estate/views/estate_menus.xml @@ -3,5 +3,9 @@ + + + +
diff --git a/estate/views/estate_property_tag_views.xml b/estate/views/estate_property_tag_views.xml new file mode 100644 index 00000000000..243a7300981 --- /dev/null +++ b/estate/views/estate_property_tag_views.xml @@ -0,0 +1,32 @@ + + + + estate.property.tag.view.list + estate.property.tag + + + + + + + + + estate.property.tag.view.form + estate.property.tag + +
+ + + + + +
+
+
+ + + Property Tags + estate.property.tag + list,form + +
diff --git a/estate/views/estate_property_type_views.xml b/estate/views/estate_property_type_views.xml new file mode 100644 index 00000000000..15d387ca8d4 --- /dev/null +++ b/estate/views/estate_property_type_views.xml @@ -0,0 +1,32 @@ + + + + estate.property.type.view.list + estate.property.type + + + + + + + + + estate.property.type.view.form + estate.property.type + +
+ + + + + +
+
+
+ + + Property Types + estate.property.type + list,form + +
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 307812e8f80..2d7b4ecf4be 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -7,6 +7,8 @@ + + @@ -27,6 +29,8 @@ + + @@ -50,6 +54,23 @@ + + + + + + + + + + + + + + + + + @@ -67,7 +88,7 @@ - +