From c7ff595ea7b9c5a8545792a309517c202d9f0653 Mon Sep 17 00:00:00 2001 From: Xavier Zambrano Date: Tue, 5 May 2026 11:02:56 -0600 Subject: [PATCH 1/6] [ADD] complete chapter 1,2 and 3 --- estate/__init__.py | 1 + estate/__manifest__.py | 8 ++++++++ estate/models/__init__.py | 1 + estate/models/estate_property.py | 23 +++++++++++++++++++++++ 4 files changed, 33 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..a922fdaa764 --- /dev/null +++ b/estate/__manifest__.py @@ -0,0 +1,8 @@ +{ + 'name': 'Estate', + 'depends': ['base'], + 'installable': True, + 'application': True, + 'author': 'Xavier Zambrano', + 'license': 'LGPL-3', +} 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..ddfa8b8d84d --- /dev/null +++ b/estate/models/estate_property.py @@ -0,0 +1,23 @@ +from odoo import fields, models + +class EstateProperty(models.Model): + _name = 'estate.property' + _description = 'Estate Property' + + name = fields.Char('Property Name', required=True) + description = fields.Text('Description') + postcode = fields.Char('Postcode') + date_availability = fields.Date('Date Availability') + expected_price = fields.Float('Expected Price', required=True) + selling_price = fields.Float('Selling Price') + bedrooms = fields.Integer('Bedrooms') + living_area = fields.Integer('Living Area') + facades = fields.Integer('Facades') + garage = fields.Boolean('Garage') + garden = fields.Boolean('Garden') + garden_area = fields.Integer('Garden Area') + garden_orientation = fields.Selection( + string='Garden Orientation', + selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')], + help='The orientation of the garden' + ) \ No newline at end of file From 00eeecbba2dbe9728dc1c31d544f4387f9310d2c Mon Sep 17 00:00:00 2001 From: Xavier Zambrano Date: Tue, 5 May 2026 11:17:23 -0600 Subject: [PATCH 2/6] [IMP] complete chapter 4 --- estate/__manifest__.py | 3 +++ estate/security/ir.model.access.csv | 2 ++ 2 files changed, 5 insertions(+) create mode 100644 estate/security/ir.model.access.csv diff --git a/estate/__manifest__.py b/estate/__manifest__.py index a922fdaa764..938e87e481c 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -5,4 +5,7 @@ 'application': True, 'author': 'Xavier Zambrano', 'license': 'LGPL-3', + 'data': [ + 'security/ir.model.access.csv' + ] } diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv new file mode 100644 index 00000000000..d01557a3fb7 --- /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_model,access_estate_property_model,model_estate_property,base.group_user,1,1,1,1 \ No newline at end of file From 32f8f4c0e481f61172fbb4e922ed3c601f012830 Mon Sep 17 00:00:00 2001 From: Xavier Zambrano Date: Tue, 5 May 2026 15:57:23 -0600 Subject: [PATCH 3/6] [IMP] complete chapter 5 --- estate/__manifest__.py | 5 ++++- estate/models/estate_property.py | 20 ++++++++++++++------ estate/views/estate_property_views.xml | 16 ++++++++++++++++ estate/views/real_estate_menus.xml | 8 ++++++++ 4 files changed, 42 insertions(+), 7 deletions(-) create mode 100644 estate/views/estate_property_views.xml create mode 100644 estate/views/real_estate_menus.xml diff --git a/estate/__manifest__.py b/estate/__manifest__.py index 938e87e481c..bfe7799be66 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -6,6 +6,9 @@ 'author': 'Xavier Zambrano', 'license': 'LGPL-3', 'data': [ - 'security/ir.model.access.csv' + 'security/ir.model.access.csv', + + 'views/estate_property_views.xml', + 'views/real_estate_menus.xml', ] } diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index ddfa8b8d84d..a7867565a3d 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -7,17 +7,25 @@ class EstateProperty(models.Model): name = fields.Char('Property Name', required=True) description = fields.Text('Description') postcode = fields.Char('Postcode') - date_availability = fields.Date('Date Availability') + date_availability = fields.Date('Available From', copy=False, default=fields.Date.add(fields.Date.today(), months=3)) expected_price = fields.Float('Expected Price', required=True) - selling_price = fields.Float('Selling Price') - bedrooms = fields.Integer('Bedrooms') - living_area = fields.Integer('Living Area') + selling_price = fields.Float('Selling Price', readonly=True, copy=False) + bedrooms = fields.Integer('Bedrooms', default=2) + living_area = fields.Integer('Living Area (sqm)') facades = fields.Integer('Facades') garage = fields.Boolean('Garage') garden = fields.Boolean('Garden') - garden_area = fields.Integer('Garden Area') + garden_area = fields.Integer('Garden Area (sqm)') garden_orientation = fields.Selection( string='Garden Orientation', selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')], help='The orientation of the garden' - ) \ No newline at end of file + ) + state = fields.Selection( + string='Status', + selection=[('new', 'New'), ('offer_received', 'Offer Received'), ('offer_accepted', 'Offer Accepted'), ('sold', 'Sold'), ('cancelled', 'Cancelled')], + required=True, + copy=False, + default='new' + ) + active = fields.Boolean('Active', default=True) \ 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..09fe07daf8f --- /dev/null +++ b/estate/views/estate_property_views.xml @@ -0,0 +1,16 @@ + + + + Properties + estate.property + list,form + +

+ Create a new property +

+

+ Let your customers know about your great Real Estate options! +

+
+
+
diff --git a/estate/views/real_estate_menus.xml b/estate/views/real_estate_menus.xml new file mode 100644 index 00000000000..df603656afc --- /dev/null +++ b/estate/views/real_estate_menus.xml @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file From d07669ba27acf9010a7723076a7773b2edd4069e Mon Sep 17 00:00:00 2001 From: Xavier Zambrano Date: Wed, 6 May 2026 12:51:05 -0600 Subject: [PATCH 4/6] [IMP] complete chapter 6 --- estate/views/estate_property_views.xml | 87 +++++++++++++++++++++++++- 1 file changed, 85 insertions(+), 2 deletions(-) diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 09fe07daf8f..b492f1768fa 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -1,5 +1,88 @@ + + + + estate.property.view.search + estate.property + + + + + + + + + + + + + + + + + + estate.property.form + estate.property + +
+ + + +
+

+ +

+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+
+ + + estate.property.list + estate.property + + + + + + + + + + + + + Properties estate.property @@ -12,5 +95,5 @@ Let your customers know about your great Real Estate options!

-
-
+ + \ No newline at end of file From f50ac04f2fe57d44943aee241b2c416343da4881 Mon Sep 17 00:00:00 2001 From: Xavier Zambrano Date: Wed, 6 May 2026 14:25:27 -0600 Subject: [PATCH 5/6] [REF] models: set name string to "Title" --- estate/models/estate_property.py | 2 +- estate/views/estate_property_views.xml | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index a7867565a3d..26cef3926b7 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -4,7 +4,7 @@ class EstateProperty(models.Model): _name = 'estate.property' _description = 'Estate Property' - name = fields.Char('Property Name', required=True) + name = fields.Char('Title', required=True) description = fields.Text('Description') postcode = fields.Char('Postcode') date_availability = fields.Date('Available From', copy=False, default=fields.Date.add(fields.Date.today(), months=3)) diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index b492f1768fa..e115e568359 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -7,7 +7,7 @@ estate.property - + @@ -72,7 +72,7 @@ estate.property - + From 31d022892c3dc2eb01d303d5e0e3db7cdd22d048 Mon Sep 17 00:00:00 2001 From: Xavier Zambrano Date: Wed, 6 May 2026 15:01:11 -0600 Subject: [PATCH 6/6] [IMP] complete chapter 7 --- estate/__manifest__.py | 4 +- estate/models/__init__.py | 5 ++- estate/models/estate_property.py | 24 ++++++----- 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 | 5 ++- estate/views/estate_property_offer_views.xml | 32 ++++++++++++++ estate/views/estate_property_tag_views.xml | 42 +++++++++++++++++++ estate/views/estate_property_type_views.xml | 44 ++++++++++++++++++++ estate/views/estate_property_views.xml | 24 +++++++++-- estate/views/real_estate_menus.xml | 4 ++ 12 files changed, 195 insertions(+), 16 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_offer_views.xml 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 bfe7799be66..c01354ae362 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -1,13 +1,15 @@ { 'name': 'Estate', 'depends': ['base'], - 'installable': True, 'application': True, 'author': 'Xavier Zambrano', 'license': 'LGPL-3', 'data': [ 'security/ir.model.access.csv', + 'views/estate_property_offer_views.xml', + 'views/estate_property_tag_views.xml', + 'views/estate_property_type_views.xml', 'views/estate_property_views.xml', 'views/real_estate_menus.xml', ] diff --git a/estate/models/__init__.py b/estate/models/__init__.py index f4c8fd6db6d..09b2099fe84 100644 --- a/estate/models/__init__.py +++ b/estate/models/__init__.py @@ -1 +1,4 @@ -from . import estate_property \ No newline at end of file +from . import estate_property +from . import estate_property_type +from . import estate_property_tag +from . import estate_property_offer \ No newline at end of file diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 26cef3926b7..e1e600d4008 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -5,19 +5,18 @@ class EstateProperty(models.Model): _description = 'Estate Property' name = fields.Char('Title', required=True) - description = fields.Text('Description') - postcode = fields.Char('Postcode') + description = fields.Text() + postcode = fields.Char() date_availability = fields.Date('Available From', copy=False, default=fields.Date.add(fields.Date.today(), months=3)) - expected_price = fields.Float('Expected Price', required=True) - selling_price = fields.Float('Selling Price', readonly=True, copy=False) - bedrooms = fields.Integer('Bedrooms', default=2) + expected_price = fields.Float(required=True) + selling_price = fields.Float(readonly=True, copy=False) + bedrooms = fields.Integer(default=2) living_area = fields.Integer('Living Area (sqm)') - facades = fields.Integer('Facades') - garage = fields.Boolean('Garage') - garden = fields.Boolean('Garden') + facades = fields.Integer() + garage = fields.Boolean() + garden = fields.Boolean() garden_area = fields.Integer('Garden Area (sqm)') garden_orientation = fields.Selection( - string='Garden Orientation', selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')], help='The orientation of the garden' ) @@ -28,4 +27,9 @@ class EstateProperty(models.Model): copy=False, default='new' ) - active = fields.Boolean('Active', default=True) \ No newline at end of file + active = fields.Boolean(default=True) + property_type_id = fields.Many2one('estate.property.type') + buyer_id = fields.Many2one('res.partner', copy=False) + salesperson_id = fields.Many2one('res.users', string='Salesman', default=lambda self: self.env.user) + property_tag_ids = fields.Many2many('estate.property.tag') + property_offer_ids = fields.One2many('estate.property.offer', 'property_id') \ No newline at end of file diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py new file mode 100644 index 00000000000..04b6e0830b8 --- /dev/null +++ b/estate/models/estate_property_offer.py @@ -0,0 +1,11 @@ +from odoo import fields, models + + +class EstatePropertyOffer(models.Model): + _name = 'estate.property.offer' + _description = 'Estate Property Offer' + + price = fields.Float() + status = fields.Selection(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/models/estate_property_tag.py b/estate/models/estate_property_tag.py new file mode 100644 index 00000000000..9bd29b43813 --- /dev/null +++ b/estate/models/estate_property_tag.py @@ -0,0 +1,8 @@ +from odoo import fields, models + + +class EstatePropertyTag(models.Model): + _name = 'estate.property.tag' + _description = 'Estate Property Tag' + + name = fields.Char(required=True) \ No newline at end of file diff --git a/estate/models/estate_property_type.py b/estate/models/estate_property_type.py new file mode 100644 index 00000000000..649283a8602 --- /dev/null +++ b/estate/models/estate_property_type.py @@ -0,0 +1,8 @@ +from odoo import fields, models + +class EstatePropertyType(models.Model): + _name = 'estate.property.type' + _description = 'Estate Property Type' + + name = fields.Char(required=True) + \ No newline at end of file diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index d01557a3fb7..99c7fd33ec3 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_estate_property_model,access_estate_property_model,model_estate_property,base.group_user,1,1,1,1 \ No newline at end of file +access_estate_property_model,access_estate_property_model,model_estate_property,base.group_user,1,1,1,1 +access_estate_property_type_model,access_estate_property_type_model,model_estate_property_type,base.group_user,1,1,1,1 +access_estate_property_tag_model,access_estate_property_tag_model,model_estate_property_tag,base.group_user,1,1,1,1 +access_estate_property_offer,access_estate_property_offer_model,model_estate_property_offer,base.group_user,1,1,1,1 \ No newline at end of file diff --git a/estate/views/estate_property_offer_views.xml b/estate/views/estate_property_offer_views.xml new file mode 100644 index 00000000000..77e86ec2bac --- /dev/null +++ b/estate/views/estate_property_offer_views.xml @@ -0,0 +1,32 @@ + + + + + estate.property.offer.form + estate.property.offer + +
+ + + + + + + +
+
+
+ + + estate.property.offer.list + estate.property.offer + + + + + + + + + +
diff --git a/estate/views/estate_property_tag_views.xml b/estate/views/estate_property_tag_views.xml new file mode 100644 index 00000000000..bc27e0644da --- /dev/null +++ b/estate/views/estate_property_tag_views.xml @@ -0,0 +1,42 @@ + + + + + estate.property.tag.form + estate.property.tag + +
+ + + + + +
+
+
+ + + estate.property.tag.list + estate.property.tag + + + + + + + + + Property Tags + estate.property.tag + list,form + +

+ Create a new property tag +

+

+ Let your customers know about your great Real Estate property tag options! +

+
+
+ +
diff --git a/estate/views/estate_property_type_views.xml b/estate/views/estate_property_type_views.xml new file mode 100644 index 00000000000..638720e8063 --- /dev/null +++ b/estate/views/estate_property_type_views.xml @@ -0,0 +1,44 @@ + + + + + estate.property.type.form + estate.property.type + +
+ +
+

+ +

+
+
+
+
+
+ + + estate.property.type.list + estate.property.type + + + + + + + + + Property Types + estate.property.type + list,form + +

+ Create a new property type +

+

+ Let your customers know about your great Real Estate property types options! +

+
+
+ +
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index e115e568359..587e08dacee 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -8,6 +8,7 @@ + @@ -30,13 +31,16 @@
-

+

+ + + @@ -61,18 +65,31 @@ + + + + + + + + + + + - + estate.property.list estate.property + + @@ -96,4 +113,5 @@

- \ No newline at end of file + + diff --git a/estate/views/real_estate_menus.xml b/estate/views/real_estate_menus.xml index df603656afc..424607fa2fe 100644 --- a/estate/views/real_estate_menus.xml +++ b/estate/views/real_estate_menus.xml @@ -4,5 +4,9 @@ + + + + \ No newline at end of file