diff --git a/estate/__init__.py b/estate/__init__.py new file mode 100644 index 00000000000..d6210b1285d --- /dev/null +++ b/estate/__init__.py @@ -0,0 +1,3 @@ +# Part of Odoo. See LICENSE file for full copyright and licensing details. + +from . import models diff --git a/estate/__manifest__.py b/estate/__manifest__.py new file mode 100644 index 00000000000..738e13396be --- /dev/null +++ b/estate/__manifest__.py @@ -0,0 +1,22 @@ +{ + 'name': 'Estate', + "version": "1.0", + "category": "Real Estate/Brokerage", + 'data': [ + 'security/security.xml', + 'security/ir.model.access.csv', + 'views/estate_property_views.xml', + 'views/estate_property_type_views.xml', + 'views/estate_property_tag_views.xml', + 'views/res_users_views.xml', + 'views/estate_menus.xml', + 'data/estate_property_type.xml', + ], + 'demo': [ + 'demo/estate_property_demo.xml', + 'demo/estate_property_offer.xml' + ], + 'application': True, + 'installable': True, + 'license': 'LGPL-3', +} diff --git a/estate/data/estate_property_type.xml b/estate/data/estate_property_type.xml new file mode 100644 index 00000000000..f38dd784bcc --- /dev/null +++ b/estate/data/estate_property_type.xml @@ -0,0 +1,18 @@ + + + + Residential + + + + Commercial + + + + Industrial + + + + Land + + diff --git a/estate/demo/estate_property_demo.xml b/estate/demo/estate_property_demo.xml new file mode 100644 index 00000000000..c7248cc2d6a --- /dev/null +++ b/estate/demo/estate_property_demo.xml @@ -0,0 +1,50 @@ + + + + Big Villa + new + A nice and big villa + 12345 + 2020-02-02 + 1600000.0 + 6 + 100 + 4 + 1 + 1 + 100000 + south + + + + Trailer Home + cancelled + Home in trailer park. + 54321 + 1970-01-01 + 100000.00 + 120000.00 + 1 + 10 + 4 + 0 + + + diff --git a/estate/demo/estate_property_offer.xml b/estate/demo/estate_property_offer.xml new file mode 100644 index 00000000000..8d559dd9b1d --- /dev/null +++ b/estate/demo/estate_property_offer.xml @@ -0,0 +1,30 @@ + + + + 1000000000 + 14 + + + + + 1500000000000 + 14 + + + + + 15000000000001 + 11 + + + + + + + + + + + + + diff --git a/estate/models/__init__.py b/estate/models/__init__.py new file mode 100644 index 00000000000..f7e4cc6f3dd --- /dev/null +++ b/estate/models/__init__.py @@ -0,0 +1,7 @@ +# Part of Odoo. See LICENSE file for full copyright and licensing details. + +from . import estate_property +from . import estate_property_type +from . import estate_property_tag +from . import estate_property_offer +from . import res_users diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py new file mode 100644 index 00000000000..44a83ad21cc --- /dev/null +++ b/estate/models/estate_property.py @@ -0,0 +1,89 @@ +from datetime import date +from dateutil.relativedelta import relativedelta +from odoo import api, fields, models +from odoo.exceptions import UserError + + +class EstateProperty(models.Model): + _name = "estate.property" + _description = "Estate Property is defined" + _order = "id desc" + _sql_constraints = [ + ('check_expectep_price', 'CHECK(expected_price > 0 AND selling_price > 0)', + 'The Price must be positve.') + ] + + name = fields.Char(required=True) + description = fields.Text() + postcode = fields.Char() + date_avaiblity = fields.Date(copy=False, default=date.today() + relativedelta(months=3)) + expected_price = fields.Float(required=True) + 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() + property_type_id = fields.Many2one("estate.property.type", string="Property Type") + salesman_id = fields.Many2one('res.users', string='Salesman', index=True, default=lambda self: self.env.user) + buyer_id = fields.Many2one('res.partner', string='Buyer', index=True, default=lambda self: self.env.user.partner_id.id) + property_tag_ids = fields.Many2many("estate.property.tag") + offer_ids = fields.One2many("estate.property.offer", "property_id", string="Offers") + total_area = fields.Float(compute="_compute_total") + best_price = fields.Float(compute="_compute_best_price", string="Best Offer Price", readonly=True) + active = fields.Boolean(default=True) + company_id = fields.Many2one("res.company", required=True, default=lambda self: self.env.company) + + @api.depends("garden_area", "living_area") + def _compute_total(self): + for record in self: + record.total_area = record.garden_area + record.living_area + + @api.depends("offer_ids.price") + def _compute_best_price(self): + for record in self: + if record.offer_ids: + record.best_price = max(record.offer_ids.mapped('price')) + else: + record.best_price = 0.0 + + garden_orientation = fields.Selection( + string='Direction', + selection=[('north', 'North'), ('south', 'South'), ('east', 'East'), ('west', 'West')], + help="This is used to locate garden's direction" + ) + + state = fields.Selection( + selection=[('new', 'New'), ('offer received', 'Offer Received'), ('offer accepted', 'Offer Accepted'), ('sold', 'Sold'), ('cancelled', 'Cancelled')], + default='new', + required=True, + copy=False, + ) + + @api.onchange("garden") + def _onchange_garden(self): + if self.garden: + self.garden_area = 10 + self.garden_orientation = "north" + else: + self.garden_area = 0 + self.garden_orientation = False + + def action_cancel(self): + for record in self: + if record.state == 'sold': + raise UserError("Sold properties cannot be cancelled.") + record.state = 'cancelled' + + def action_sold(self): + for record in self: + if record.state == 'cancelled': + raise UserError("Cancelled properties cannot be marked as sold.") + record.state = 'sold' + + @api.ondelete(at_uninstall=False) + def _check_state_delete(self): + for record in self: + if record.state not in ['new', 'cancelled']: + raise UserError("Only properties in 'New' or 'Cancelled' state can be deleted.") diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py new file mode 100644 index 00000000000..1fb236dd533 --- /dev/null +++ b/estate/models/estate_property_offer.py @@ -0,0 +1,83 @@ +from datetime import timedelta +from odoo import api, fields, models +from odoo.exceptions import UserError, ValidationError +from odoo.tools import float_compare + + +class EstatePropertyOffer(models.Model): + _name = "estate.property.offer" + _description = "Offers related to property are made" + _order = "price desc" + _sql_constraints = [ + ('check_price', 'CHECK(price > 0)', 'The offer price must be greater than 0') + ] + + price = fields.Float() + partner_id = fields.Many2one( + "res.partner", string='Partner', index=True, default=lambda self: self.env.user.partner_id.id + ) + property_id = fields.Many2one("estate.property", index=True, required=True) + validity = fields.Integer(default=7) + date_deadline = fields.Date( + string="Deadline", + compute="_compute_date_deadline", + inverse="_inverse_date_deadline", + store=True + ) + property_type_id = fields.Many2one('estate.property.type', index=True) + + @api.depends('create_date', 'validity') + def _compute_date_deadline(self): + for record in self: + create_date = record.create_date or fields.Date.today() + record.date_deadline = create_date + timedelta(days=record.validity) + + def _inverse_date_deadline(self): + for record in self: + create_date = record.create_date or fields.Date.today() + if record.date_deadline: + record.validity = (record.date_deadline.day - create_date.day) + + status = fields.Selection( + selection=[('accepted', 'Accepted'), ('refused', 'Refuse')], + copy=False, + ) + + def action_confirm(self): + for record in self: + if any( + offer.status == 'accepted' + for offer in record.property_id.offer_ids + ): + raise UserError("Only one offer can be accepted per property.") + min_price = record.property_id.expected_price * 0.9 + if float_compare(record.price, min_price, precision_digits=2) < 0: + raise ValidationError("Offer must be at least 90% of the expected price to be accepted.") + record.status = 'accepted' + record.property_id.state = 'offer accepted' + other_offers = record.property_id.offer_ids - record + other_offers.write({'status': 'refused'}) + record.property_id.selling_price = record.price + record.property_id.buyer_id = record.partner_id + + def action_refuse(self): + for record in self: + record.status = 'refused' + + @api.model_create_multi + def create(self, vals_list): + for vals in vals_list: + property_id = vals.get("property_id") + price = vals.get("price", 0.0) + if property_id: + property_obj = self.env["estate.property"].browse(property_id) + best_price = property_obj.best_price or 0.0 + if price < best_price: + raise UserError( + "Offer price must be greater than or equal to the best offer price." + ) + records = super().create(vals_list) + for record in records: + if record.partner_id: + record.property_id.state = "offer received" + return records diff --git a/estate/models/estate_property_tag.py b/estate/models/estate_property_tag.py new file mode 100644 index 00000000000..f555fd230dc --- /dev/null +++ b/estate/models/estate_property_tag.py @@ -0,0 +1,13 @@ +from odoo import fields, models + + +class EstatePropertyTag(models.Model): + _name = "estate.property.tag" + _description = "Tags of estate are defined" + _order = "name asc" + _sql_constraints = [ + ('check_unique_name', 'UNIQUE(name)', 'The name of the tag should be unique') + ] + + name = fields.Char(required=True) + color = fields.Integer() diff --git a/estate/models/estate_property_type.py b/estate/models/estate_property_type.py new file mode 100644 index 00000000000..d19bb8d10da --- /dev/null +++ b/estate/models/estate_property_type.py @@ -0,0 +1,21 @@ +from odoo import api, fields, models + + +class EstatePropertyType(models.Model): + _name = "estate.property.type" + _description = "Type of estate is defined" + _order = "name asc" + _sql_constraints = [ + ('check_unique_property_type', 'UNIQUE(name)', 'The Property type should be unique') + ] + + name = fields.Char(required=True) + property_ids = fields.One2many('estate.property', 'property_type_id') + sequence = fields.Integer() + offer_ids = fields.One2many('estate.property.offer', 'property_type_id') + offer_count = fields.Integer(string="Offer Count", compute="_compute_offer_count") + + @api.depends('offer_ids') + def _compute_offer_count(self): + for record in self: + record.offer_count = sum(len(property.offer_ids) for property in record.property_ids) diff --git a/estate/models/res_users.py b/estate/models/res_users.py new file mode 100644 index 00000000000..aedc23284b0 --- /dev/null +++ b/estate/models/res_users.py @@ -0,0 +1,7 @@ +from odoo import fields, models + + +class ResUsers(models.Model): + _inherit = 'res.users' + + property_ids = fields.One2many('estate.property', 'salesman_id', domain=[('state', 'not in', ['sold', 'cancelled'])]) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv new file mode 100644 index 00000000000..e2fe519edd1 --- /dev/null +++ b/estate/security/ir.model.access.csv @@ -0,0 +1,10 @@ +id,name,model_id/id,group_id/id,perm_read,perm_write,perm_create,perm_unlink +access_property_manager,access_estate_property_manager,model_estate_property,estate_group_manager,1,1,1,0 +access_property_type_manager,access_estate_property_type_manager,model_estate_property_type,estate_group_manager,1,1,1,1 +access_property_tag_manager,access_estate_property_tag_manager,model_estate_property_tag,estate_group_manager,1,1,1,1 +access_property_offer_manager,access_estate_property_offer_manager,model_estate_property_offer,estate_group_manager,1,1,1,1 +access_res_users_manager,access_res_users_manager,model_res_users,estate_group_manager,1,1,1,1 +access_property_agent,access_estate_property_agent,model_estate_property,estate_group_user,1,1,1,0 +access_property_offer_agent,access_estate_property_offer_agent,model_estate_property_offer,estate_group_user,1,1,1,0 +access_property_type_agent,access_estate_property_type_agent,model_estate_property_type,estate_group_user,1,0,0,0 +access_property_tag_agent,access_estate_property_tag_agent,model_estate_property_tag,estate_group_user,1,0,0,0 diff --git a/estate/security/security.xml b/estate/security/security.xml new file mode 100644 index 00000000000..d89042277e9 --- /dev/null +++ b/estate/security/security.xml @@ -0,0 +1,26 @@ + + + + Agent + Real estate agents can manage the properties under their care, or properties which are not specifically under the care of any agent. + + + + + Manager + Real estate managers can configure the system (manage available types and tags) as well as oversee every property in the pipeline. + + + + + + company: see or modify properties of my company only + + + + + + + [('company_id', '=', user.company_id.id)] + + diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml new file mode 100644 index 00000000000..06fe08a9720 --- /dev/null +++ b/estate/views/estate_menus.xml @@ -0,0 +1,13 @@ + + + + + + + + + + + + + diff --git a/estate/views/estate_property_tag_views.xml b/estate/views/estate_property_tag_views.xml new file mode 100644 index 00000000000..78e330d4b45 --- /dev/null +++ b/estate/views/estate_property_tag_views.xml @@ -0,0 +1,28 @@ + + + + Property Tags + estate.property.tag + list,form + + + + estate.property.tag.list + estate.property.tag + + + + + + + + + estate.property.tag.search + 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..fda463384c9 --- /dev/null +++ b/estate/views/estate_property_type_views.xml @@ -0,0 +1,85 @@ + + + + Property Types + estate.property.type + list,form + + + + Offers + estate.property.offer + list,form + {} + [('property_id.property_type_id', '=', active_id)] + + + + estate.property.type + estate.property.type + + + + + + + + + + estate.property.type.form + estate.property.type + +
+ +
+ +
+

+ +

+ + + + + + + + + + + + +
+
+
+
+ + + estate.property.type.search + estate.property.type + + + + + + + + + estate.property.offer.tree + estate.property.offer + + + + + + + + + + +
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml new file mode 100644 index 00000000000..c5aebc0a937 --- /dev/null +++ b/estate/views/estate_property_views.xml @@ -0,0 +1,167 @@ + + + + Properties + estate.property + list,form,kanban + {'search_default_state': 1} + + + + estate.property.list + estate.property + + + + + + + + + + + + + + + + + estate.property.kanban + estate.property + + + + + + +
+
+ +
+
+ +
+
+ Expected Price: +
+
+ Best Price: +
+
+ Selling Price: +
+
+
+
+
+
+
+ + + estate.property.form + estate.property + +
+
+
+ +

+ +

+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +