diff --git a/.gitignore b/.gitignore
index b6e47617de1..810aac7ee2e 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,3 +1,7 @@
+# custom ignore
+custom_module/
+
+
# Byte-compiled / optimized / DLL files
__pycache__/
*.py[cod]
diff --git a/estate/__init__.py b/estate/__init__.py
new file mode 100644
index 00000000000..0650744f6bc
--- /dev/null
+++ b/estate/__init__.py
@@ -0,0 +1 @@
+from . import models
diff --git a/estate/__manifest__.py b/estate/__manifest__.py
new file mode 100644
index 00000000000..820963ffaa8
--- /dev/null
+++ b/estate/__manifest__.py
@@ -0,0 +1,18 @@
+{
+ "name": "Estate",
+ "application": True,
+ "description": "Specific Real Estate Module",
+ "depends": ["base"],
+ "data": [
+ "security/res_groups.xml",
+ "security/ir.model.access.csv",
+ "views/estate_property_views.xml",
+ "views/estate_property_tag_views.xml",
+ "views/estate_property_type_views.xml",
+ "views/estate_property_offer_views.xml",
+ "views/estate_menus.xml",
+ ],
+ "installable": True,
+ "author": "Maram-Odoo",
+ "license": "LGPL-3",
+}
diff --git a/estate/models/__init__.py b/estate/models/__init__.py
new file mode 100644
index 00000000000..4b57c1674fc
--- /dev/null
+++ b/estate/models/__init__.py
@@ -0,0 +1,4 @@
+from . import estate_property
+from . import estate_property_tag
+from . import estate_property_type
+from . import estate_property_offer
diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py
new file mode 100644
index 00000000000..6e8e1a3d032
--- /dev/null
+++ b/estate/models/estate_property.py
@@ -0,0 +1,106 @@
+from dateutil.relativedelta import relativedelta
+from odoo import fields, models, api
+from odoo.exceptions import UserError, ValidationError
+
+
+class EstateProperty(models.Model):
+ _name = "estate.property"
+ _description = "Estate Property Management Module"
+
+ name = fields.Char(string="Title", required=True)
+ description = fields.Text(string="Description")
+ postcode = fields.Char(string="Post Code")
+ date_availability = fields.Date(
+ string="Availability From",
+ default=lambda self: 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)
+ bedrooms = fields.Integer(string="Bed Rooms", 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")
+ garden_orientation = fields.Selection(
+ string="Garden Orientation",
+ selection=[
+ ("north", "North"),
+ ("south", "South"),
+ ("east", "East"),
+ ("west", "West"),
+ ],
+ )
+ state = fields.Selection(
+ string="Status",
+ default="new",
+ copy=False,
+ required=True,
+ selection=[
+ ("new", "New"),
+ ("offer_received", "Offer Received"),
+ ("accepted", "Accepted"),
+ ("sold", "Sold"),
+ ("canceled", "Cancelled"),
+ ],
+ )
+ active = fields.Boolean(string="Active", default=True)
+ tag_ids = fields.Many2many("estate.property.tag", string="Property Tag")
+ property_type_id = fields.Many2one("estate.property.type", string="Property Type")
+ salesman_id = fields.Many2one(
+ "res.users", string="Salesman", default=lambda self: self.env.user
+ )
+ buyer_id = fields.Many2one("res.partner", string="Buyer", copy=False)
+ offer_ids = fields.One2many("estate.property.offer", "property_id", string="Offers")
+ total_area = fields.Float(compute="_compute_total_area")
+ best_offer = fields.Float(compute="_compute_best_offer")
+
+ @api.depends("living_area", "garden_area")
+ def _compute_total_area(self):
+ for record in self:
+ record.total_area = record.living_area + record.garden_area
+
+ @api.depends("offer_ids.price")
+ def _compute_best_offer(self):
+ for record in self:
+ if record.offer_ids:
+ prices = record.offer_ids.mapped("price")
+ record.best_offer = max(prices)
+ else:
+ record.best_offer = 0.0
+
+ @api.onchange("garden")
+ def onchange_garden(self):
+ if self.garden:
+ self.garden_area = 1000
+ self.garden_orientation = "north"
+ else:
+ self.garden_area = 0
+ self.garden_orientation = False
+
+ def action_sold(self):
+ for record in self:
+ if record.state != "canceled":
+ record.state = "sold"
+ else:
+ raise UserError(
+ "You cannot move to canceled stage after sold the property"
+ )
+ return True
+
+ def action_cencel(self):
+ for record in self:
+ if record.state != "sold":
+ record.state = "canceled"
+ else:
+ raise UserError(
+ "You cannot move to sold stage after cenceled the property"
+ )
+ return True
+
+ @api.constrains("expected_price")
+ def _check_positive(self):
+ for record in self:
+ if record.expected_price <= 0:
+ raise ValidationError("Expected Price Must be in Positive")
diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py
new file mode 100644
index 00000000000..da4b9cddcf5
--- /dev/null
+++ b/estate/models/estate_property_offer.py
@@ -0,0 +1,70 @@
+from odoo import fields, models, api
+from odoo.exceptions import UserError
+
+
+class EstatePropertyOffer(models.Model):
+ _name = "estate.property.offer"
+ _description = "Estate Property Offer"
+
+ price = fields.Float(required=True, string="Offer Price")
+ partner_id = fields.Many2one("res.partner", required=True, string="Partner")
+ property_id = fields.Many2one("estate.property", required=True, string="Property")
+ status = fields.Selection(
+ string="Status",
+ copy=False,
+ selection=[
+ ("accepted", "Accepted"),
+ ("refuse", "Refused"),
+ ],
+ )
+ date_deadline = fields.Date(
+ string="Deadline Date",
+ compute="_compute_date_deadline",
+ inverse="_inverse_date_deadline",
+ store=True,
+ )
+ validity = fields.Integer(
+ string="Validity",
+ default=7,
+ store=True,
+ )
+
+ # alternative : create_date = record.create_date.date()
+ @api.depends("validity")
+ def _compute_date_deadline(self):
+ for record in self:
+ if record.validity:
+ create_date = (
+ fields.Date.to_date(record.create_date) or fields.Date.today()
+ )
+ record.date_deadline = fields.Date.add(
+ create_date, days=record.validity
+ )
+
+ def _inverse_date_deadline(self):
+ for record in self:
+ if record.date_deadline:
+ create_date = (
+ fields.Date.to_date(record.create_date) or fields.Date.today()
+ )
+ record.validity = (record.date_deadline - create_date).days
+
+ @api.onchange("date_deadline")
+ def _onchange_validity(self):
+ if self.date_deadline:
+ create_date = fields.Date.to_date(self.create_date) or fields.Date.today()
+ self.validity = (self.date_deadline - create_date).days
+
+ def action_accept_offer(self):
+ for record in self:
+ if not (record.property_id.selling_price):
+ record.property_id.selling_price = record.price
+ record.status = "accepted"
+ record.property_id.buyer_id = record.partner_id
+ else:
+ raise UserError("You cannot accept multiple offer")
+
+ def action_refuse_offer(self):
+ for record in self:
+ record.status = "refuse"
+ return True
diff --git a/estate/models/estate_property_tag.py b/estate/models/estate_property_tag.py
new file mode 100644
index 00000000000..0ef4309e229
--- /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(string="Property Tags", required=True)
diff --git a/estate/models/estate_property_type.py b/estate/models/estate_property_type.py
new file mode 100644
index 00000000000..eb88993ff70
--- /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 Types"
+
+ name = fields.Char(string="Property Types", required=True)
diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv
new file mode 100644
index 00000000000..b9786ca05f0
--- /dev/null
+++ b/estate/security/ir.model.access.csv
@@ -0,0 +1,7 @@
+id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
+access_estate_property,estate.property,model_estate_property,base.group_user,1,1,1,1
+access_estate_property_tag,estate.property.tag,model_estate_property_tag,base.group_user,1,1,1,1
+access_estate_property_type,estate.property.type,model_estate_property_type,base.group_user,1,1,1,1
+access_estate_property_offer,estate.property.offer,model_estate_property_offer,base.group_user,1,1,1,1
+access_estate_property_manager,estate.property manager,model_estate_property,group_estate_manager,1,1,1,1
+access_estate_property_user,estate.property user,model_estate_property,group_estate_user,1,0,1,0
diff --git a/estate/security/res_groups.xml b/estate/security/res_groups.xml
new file mode 100644
index 00000000000..4d88e409f25
--- /dev/null
+++ b/estate/security/res_groups.xml
@@ -0,0 +1,12 @@
+
+
+
+
+ Estate Manager
+
+
+
+ Estate User
+
+
+
diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml
new file mode 100644
index 00000000000..b16d06afcaa
--- /dev/null
+++ b/estate/views/estate_menus.xml
@@ -0,0 +1,31 @@
+
+
+
+
+
+
diff --git a/estate/views/estate_property_offer_views.xml b/estate/views/estate_property_offer_views.xml
new file mode 100644
index 00000000000..5cc50f97753
--- /dev/null
+++ b/estate/views/estate_property_offer_views.xml
@@ -0,0 +1,22 @@
+
+
+
+
+ estate.property.offer.view.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..be8d175a9fa
--- /dev/null
+++ b/estate/views/estate_property_tag_views.xml
@@ -0,0 +1,34 @@
+
+
+
+
+ estate.property.tag.view.list
+ estate.property.tag
+
+
+
+
+
+
+
+
+ estate.property.tag.view.form
+ estate.property.tag
+
+
+
+
+
+
+ Estate 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..c14bb8422df
--- /dev/null
+++ b/estate/views/estate_property_type_views.xml
@@ -0,0 +1,34 @@
+
+
+
+
+ estate.property.type.view.list
+ estate.property.type
+
+
+
+
+
+
+
+
+ estate.property.type.view.form
+ estate.property.type
+
+
+
+
+
+
+ Estate Property Type
+ estate.property.type
+ list,form
+
+
+
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml
new file mode 100644
index 00000000000..52dfedfdba8
--- /dev/null
+++ b/estate/views/estate_property_views.xml
@@ -0,0 +1,120 @@
+
+
+
+
+ estate.property.view.list
+ estate.property
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ estate.property.view.form
+ estate.property
+
+
+
+
+
+
+ estate.property.search
+ estate.property
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Estate Property
+ estate.property
+ list,form
+ {'search_default_available':1}
+
+
+
+
+