From 5c40c2382d147dae82d26a0f3742cbcb6a6d3775 Mon Sep 17 00:00:00 2001 From: abdrahmanrashed Date: Mon, 19 May 2025 15:16:34 +0200 Subject: [PATCH 01/23] [ADD] created estate module - ch2 --- estate/__init__.py | 0 estate/__manifest__.py | 10 ++++++++++ 2 files changed, 10 insertions(+) create mode 100644 estate/__init__.py create mode 100644 estate/__manifest__.py diff --git a/estate/__init__.py b/estate/__init__.py new file mode 100644 index 00000000000..e69de29bb2d diff --git a/estate/__manifest__.py b/estate/__manifest__.py new file mode 100644 index 00000000000..e6dc01f1001 --- /dev/null +++ b/estate/__manifest__.py @@ -0,0 +1,10 @@ +{ + 'name': 'ESTATE', + 'description': "aras estate tutorial module", + 'website': 'https://www.odoo.com/page/estate', + 'depends': [ + 'base', + ], + 'installable': True, + 'application': True, +} \ No newline at end of file From 93035f1ff2c69e3c7a859cf64375b55a260ebc17 Mon Sep 17 00:00:00 2001 From: abdrahmanrashed Date: Tue, 20 May 2025 11:37:45 +0200 Subject: [PATCH 02/23] [IMP] added estate property model and access csv - ch3 & ch4 --- estate/__init__.py | 1 + estate/__manifest__.py | 5 ++++- estate/models/__init__.py | 1 + estate/models/estate_property.py | 26 ++++++++++++++++++++++++++ estate/security/ir.model.access.csv | 2 ++ 5 files changed, 34 insertions(+), 1 deletion(-) 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 index e69de29bb2d..0650744f6bc 100644 --- a/estate/__init__.py +++ b/estate/__init__.py @@ -0,0 +1 @@ +from . import models diff --git a/estate/__manifest__.py b/estate/__manifest__.py index e6dc01f1001..ae59d923abe 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -5,6 +5,9 @@ '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 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..09ae0531384 --- /dev/null +++ b/estate/models/estate_property.py @@ -0,0 +1,26 @@ +from odoo import fields, models + + +class EstateProperty(models.Model): + _name = "estate.property" + _description = "ch3 exercise tutorial" + + name = fields.Char(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() + facades = fields.Integer() + garage = fields.Boolean() + garden = fields.Boolean() + garden_area = fields.Integer() + garden_orientation = fields.Selection( + string = 'Type', + selection = [('north', 'North'), ('south', 'South'), ('east', 'East'), ('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..405022dbef4 --- /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 From f6bc65e6a4098a895f91891291e577dd6f5509a8 Mon Sep 17 00:00:00 2001 From: abdrahmanrashed Date: Tue, 20 May 2025 15:28:18 +0200 Subject: [PATCH 03/23] [IMP] added estate views - ch5 --- estate/__manifest__.py | 2 ++ estate/models/estate_property.py | 14 +++++++++----- estate/views/estate_menus.xml | 9 +++++++++ estate/views/estate_property_views.xml | 9 +++++++++ 4 files changed, 29 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 ae59d923abe..277803288d8 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -7,6 +7,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 09ae0531384..cc4459a1272 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -1,24 +1,28 @@ from odoo import fields, models - class EstateProperty(models.Model): _name = "estate.property" _description = "ch3 exercise tutorial" name = fields.Char(required=True) + active = False + state = fields.Selection( + string='Property State', + selection= [('new', 'New'), ('offer_received', 'Offer Received'), ('offer_accepted', 'Offer Accepted'), ('sold', 'Sold'), ('cancelled', 'Cancelled')] + ) description = fields.Text() postcode = fields.Char() - date_availability = fields.Date() + date_availability = fields.Date(default=lambda _: fields.Date.add(fields.Date.today(), months=3) , copy=False) expected_price = fields.Float() - 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( - string = 'Type', + string = 'Garden Orientation', selection = [('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')] ) diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml new file mode 100644 index 00000000000..15b9f086f35 --- /dev/null +++ b/estate/views/estate_menus.xml @@ -0,0 +1,9 @@ + + + + + + + + + diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml new file mode 100644 index 00000000000..de1dcecaa8f --- /dev/null +++ b/estate/views/estate_property_views.xml @@ -0,0 +1,9 @@ + + + + + Estate Property + estate.property + list,form + + From 69db2b7af5efda5c7665bd3e6bce7bc9423c7ca3 Mon Sep 17 00:00:00 2001 From: abdrahmanrashed Date: Tue, 20 May 2025 17:12:27 +0200 Subject: [PATCH 04/23] [IMP] created view layouts for list, form, and search - ch6 --- estate/models/estate_property.py | 2 +- estate/views/estate_property_views.xml | 83 ++++++++++++++++++++++++++ 2 files changed, 84 insertions(+), 1 deletion(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index cc4459a1272..596470f710f 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -5,7 +5,7 @@ class EstateProperty(models.Model): _description = "ch3 exercise tutorial" name = fields.Char(required=True) - active = False + active = fields.Boolean(default=False) state = fields.Selection( string='Property State', selection= [('new', 'New'), ('offer_received', 'Offer Received'), ('offer_accepted', 'Offer Accepted'), ('sold', 'Sold'), ('cancelled', 'Cancelled')] diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index de1dcecaa8f..0a90d7b7957 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -6,4 +6,87 @@ estate.property list,form + + + estate.property.list + estate.property + + + + + + + + + + + + + + + estate.property.form + estate.property + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+
+
+ + + estate.property.search + estate.property + + + + + + + + + + + + + + + + From 1c513a2d228a3c2961e76d6e2234989701734019 Mon Sep 17 00:00:00 2001 From: abdrahmanrashed Date: Wed, 21 May 2025 15:21:33 +0200 Subject: [PATCH 05/23] [IMP] -Created the estate property type model with form and list views, and added ability to set property types for properties. -Created the estate property tag with action, to add tags to properties. -Added settings menu to contain the type and tag menus. -Created estate property offer model with list and form views, to provide ability for partners to set offers on available properties. -Gave security access to new models. --- estate/__manifest__.py | 3 ++ estate/models/__init__.py | 3 ++ estate/models/estate_property.py | 10 +++-- estate/models/estate_property_offer.py | 14 +++++++ estate/models/estate_property_tag.py | 7 ++++ estate/models/estate_property_type.py | 7 ++++ estate/security/ir.model.access.csv | 3 ++ estate/views/estate_menus.xml | 7 +++- estate/views/estate_property_offer_views.xml | 31 ++++++++++++++ estate/views/estate_property_tag_views.xml | 8 ++++ estate/views/estate_property_type_views.xml | 43 ++++++++++++++++++++ estate/views/estate_property_views.xml | 32 +++++++++------ 12 files changed, 150 insertions(+), 18 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 277803288d8..deee250ad12 100644 --- a/estate/__manifest__.py +++ b/estate/__manifest__.py @@ -8,6 +8,9 @@ 'data': [ 'security/ir.model.access.csv', 'views/estate_property_views.xml', + 'views/estate_property_type_views.xml', + 'views/estate_property_tag_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 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 596470f710f..177c4a2a399 100644 --- a/estate/models/estate_property.py +++ b/estate/models/estate_property.py @@ -5,7 +5,7 @@ class EstateProperty(models.Model): _description = "ch3 exercise tutorial" name = fields.Char(required=True) - active = fields.Boolean(default=False) + # active = fields.Boolean(default=False) state = fields.Selection( string='Property State', selection= [('new', 'New'), ('offer_received', 'Offer Received'), ('offer_accepted', 'Offer Accepted'), ('sold', 'Sold'), ('cancelled', 'Cancelled')] @@ -25,6 +25,8 @@ class EstateProperty(models.Model): string = 'Garden Orientation', selection = [('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')] ) - - - \ No newline at end of file + property_type_id = fields.Many2one("estate.property.type", string="Property Type") + buyer = fields.Many2one("res.partner", string="Buyer", copy=False) + sales_person = fields.Many2one("res.users", string="Sales Person", default=lambda self: self.env.user) + property_tag_ids = fields.Many2many("estate.property.tag") + 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..4d213b3e1e9 --- /dev/null +++ b/estate/models/estate_property_offer.py @@ -0,0 +1,14 @@ +from odoo import fields, models + +class EstatePropertyOffer(models.Model): + _name = "estate.property.offer" + _description = "ch7 exercise tutorial" + + price = fields.Float() + status = fields.Selection( + copy=False, + string='Offer Status', + selection= [('accepted', 'Accepted'), ('refused', 'Refused')] + ) + partner_id = fields.Many2one("res.partner", string="Partner", 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..8ee25409072 --- /dev/null +++ b/estate/models/estate_property_tag.py @@ -0,0 +1,7 @@ +from odoo import fields, models + +class EstatePropertyTag(models.Model): + _name = "estate.property.tag" + _description = "ch7 exercise tutorial" + + 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..f846de27936 --- /dev/null +++ b/estate/models/estate_property_type.py @@ -0,0 +1,7 @@ +from odoo import fields, models + +class EstatePropertyType(models.Model): + _name = "estate.property.type" + _description = "ch7 exercise tutorial" + + name = fields.Char(required=True) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv index 405022dbef4..6587527c500 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","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 \ No newline at end of file diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml index 15b9f086f35..000f1996766 100644 --- a/estate/views/estate_menus.xml +++ b/estate/views/estate_menus.xml @@ -1,9 +1,14 @@ - + + + + + + diff --git a/estate/views/estate_property_offer_views.xml b/estate/views/estate_property_offer_views.xml new file mode 100644 index 00000000000..53ad6341902 --- /dev/null +++ b/estate/views/estate_property_offer_views.xml @@ -0,0 +1,31 @@ + + + + + estate.property.offer.list + estate.property.offer + + + + + + + + + + + estate.property.offer.form + 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..968ba9bcde6 --- /dev/null +++ b/estate/views/estate_property_tag_views.xml @@ -0,0 +1,8 @@ + + + + + + estate.property.tag + + diff --git a/estate/views/estate_property_type_views.xml b/estate/views/estate_property_type_views.xml new file mode 100644 index 00000000000..0431ed7b535 --- /dev/null +++ b/estate/views/estate_property_type_views.xml @@ -0,0 +1,43 @@ + + + + + Property Types + estate.property.type + list,form,search + + + + estate.property.type.list + estate.property.type + + + + + + + + + estate.property.type.form + estate.property.type + +
+ + + + + +
+
+
+ + + estate.property.type.search + estate.property.type + + + + + + +
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 0a90d7b7957..66b593daa39 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -4,7 +4,7 @@ Estate Property estate.property - list,form + list,form,search @@ -13,6 +13,8 @@ + + @@ -32,36 +34,40 @@ + + + - - - - - - - - + + + + + - + + + + - + + @@ -76,17 +82,17 @@ + - + - From 52dd22078329669c0745f72a814e5fa8cf93141a Mon Sep 17 00:00:00 2001 From: abdrahmanrashed Date: Wed, 21 May 2025 17:34:41 +0200 Subject: [PATCH 06/23] [IMP] - Created total area field in estate property model to present the total area of the property, and added a compute method for it. - Created a best price field in estate method to determine the best price in all offers, and a compute method for it. -Created deadline and validity fields for offers to set a maximum time for offer validities, and created compute methods to compute them from each other so only 1 value needs to be inserted. - (ch8 tutorial) --- estate/models/estate_property.py | 26 +++++++++++++++++++- estate/models/estate_property_offer.py | 14 ++++++++++- estate/views/estate_property_offer_views.xml | 4 +++ estate/views/estate_property_tag_views.xml | 2 +- estate/views/estate_property_views.xml | 2 ++ 5 files changed, 45 insertions(+), 3 deletions(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index 177c4a2a399..fc9a3cf8d4f 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 EstateProperty(models.Model): _name = "estate.property" @@ -30,3 +30,27 @@ class EstateProperty(models.Model): sales_person = fields.Many2one("res.users", string="Sales Person", default=lambda self: self.env.user) property_tag_ids = fields.Many2many("estate.property.tag") offer_ids = fields.One2many("estate.property.offer", "property_id") + total_area = fields.Integer(compute="_compute_total_area") + best_price = fields.Float(compute="_compute_best_price", string="Best Offer") + + @api.depends("living_area", "garden_area") + def _compute_total_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): + self.best_price = 0 + for id in self.offer_ids: + if self.best_price < id.price: + self.best_price = id.price + + @api.onchange("garden") + def _onchange_partner_id(self): + if self.garden: + self.garden_area = 10 + self.garden_orientation = 'north' + else: + self.garden_area = None + self.garden_orientation = None + diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py index 4d213b3e1e9..f01ae674fc6 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 fields, models, api class EstatePropertyOffer(models.Model): _name = "estate.property.offer" @@ -12,3 +12,15 @@ class EstatePropertyOffer(models.Model): ) partner_id = fields.Many2one("res.partner", string="Partner", required=True) property_id = fields.Many2one("estate.property", string="Property", required=True) + validity = fields.Integer(default=7, string="Validity") + date_deadline = fields.Date(compute="_compute_date_deadline", inverse="_inverse_date_deadline", string="Deadline") + create_date = fields.Date(default=fields.Date.today) + + @api.depends("create_date", "validity") + def _compute_date_deadline(self): + for offer in self: + offer.date_deadline = fields.Date.add(offer.create_date, days=offer.validity) + + def _inverse_date_deadline(self): + for offer in self: + offer.validity = (offer.date_deadline - offer.create_date).days diff --git a/estate/views/estate_property_offer_views.xml b/estate/views/estate_property_offer_views.xml index 53ad6341902..d5e91059d3b 100644 --- a/estate/views/estate_property_offer_views.xml +++ b/estate/views/estate_property_offer_views.xml @@ -8,6 +8,8 @@ + + @@ -22,6 +24,8 @@ + + diff --git a/estate/views/estate_property_tag_views.xml b/estate/views/estate_property_tag_views.xml index 968ba9bcde6..d0241b3e72b 100644 --- a/estate/views/estate_property_tag_views.xml +++ b/estate/views/estate_property_tag_views.xml @@ -2,7 +2,7 @@ - + Property Tags estate.property.tag diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml index 66b593daa39..2e1924cc83a 100644 --- a/estate/views/estate_property_views.xml +++ b/estate/views/estate_property_views.xml @@ -41,6 +41,7 @@ + @@ -55,6 +56,7 @@ + From 9dcc21f3f570c8bd572ffa58ccdec2cb7634b4b1 Mon Sep 17 00:00:00 2001 From: abdrahmanrashed Date: Thu, 22 May 2025 10:40:52 +0200 Subject: [PATCH 07/23] [IMP] -Created header buttons and actions to set the status(cancelled/sold) of estate.property. -Created buttons and actions to set the status(accepted/refused) of estate.property.offer. -Update the buyer and selling_price of estate.property when a correspondung estate.property.offer is accepted. -Set all action/compute loops to iterate on records for better practice and clarity. -(ch9) --- estate/models/estate_property.py | 45 ++++++++++++++------ estate/models/estate_property_offer.py | 27 +++++++++--- estate/views/estate_property_offer_views.xml | 2 + estate/views/estate_property_views.xml | 5 +++ 4 files changed, 60 insertions(+), 19 deletions(-) diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py index fc9a3cf8d4f..1e13f7e0633 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 api, fields, models, exceptions class EstateProperty(models.Model): _name = "estate.property" @@ -7,7 +7,7 @@ class EstateProperty(models.Model): name = fields.Char(required=True) # active = fields.Boolean(default=False) state = fields.Selection( - string='Property State', + string='Status', selection= [('new', 'New'), ('offer_received', 'Offer Received'), ('offer_accepted', 'Offer Accepted'), ('sold', 'Sold'), ('cancelled', 'Cancelled')] ) description = fields.Text() @@ -35,22 +35,39 @@ class EstateProperty(models.Model): @api.depends("living_area", "garden_area") def _compute_total_area(self): - for property in self: - property.total_area = property.living_area + property.garden_area + for record in self: + record.total_area = record.living_area + record.garden_area @api.depends("offer_ids.price") def _compute_best_price(self): - self.best_price = 0 - for id in self.offer_ids: - if self.best_price < id.price: - self.best_price = id.price + for record in self: + record.best_price = 0 + for id in record.offer_ids: + if record.best_price < id.price: + record.best_price = id.price @api.onchange("garden") def _onchange_partner_id(self): - if self.garden: - self.garden_area = 10 - self.garden_orientation = 'north' - else: - self.garden_area = None - self.garden_orientation = None + for record in self: + if record.garden: + record.garden_area = 10 + record.garden_orientation = 'north' + else: + record.garden_area = None + record.garden_orientation = None + def action_property_sold(self): + for record in self: + if record.state == 'cancelled': + raise exceptions.UserError("Cancelled properties cannot be sold.") + else: + record.state = 'sold' + return True + + def action_property_cancelled(self): + for record in self: + if record.state == 'sold': + raise exceptions.UserError("Sold properties cannot be cancelled.") + else: + record.state = 'cancelled' + return True diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py index f01ae674fc6..64ce7f84e61 100644 --- a/estate/models/estate_property_offer.py +++ b/estate/models/estate_property_offer.py @@ -1,4 +1,4 @@ -from odoo import fields, models, api +from odoo import fields, models, api, exceptions class EstatePropertyOffer(models.Model): _name = "estate.property.offer" @@ -18,9 +18,26 @@ class EstatePropertyOffer(models.Model): @api.depends("create_date", "validity") def _compute_date_deadline(self): - for offer in self: - offer.date_deadline = fields.Date.add(offer.create_date, days=offer.validity) + for record in self: + record.date_deadline = fields.Date.add(record.create_date, days=record.validity) def _inverse_date_deadline(self): - for offer in self: - offer.validity = (offer.date_deadline - offer.create_date).days + for record in self: + record.validity = (record.date_deadline - record.create_date).days + + def action_offer_accept(self): + for record in self: + if record.status == 'accepted': + raise exceptions.UserError("A offer has already been accepted.") + record.status = 'accepted' + for id in record.property_id: + id.selling_price = record.price + id.buyer = record.partner_id + return True + + def action_offer_refuse(self): + for record in self: + if record.status == 'accepted': + raise exceptions.UserError("You cannot refuse an accepted offer.") + record.status = 'refused' + return True diff --git a/estate/views/estate_property_offer_views.xml b/estate/views/estate_property_offer_views.xml index d5e91059d3b..b3a9eddd6a1 100644 --- a/estate/views/estate_property_offer_views.xml +++ b/estate/views/estate_property_offer_views.xml @@ -10,6 +10,8 @@ +