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.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