diff --git a/awesome_owl/static/src/Card/card.js b/awesome_owl/static/src/Card/card.js
new file mode 100644
index 00000000000..f934fbf140f
--- /dev/null
+++ b/awesome_owl/static/src/Card/card.js
@@ -0,0 +1,9 @@
+import { Component } from "@odoo/owl";
+
+export class Card extends Component {
+ static template = "awesome_owl.card";
+ static props = {
+ title: String,
+ content: String,
+ };
+}
diff --git a/awesome_owl/static/src/Card/card.xml b/awesome_owl/static/src/Card/card.xml
new file mode 100644
index 00000000000..3895a39ab8b
--- /dev/null
+++ b/awesome_owl/static/src/Card/card.xml
@@ -0,0 +1,15 @@
+
+
+
+
+
+
diff --git a/awesome_owl/static/src/Counter/counter.js b/awesome_owl/static/src/Counter/counter.js
new file mode 100644
index 00000000000..12be97c5229
--- /dev/null
+++ b/awesome_owl/static/src/Counter/counter.js
@@ -0,0 +1,13 @@
+import { Component, useState } from "@odoo/owl";
+
+export class Counter extends Component {
+ static template = "awesome_owl.counter";
+ static props = {};
+ setup() {
+ this.state = useState({ value: 0 });
+ }
+
+ increment() {
+ this.state.value++;
+ }
+}
diff --git a/awesome_owl/static/src/Counter/counter.xml b/awesome_owl/static/src/Counter/counter.xml
new file mode 100644
index 00000000000..748521c23b2
--- /dev/null
+++ b/awesome_owl/static/src/Counter/counter.xml
@@ -0,0 +1,11 @@
+
+
+
+
+ hello world
+
Counter:
+
+
+
+
+
diff --git a/awesome_owl/static/src/playground.js b/awesome_owl/static/src/playground.js
index 4ac769b0aa5..9d8b6a2e951 100644
--- a/awesome_owl/static/src/playground.js
+++ b/awesome_owl/static/src/playground.js
@@ -1,5 +1,9 @@
-import { Component } from "@odoo/owl";
+import { Component, markup } from "@odoo/owl";
+import { Counter } from "./Counter/counter";
+import { Card } from "./Card/card";
export class Playground extends Component {
static template = "awesome_owl.playground";
+ static components = {Counter, Card}
+ value1 = markup("
This is the first card content
");
}
diff --git a/awesome_owl/static/src/playground.xml b/awesome_owl/static/src/playground.xml
index 4fb905d59f9..fc193fa23b3 100644
--- a/awesome_owl/static/src/playground.xml
+++ b/awesome_owl/static/src/playground.xml
@@ -1,10 +1,9 @@
-
+
-
-
- hello world
-
+
+
+
+
-
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..daece280d85
--- /dev/null
+++ b/estate/__manifest__.py
@@ -0,0 +1,18 @@
+{
+ 'name': "estate",
+ 'author': "pkhu",
+ 'license': "LGPL-3",
+ 'depends': ['base', 'mail'],
+ 'application': True,
+ 'data': [
+ 'security/ir.model.access.csv',
+ 'views/estate_property_views.xml',
+ 'views/estate_property_maintenance_view.xml',
+ 'views/estate_property_offer_views.xml',
+ 'views/estate_property_type_views.xml',
+ 'views/estate_property_tag_views.xml',
+ 'views/res_users_views.xml',
+ 'views/estate_investor_views.xml',
+ 'views/estate_menus.xml',
+ ],
+}
diff --git a/estate/models/__init__.py b/estate/models/__init__.py
new file mode 100644
index 00000000000..8e1e543760c
--- /dev/null
+++ b/estate/models/__init__.py
@@ -0,0 +1,7 @@
+from . import estate_property
+from . import estate_property_offer
+from . import estate_property_tag
+from . import etsate_property_type
+from . import estate_property_maintenance
+from . import res_users
+from . import estate_investor
diff --git a/estate/models/estate_investor.py b/estate/models/estate_investor.py
new file mode 100644
index 00000000000..08e786dfd6e
--- /dev/null
+++ b/estate/models/estate_investor.py
@@ -0,0 +1,9 @@
+from odoo import fields, models
+
+
+class EstateInvestor(models.Model):
+ _name = "estate.investor"
+ _description = "investor details"
+ _rec_name = 'id'
+
+ name = fields.Many2one('res.partner')
diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py
new file mode 100644
index 00000000000..ed792639cf1
--- /dev/null
+++ b/estate/models/estate_property.py
@@ -0,0 +1,130 @@
+from datetime import timedelta
+
+from odoo import _, api, fields, models
+from odoo.exceptions import UserError, ValidationError
+from odoo.tools.float_utils import float_compare, float_is_zero
+
+
+class EstateProperty(models.Model):
+ _name = 'estate.property'
+ _description = 'estate property details'
+ _order = 'id desc'
+ _inherit = 'mail.thread'
+
+ name = fields.Char(required=True)
+ description = fields.Text()
+ postcode = fields.Char()
+ date_availability = fields.Date(
+ copy=False, default=lambda self: fields.Date.today() + timedelta(days=90)
+ )
+ expected_price = fields.Float(required=True)
+ selling_price = fields.Float(readonly=True, copy=False)
+ bedrooms = fields.Integer(default=2)
+ living_area = fields.Integer(string="Living Area(sqm)")
+ facades = fields.Integer()
+ garage = fields.Boolean()
+ garden = fields.Boolean()
+ garden_area = fields.Integer(string="Garden Area (sqm)")
+ garden_orientation = fields.Selection(
+ selection=[
+ ('north', "North"),
+ ('west', "West"),
+ ('east', "East"),
+ ('south', "South"),
+ ]
+ )
+ active = fields.Boolean(default=True)
+ state = fields.Selection(
+ selection=[
+ ('new', "New"),
+ ('offer_received', "Offer Received"),
+ ('offer_accepted', "Offer Accepted"),
+ ('sold', "Sold"),
+ ('cancelled', "Cancelled"),
+ ],
+ default='new',
+ copy=False,
+ required=True,
+ )
+ property_type_id = fields.Many2one(
+ 'estate.property.type', string="Property Type")
+ user_id = fields.Many2one(
+ 'res.users', string="Salesperson", default=lambda self: self.env.user)
+ partner_id = fields.Many2one('res.partner', string="Buyer", readonly=True)
+ tag_ids = fields.Many2many('estate.property.tag', string="Tags")
+ offer_ids = fields.One2many(
+ 'estate.property.offer', 'property_id')
+ total_area = fields.Integer(
+ compute='_compute_total_area', string="Total Area(sqm)")
+ best_price = fields.Float(compute='_compute_best_price', store=True)
+ property_maintainance_ids = fields.One2many(
+ 'estate.property.maintenance', 'property_id')
+ total_maintenance_cost = fields.Float(
+ compute='_compute_total_maintenance_cost')
+ investor = fields.Many2one('estate.investor')
+ _check_expected_price = models.Constraint(
+ 'CHECK(expected_price > 0)',
+ "The Expected price cannot be negative or zero."
+ )
+ _check_selling_price = models.Constraint(
+ 'CHECK(selling_price > 0)',
+ "The Selling price cannot be negative."
+ )
+
+ @api.depends('garden_area', 'living_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_price(self):
+ best_price = dict(self.env['estate.property.offer']._read_group(domain=[
+ ('property_id', 'in', self.ids)], aggregates=['price:max'], groupby=['property_id']))
+ for record in self:
+ record.best_price = best_price.get(record, 0.0)
+
+ @api.onchange('garden')
+ def _onchange_garden(self):
+ if self.garden:
+ self.garden_area = 10
+ self.garden_orientation = 'north'
+ else:
+ self.garden_area = None
+ self.garden_orientation = None
+
+ @api.constrains('selling_price', 'expected_price')
+ def _constraint_selling_price(self):
+ if float_is_zero(self.selling_price, precision_rounding=0.01):
+ return
+ elif float_compare(self.selling_price, self.expected_price * 0.9, precision_rounding=0.01) < 0:
+ raise ValidationError(_(
+ "Selling price cannot be lower than 90% of the expected price."))
+
+ def action_property_sold(self):
+ if self.state != 'offer_accepted':
+ raise UserError(_("Atleast one offer should be accepted."))
+ for record in self.property_maintainance_ids:
+ if record.status != 'done':
+ raise UserError(_("Maintenance Request are still pending."))
+ self.state = 'sold'
+
+ def action_property_cancel(self):
+ self.state = 'cancelled'
+
+ def action_accept_best_offer(self):
+ data = self.env['estate.property.offer'].search(
+ domain=[('property_id', 'in', self.ids), ('price', '=', self.best_price)], limit=1)
+ data.action_accepted()
+
+ @api.depends('property_maintainance_ids.cost')
+ def _compute_total_maintenance_cost(self):
+ maintenace_cost = dict(self.env['estate.property.maintenance']._read_group(domain=[(
+ 'property_id', 'in', self.ids)], aggregates=['cost:sum'], groupby=['property_id']))
+ for record in self:
+ record.total_maintenance_cost = maintenace_cost.get(record, 0.0)
+
+ @api.ondelete(at_uninstall=False)
+ def _unlink_property(self):
+ if self.state not in ['new', 'cancelled']:
+ raise UserError(
+ _("Only new and cancelled property can be deleted."))
diff --git a/estate/models/estate_property_maintenance.py b/estate/models/estate_property_maintenance.py
new file mode 100644
index 00000000000..1e0705d2aaf
--- /dev/null
+++ b/estate/models/estate_property_maintenance.py
@@ -0,0 +1,20 @@
+from odoo import _, api, fields, models
+from odoo.exceptions import UserError
+from odoo.tools.float_utils import float_is_zero
+
+
+class PropertyMantainance(models.Model):
+ _name = 'estate.property.maintenance'
+ _description = 'show propety maintenance request'
+
+ name = fields.Char(string="Title", required=True)
+ cost = fields.Float()
+ status = fields.Selection(selection=[(
+ 'new', "New"), ('approved', "Approved"), ('done', "Done")], default='new')
+ property_id = fields.Many2one('estate.property')
+
+ @api.onchange('status')
+ def _onchange_status(self):
+ for record in self:
+ if record.status == 'approved' and float_is_zero(record.cost, precision_rounding=0.01):
+ raise UserError(_("Cost must be greater than zero."))
diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py
new file mode 100644
index 00000000000..3ccb54b2182
--- /dev/null
+++ b/estate/models/estate_property_offer.py
@@ -0,0 +1,88 @@
+from datetime import timedelta
+
+from odoo import _, api, fields, models
+from odoo.exceptions import UserError, ValidationError
+from odoo.tools.float_utils import float_compare
+
+
+class PropertyOffer(models.Model):
+ _name = 'estate.property.offer'
+ _description = 'Property offer for each property.'
+ _order = 'price desc'
+
+ price = fields.Float()
+ status = fields.Selection(
+ selection=[('accepted', "Accepted"), ('refused', "Refused")], copy=False
+ )
+ 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)
+ date_deadline = fields.Date(
+ compute='_compute_date_deadline', inverse='_inverse_date_deadline'
+ )
+ property_type_id = fields.Many2one(
+ 'estate.property.type',
+ related='property_id.property_type_id',
+ store=True
+ )
+
+ _check_price = models.Constraint(
+ 'CHECK(price >= 0)',
+ "The Offer price cannot be negative."
+ )
+
+ @api.depends('validity')
+ def _compute_date_deadline(self):
+ for record in self:
+ record.date_deadline = (record.create_date or fields.Date.today()) + \
+ timedelta(days=record.validity)
+
+ def _inverse_date_deadline(self):
+ for record in self:
+ record.validity = (
+ record.date_deadline -
+ (record.create_date.date() or fields.Date.today())
+ ).days
+
+ def action_accepted(self):
+ offers = self.property_id.offer_ids.filtered(
+ lambda a: a.id != self.id)
+ for offer in offers:
+ offer.status = 'refused'
+ self.status = 'accepted'
+ self.property_id.partner_id = self.partner_id
+ self.property_id.selling_price = self.price
+ self.property_id.state = 'offer_accepted'
+
+ def action_refused(self):
+ if self.status == 'accepted':
+ self.property_id.partner_id = None
+ self.property_id.selling_price = None
+ self.property_id.state = 'offer_received'
+ self.status = 'refused'
+
+ @api.ondelete(at_uninstall=False)
+ def _ondelete_offer(self):
+ for records in self:
+ if records.status == 'accepted':
+ raise ValidationError(_("Accepted offer cannot be deleted."))
+
+ @api.model
+ def create(self, vals):
+ for val in vals:
+ price = val.get('price')
+ property_id = val.get('property_id')
+ property = self.env['estate.property'].browse(property_id)
+ if property.state == 'new':
+ property.best_price = price
+ elif float_compare(price, property.best_price, precision_rounding=0.01) < 0:
+ raise UserError(
+ _("Price should be greater than %s", property.best_price))
+ else:
+ property.best_price = price
+ if property and property.state == 'new':
+ property.state = 'offer_received'
+
+ return super().create(vals)
diff --git a/estate/models/estate_property_tag.py b/estate/models/estate_property_tag.py
new file mode 100644
index 00000000000..53aad86f51d
--- /dev/null
+++ b/estate/models/estate_property_tag.py
@@ -0,0 +1,15 @@
+from odoo import fields, models
+
+
+class PropertyTag(models.Model):
+ _name = 'estate.property.tag'
+ _description = 'Property Tags to describe property such new, renovated...'
+ _order = 'name'
+
+ name = fields.Char(required=True)
+ color = fields.Integer()
+
+ _unique_name = models.Constraint(
+ 'UNIQUE(name)',
+ "Property Tag must be unique."
+ )
diff --git a/estate/models/etsate_property_type.py b/estate/models/etsate_property_type.py
new file mode 100644
index 00000000000..ef984d2d1fb
--- /dev/null
+++ b/estate/models/etsate_property_type.py
@@ -0,0 +1,23 @@
+from odoo import api, fields, models
+
+
+class PropertyType(models.Model):
+ _name = 'estate.property.type'
+ _description = 'Define Type of property (House, Apartment, Penthouse, Castleā¦)'
+ _order = 'sequence, name desc'
+
+ name = fields.Char(required=True)
+ property_ids = fields.One2many('estate.property', 'property_type_id')
+ sequence = fields.Integer(default=10)
+ offer_ids = fields.One2many('estate.property.offer', 'property_type_id')
+ offer_count = fields.Integer(compute='_compute_offer_count')
+
+ _unique_name = models.Constraint(
+ 'UNIQUE(name)',
+ "Property Type must be unique."
+ )
+
+ @api.depends('offer_ids')
+ def _compute_offer_count(self):
+ for record in self:
+ record.offer_count = len(record.offer_ids)
diff --git a/estate/models/res_users.py b/estate/models/res_users.py
new file mode 100644
index 00000000000..a320f45e780
--- /dev/null
+++ b/estate/models/res_users.py
@@ -0,0 +1,15 @@
+from odoo import api, fields, models
+
+
+class ResUsers(models.Model):
+ _inherit = 'res.users'
+
+ property_ids = fields.One2many('estate.property', 'user_id')
+ unsold_value = fields.Float(compute='_compute_unsold_value')
+
+ @api.depends('property_ids.expected_price', 'property_ids.state')
+ def _compute_unsold_value(self):
+ sum_unsold_value = dict(self.env['estate.property']._read_group(domain=[('state', '!=', 'sold'), (
+ 'user_id', 'in', self.ids)], aggregates=['expected_price:sum'], groupby=['user_id']))
+ for record in self:
+ record.unsold_value = sum_unsold_value.get(record)
diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv
new file mode 100644
index 00000000000..3e4093797f0
--- /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
+model_estate,model_estate,model_estate_property,base.group_user,1,1,1,1
+model_estate_type,model_estate_type,model_estate_property_type,base.group_user,1,1,1,1
+model_estate_tag,model_estate_tag,model_estate_property_tag,base.group_user,1,1,1,1
+model_estate_offer,model_estate_offer,model_estate_property_offer,base.group_user,1,1,1,1
+model_estate_maintenance,model_estate_maintenance,model_estate_property_maintenance,base.group_user,1,1,1,1
+estate.access_estate_investor,access_estate_investor,estate.model_estate_investor,base.group_user,1,1,1,1
\ No newline at end of file
diff --git a/estate/views/estate_investor_views.xml b/estate/views/estate_investor_views.xml
new file mode 100644
index 00000000000..f11312b21ca
--- /dev/null
+++ b/estate/views/estate_investor_views.xml
@@ -0,0 +1,9 @@
+
+
+
+ Investor
+ estate.investor
+ list,form
+
+
+
diff --git a/estate/views/estate_menus.xml b/estate/views/estate_menus.xml
new file mode 100644
index 00000000000..462370c7773
--- /dev/null
+++ b/estate/views/estate_menus.xml
@@ -0,0 +1,16 @@
+
+
+
+
+
diff --git a/estate/views/estate_property_maintenance_view.xml b/estate/views/estate_property_maintenance_view.xml
new file mode 100644
index 00000000000..2d5ec88bd95
--- /dev/null
+++ b/estate/views/estate_property_maintenance_view.xml
@@ -0,0 +1,22 @@
+
+
+
+ estate.property.maintenance.view.list
+ estate.property.maintenance
+
+
+
+
+
+
+
+
+
+
+
+ Maintenance Request
+ estate.property.maintenance
+ list,form
+
+
+
\ 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..641674d84dd
--- /dev/null
+++ b/estate/views/estate_property_offer_views.xml
@@ -0,0 +1,45 @@
+
+
+
+
+ estate.property.offer.view.list
+ estate.property.offer
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ estate.property.offer.view.form
+ estate.property.offer
+
+
+
+
+
+
+ Property Offer
+ estate.property.offer
+ list,form
+ [('property_type_id', '=', active_id)]
+
+
+
diff --git a/estate/views/estate_property_tag_views.xml b/estate/views/estate_property_tag_views.xml
new file mode 100644
index 00000000000..2e76168df67
--- /dev/null
+++ b/estate/views/estate_property_tag_views.xml
@@ -0,0 +1,35 @@
+
+
+
+ Property Tags
+ estate.property.tag
+ list,form
+
+
+
+ estate.property.tag.view.list
+ estate.property.tag
+
+
+
+
+
+
+
+
+
+ estate.property.tag.view.form
+ 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..b64f2b9df44
--- /dev/null
+++ b/estate/views/estate_property_type_views.xml
@@ -0,0 +1,69 @@
+
+
+
+ estate.property.type.view.list
+ estate.property.type
+
+
+
+
+
+
+
+
+
+ estate.property.type.view.form
+ estate.property.type
+
+
+
+
+
+
+ estate.property.type.view.search
+ estate.property.type
+
+
+
+
+
+
+
+
+
+ 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..479e75648ff
--- /dev/null
+++ b/estate/views/estate_property_views.xml
@@ -0,0 +1,174 @@
+
+
+
+
+ estate.property.view.list
+ estate.property
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ estate.property.view.form
+ estate.property
+
+
+
+
+
+
+ estate.property.view.search
+ estate.property
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ estate.property.view.kanban
+ estate.property
+
+
+
+
+
+
+
+
+
+
+
+ Expected Price:
+
+
+
+ Best Price:
+
+
+
+ Selling Price:
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Properties
+ estate.property
+ list,form,kanban
+ {'search_default_best_price': 0}
+
+
+
diff --git a/estate/views/res_users_views.xml b/estate/views/res_users_views.xml
new file mode 100644
index 00000000000..3dd8c4f0269
--- /dev/null
+++ b/estate/views/res_users_views.xml
@@ -0,0 +1,31 @@
+
+
+ res.users.view.form.inherit.property
+ res.users
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/estate_account/__init__.py b/estate_account/__init__.py
new file mode 100644
index 00000000000..0650744f6bc
--- /dev/null
+++ b/estate_account/__init__.py
@@ -0,0 +1 @@
+from . import models
diff --git a/estate_account/__manifest__.py b/estate_account/__manifest__.py
new file mode 100644
index 00000000000..e8164f1b75c
--- /dev/null
+++ b/estate_account/__manifest__.py
@@ -0,0 +1,8 @@
+{
+ 'name': "estate_account",
+ 'author': "pkhu",
+ 'license': "LGPL-3",
+ 'depends': ['estate', 'account'],
+ 'application': True,
+ 'data': [],
+}
diff --git a/estate_account/models/__init__.py b/estate_account/models/__init__.py
new file mode 100644
index 00000000000..5e1963c9d2f
--- /dev/null
+++ b/estate_account/models/__init__.py
@@ -0,0 +1 @@
+from . import estate_property
diff --git a/estate_account/models/estate_property.py b/estate_account/models/estate_property.py
new file mode 100644
index 00000000000..5a67df5eac0
--- /dev/null
+++ b/estate_account/models/estate_property.py
@@ -0,0 +1,27 @@
+from odoo import Command, models
+
+
+class EstateProperty(models.Model):
+ _inherit = "estate.property"
+
+ def action_property_sold(self):
+ res = super().action_property_sold()
+ for property in self:
+ vals = {
+ 'partner_id': property.partner_id.id,
+ 'move_type': 'out_invoice',
+ 'invoice_line_ids': [
+ Command.create({
+ 'name': property.name,
+ 'quantity': 1,
+ 'price_unit': property.selling_price * 0.06
+ }),
+ Command.create({
+ 'name': "Administrative fees",
+ 'quantity': 1,
+ 'price_unit': 100
+ }),
+ ]
+ }
+ self.env['account.move'].create(vals)
+ return res