diff --git a/estate/__init__.py b/estate/__init__.py
new file mode 100644
index 00000000000..a9e3372262c
--- /dev/null
+++ b/estate/__init__.py
@@ -0,0 +1,2 @@
+
+from . import models
diff --git a/estate/__manifest__.py b/estate/__manifest__.py
new file mode 100644
index 00000000000..e5838f68794
--- /dev/null
+++ b/estate/__manifest__.py
@@ -0,0 +1,17 @@
+{
+ 'name': 'Estate',
+ 'category': 'Sales',
+ 'sequence': 1,
+ 'summary': 'Sell and bid on the hottest real estate properties.',
+ 'website': 'https://www.odoo.com/app/estate',
+ 'depends': [
+ 'base_setup',
+ 'web',
+ ],
+ 'data': [
+ 'security/ir.model.access.csv',
+ 'views/estate_property_views.xml',
+ 'views/estate_menus.xml',
+ ],
+ 'application': True
+}
diff --git a/estate/models/__init__.py b/estate/models/__init__.py
new file mode 100644
index 00000000000..a544207d8f5
--- /dev/null
+++ b/estate/models/__init__.py
@@ -0,0 +1,3 @@
+
+from . import estate_property
+from . import estate_property_type
diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py
new file mode 100644
index 00000000000..d2b48001de5
--- /dev/null
+++ b/estate/models/estate_property.py
@@ -0,0 +1,49 @@
+from odoo import fields, models
+from dateutil.relativedelta import relativedelta
+
+
+class EstateProperty(models.Model):
+ _name = 'estate.property'
+ _description = "Real Estate Property"
+
+ name = fields.Char("Title", required=True, translate=True)
+ property_type_id = fields.Many2one(
+ 'estate.property.type',
+ string="Property Type",
+ )
+ postcode = fields.Char("Postcode", required=True)
+ availability = fields.Date(
+ "Available From",
+ required=True,
+ copy=False,
+ default=lambda self: fields.Date.today() + relativedelta(months=3),
+ )
+ description = fields.Text("Description")
+ bedrooms = fields.Integer("Bedrooms", required=True, default=2)
+ living_area = fields.Integer("Living Area (sqm)", required=True)
+ currency_id = fields.Many2one('res.currency', string="Currency", default=lambda self: self.env.company.currency_id.id)
+ expected_price = fields.Monetary("Expected Price", required=True)
+ selling_price = fields.Monetary("Selling Price", readonly=True, copy=False)
+ # best_offer_id = fields.Many2one('estate.property.offer', string='Best Offer', readonly=True)
+ facades = fields.Integer("Facades", default=False)
+ garage = fields.Boolean("Garage", default=False)
+ garden = fields.Boolean("Garden", default=False)
+ garden_area = fields.Integer("Garden Area (sqm)", required=False)
+ garden_orientation = fields.Selection(selection=[('north', "North"), ('south', "South"), ('east', "East"), ('west', "West")], string="Garden Orientation")
+ total_area = fields.Integer("Total Area (sqm)")
+
+ state = fields.Selection(
+ string="Status",
+ selection=[('new', "New"), ('offer_received', "Offer Received"), ('offer_accepted', "Offer Accepted"), ('sold', "Sold"), ('canceled', "Canceled")],
+ required=True,
+ copy=False,
+ default='new',
+ )
+ active = fields.Boolean("Active", default=True)
+ buyer_id = fields.Many2one('res.partner', string="Buyer", copy=False)
+ seller_id = fields.Many2one('res.users', string="Salesperson", default=lambda self: self.env.user)
+
+ # _check_expected_price = models.Constraint(
+ # 'CHECK(expected_price) >= 0',
+ # "The expected price can't be negative",
+ # )
diff --git a/estate/models/estate_property_type.py b/estate/models/estate_property_type.py
new file mode 100644
index 00000000000..77bc4eb6ad0
--- /dev/null
+++ b/estate/models/estate_property_type.py
@@ -0,0 +1,11 @@
+from odoo import fields, models
+
+
+class EstateProperty(models.Model):
+ _name = 'estate.property.type'
+ _description = "Real Estate Property Type"
+
+ name = fields.Char(
+ "Name",
+ required=True,
+ )
diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv
new file mode 100644
index 00000000000..622378709a1
--- /dev/null
+++ b/estate/security/ir.model.access.csv
@@ -0,0 +1,3 @@
+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_type,estate.property.type,model_estate_property_type,base.group_user,1,1,1,1
diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml
new file mode 100644
index 00000000000..aa1c7bda34e
--- /dev/null
+++ b/estate/views/estate_menus.xml
@@ -0,0 +1,11 @@
+
+
+
+
diff --git a/estate/views/estate_property_views.xml b/estate/views/estate_property_views.xml
new file mode 100644
index 00000000000..6244436f750
--- /dev/null
+++ b/estate/views/estate_property_views.xml
@@ -0,0 +1,98 @@
+
+
+
+
+ estate.property.list.view
+ estate.property
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ estate.property.form.view
+ estate.property
+
+
+
+
+
+
+ Property
+ estate.property
+ list,form
+
+
+
+ estate.property.search
+ estate.property
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Property Types
+ estate.property.type
+ list
+
+
diff --git a/ruff.toml b/ruff.toml
new file mode 100644
index 00000000000..e84deba95cb
--- /dev/null
+++ b/ruff.toml
@@ -0,0 +1,44 @@
+# Exclude a variety of commonly ignored directories.
+exclude = [
+ ".bzr",
+ ".direnv",
+ ".eggs",
+ ".git",
+ ".git-rewrite",
+ ".hg",
+ ".ipynb_checkpoints",
+ ".mypy_cache",
+ ".nox",
+ ".pants.d",
+ ".pyenv",
+ ".pytest_cache",
+ ".pytype",
+ ".ruff_cache",
+ ".svn",
+ ".tox",
+ ".venv",
+ ".vscode",
+ "__pypackages__",
+ "_build",
+ "buck-out",
+ "build",
+ "dist",
+ "node_modules",
+ "site-packages",
+ "venv",
+]
+
+# Assume Python 3.12
+target-version = "py312"
+
+line-length = 255
+[lint]
+# Enable Pyflakes (`F`) and a subset of the pycodestyle (`E`) codes by default.
+select = ["ALL"]
+ignore = ["A","ARG","ANN","B","C901","D","DTZ","DOC","E501","ERA001","FBT","N","PD","PERF","PIE790","PLR","PT","Q","RSE102","RUF001","RUF012","S","SIM102","SIM108","SLF001","TID252","UP031","TRY003","TRY300","E713","SIM117","PGH003","RUF005","FIX","TD","TRY400","C408","PLW2901","PTH","EM102","INP001","CPY001","E266","PIE808","PLC2701","RUF100","FA100","FURB","C420","COM812","TRY002","B904","EM101","I001","UP006","UP007","RET","RUF021","E741","FAST","ASYNC","AIR","DJ","NPY","FA102","F401"]
+
+# Allow fix for all enabled rules (when `--fix`) is provided.
+fixable = ["ALL"]
+unfixable = ["A","ARG","ANN","B","C901","D","DTZ","DOC","E501","ERA001","FBT","N","PD","PERF","PIE790","PLR","PT","Q","RSE102","RUF001","RUF012","S","SIM102","SIM108","SLF001","TID252","UP031","TRY003","TRY300","E713","SIM117","PGH003","RUF005","FIX","TD","TRY400","C408","PLW2901","PTH","EM102","INP001","CPY001","E266","PIE808","PLC2701","RUF100","FA100","FURB","C420","COM812","TRY002","B904","EM101","I001","UP006","UP007","RET","RUF021","E741","FAST","ASYNC","AIR","DJ","NPY","FA102","F401"]
+[format]
+quote-style = "preserve"