diff --git a/awesome_dashboard/__manifest__.py b/awesome_dashboard/__manifest__.py index a1cd72893d7..58817d5fbd4 100644 --- a/awesome_dashboard/__manifest__.py +++ b/awesome_dashboard/__manifest__.py @@ -1,4 +1,3 @@ -# -*- coding: utf-8 -*- { 'name': "Awesome Dashboard", @@ -21,10 +20,16 @@ 'data': [ 'views/views.xml', ], + 'assets': { 'web.assets_backend': [ 'awesome_dashboard/static/src/**/*', ], + + 'awesome_dashboard.dashboard': [ + 'awesome_dashboard/static/src/dashboard/**/*' + ], }, + 'license': 'AGPL-3' } diff --git a/awesome_dashboard/static/src/dashboard.js b/awesome_dashboard/static/src/dashboard.js deleted file mode 100644 index c4fb245621b..00000000000 --- a/awesome_dashboard/static/src/dashboard.js +++ /dev/null @@ -1,8 +0,0 @@ -import { Component } from "@odoo/owl"; -import { registry } from "@web/core/registry"; - -class AwesomeDashboard extends Component { - static template = "awesome_dashboard.AwesomeDashboard"; -} - -registry.category("actions").add("awesome_dashboard.dashboard", AwesomeDashboard); diff --git a/awesome_dashboard/static/src/dashboard.xml b/awesome_dashboard/static/src/dashboard.xml deleted file mode 100644 index 1a2ac9a2fed..00000000000 --- a/awesome_dashboard/static/src/dashboard.xml +++ /dev/null @@ -1,8 +0,0 @@ - - - - - hello dashboard - - - diff --git a/awesome_dashboard/static/src/dashboard/dashboard.js b/awesome_dashboard/static/src/dashboard/dashboard.js new file mode 100644 index 00000000000..2c988a966f1 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/dashboard.js @@ -0,0 +1,102 @@ +/** @odoo-module **/ +import { Component, useState } from "@odoo/owl"; +import { registry } from "@web/core/registry"; +import { Layout } from "@web/search/layout"; +import { useService } from "@web/core/utils/hooks"; +import { DashboardItem } from "./dashboard_item/dashboard_item"; +import { Dialog } from "@web/core/dialog/dialog"; +import { CheckBox } from "@web/core/checkbox/checkbox"; +import { browser } from "@web/core/browser/browser"; +import "./statistics_service"; +import "./dashboard_items"; + +class AwesomeDashboard extends Component { + static template = "awesome_dashboard.AwesomeDashboard"; + static components = { Layout, DashboardItem }; + + setup() { + this.action = useService("action"); + this.dialog = useService("dialog"); + this.display = { controlPanel: {} }; + this.stats = useService("awesome_dashboard.statistics"); + + this.state = useState({ + disabledItems: JSON.parse( + browser.localStorage.getItem("disabledDashboardItems") || "[]" + ) + }); + + this.allItems = registry + .category("awesome_dashboard.items") + .getAll(); + } + + get items() { + return this.allItems.filter( + item => !this.state.disabledItems.includes(item.id) + ); + } + + openConfiguration() { + this.dialog.add(ConfigurationDialog, { + items: this.allItems, + disabledItems: this.state.disabledItems, + onUpdate: this.updateConfiguration.bind(this), + }); + } + + updateConfiguration(newDisabledItems) { + this.state.disabledItems = newDisabledItems; + browser.localStorage.setItem( + "disabledDashboardItems", + JSON.stringify(newDisabledItems) + ); + } + + openCustomers() { + this.action.doAction("base.action_partner_form"); + } + + openLeads() { + this.action.doAction({ + type: "ir.actions.act_window", + name: "Leads", + res_model: "crm.lead", + views: [[false, "list"], [false, "form"]], + }); + } +} + +class ConfigurationDialog extends Component { + static template = "awesome_dashboard.ConfigurationDialog"; + static components = { Dialog, CheckBox }; + static props = ["close", "items", "disabledItems", "onUpdate"]; + + setup() { + const disabled = Array.isArray(this.props.disabledItems) + ? this.props.disabledItems + : []; + + this.items = useState( + this.props.items.map(item => ({ + ...item, + enabled: !disabled.includes(item.id), + })) + ); + } + + onChange(checked, item) { + item.enabled = checked; + const disabled = this.items + .filter(i => !i.enabled) + .map(i => i.id); + this.props.onUpdate(disabled); + } + + done() { + this.props.close(); + } +} +registry + .category("lazy_components") + .add("awesome_dashboard.AwesomeDashboard", AwesomeDashboard); diff --git a/awesome_dashboard/static/src/dashboard/dashboard.scss b/awesome_dashboard/static/src/dashboard/dashboard.scss new file mode 100644 index 00000000000..ffd265b1982 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/dashboard.scss @@ -0,0 +1,10 @@ +.o_dashboard { + background-color: rgb(16, 18, 20); +} + +@media (max-width: 768px) { + + .o_dashboard_item { + width: 100% !important; + } +} diff --git a/awesome_dashboard/static/src/dashboard/dashboard.xml b/awesome_dashboard/static/src/dashboard/dashboard.xml new file mode 100644 index 00000000000..4df87c37ab1 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/dashboard.xml @@ -0,0 +1,58 @@ + + + + + + + + + + + + + + + + + diff --git a/awesome_dashboard/static/src/dashboard/dashboard_item/dashboard_item.js b/awesome_dashboard/static/src/dashboard/dashboard_item/dashboard_item.js new file mode 100644 index 00000000000..818c69c580f --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/dashboard_item/dashboard_item.js @@ -0,0 +1,9 @@ +import { Component } from "@odoo/owl"; + +export class DashboardItem extends Component { + static template = "awesome_dashboard.DashboardItem"; + + static slots = { + default: {}, + }; +} diff --git a/awesome_dashboard/static/src/dashboard/dashboard_item/dashboard_item.xml b/awesome_dashboard/static/src/dashboard/dashboard_item/dashboard_item.xml new file mode 100644 index 00000000000..82e3f8055d3 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/dashboard_item/dashboard_item.xml @@ -0,0 +1,11 @@ + + + + + +
+ +
+ +
+
diff --git a/awesome_dashboard/static/src/dashboard/dashboard_items.js b/awesome_dashboard/static/src/dashboard/dashboard_items.js new file mode 100644 index 00000000000..695b557c2f8 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/dashboard_items.js @@ -0,0 +1,71 @@ +/** @odoo-module **/ +import { dashboardItemRegistry } from "./dashboard_registry"; +import { NumberCard } from "./number_card/number_card"; +import { PieChartCard } from "./pie_chart_card/pie_chart_card"; +import { _t } from "@web/core/l10n/translation"; + +dashboardItemRegistry.add("new_orders", { + id: "new_orders", + description: _t("New orders this month"), + Component: NumberCard, + size: 1, + props: (data) => ({ + title: _t("Number of new orders"), + value: data?.nb_new_orders || 0, + }), +}); + +dashboardItemRegistry.add("total_amount", { + id: "total_amount", + description: _t("Total amount orders this month"), + Component: NumberCard, + size: 1, + props: (data) => ({ + title: _t("Total amount"), + value: data?.total_amount || 0, + }), +}); + +dashboardItemRegistry.add("avg_quantity", { + id: "avg_quantity", + description: _t("Average amount of t-shirt"), + Component: NumberCard, + size: 1, + props: (data) => ({ + title: _t("Average quantity"), + value: data?.average_quantity || 0, + }), +}); + +dashboardItemRegistry.add("average_time", { + id: "average_time", + description: _t("Average time new to sent/cancelled"), + Component: NumberCard, + size: 1, + props: (data) => ({ + title: _t("Average time"), + value: data?.average_time || 0, + }), +}); + +dashboardItemRegistry.add("nb_cancelled_orders", { + id: "nb_cancelled_orders", + description: _t("Cancelled orders this month"), + Component: NumberCard, + size: 1, + props: (data) => ({ + title: _t("Number of cancelled orders"), + value: data?.nb_cancelled_orders || 0, + }), +}); + +dashboardItemRegistry.add("pie_chart", { + id: "pie_chart", + description: _t("Shirt orders by size"), + Component: PieChartCard, + size: 2, + props: (data) => ({ + title: _t("Orders by size"), + data: data?.orders_by_size || {}, + }), +}); diff --git a/awesome_dashboard/static/src/dashboard/dashboard_registry.js b/awesome_dashboard/static/src/dashboard/dashboard_registry.js new file mode 100644 index 00000000000..e3fe596f7ad --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/dashboard_registry.js @@ -0,0 +1,3 @@ +import { registry } from "@web/core/registry"; + +export const dashboardItemRegistry = registry.category("awesome_dashboard.items"); diff --git a/awesome_dashboard/static/src/dashboard/number_card/number_card.js b/awesome_dashboard/static/src/dashboard/number_card/number_card.js new file mode 100644 index 00000000000..95cc39c8b33 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/number_card/number_card.js @@ -0,0 +1,9 @@ +import { Component } from "@odoo/owl"; + +export class NumberCard extends Component { + static template = "awesome_dashboard.NumberCard"; + static props = { + title: String, + value: { type: Number, optional: true }, + }; +} diff --git a/awesome_dashboard/static/src/dashboard/number_card/number_card.xml b/awesome_dashboard/static/src/dashboard/number_card/number_card.xml new file mode 100644 index 00000000000..1da288b98a9 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/number_card/number_card.xml @@ -0,0 +1,9 @@ + + + +
+
+

+
+
+
diff --git a/awesome_dashboard/static/src/dashboard/pie_chart/pie_chart.js b/awesome_dashboard/static/src/dashboard/pie_chart/pie_chart.js new file mode 100644 index 00000000000..4fb8c3bacaa --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/pie_chart/pie_chart.js @@ -0,0 +1,34 @@ +import { Component, onWillStart, onMounted, useRef } from "@odoo/owl"; +import { loadJS } from "@web/core/assets"; + +export class PieChart extends Component { + static template = "awesome_dashboard.PieChart"; + + static props = { + data: Object, + }; + + setup() { + this.canvasRef = useRef("chart"); + onWillStart(async () => { + await loadJS("/web/static/lib/Chart/Chart.js"); + }); + onMounted(() => { + this.renderChart(); + }); + } + + renderChart() { + const ctx = this.canvasRef.el.getContext("2d"); + + new Chart(ctx, { + type: "pie", + data: { + labels: Object.keys(this.props.data), + datasets: [{ + data: Object.values(this.props.data), + }], + }, + }); + } +} diff --git a/awesome_dashboard/static/src/dashboard/pie_chart/pie_chart.xml b/awesome_dashboard/static/src/dashboard/pie_chart/pie_chart.xml new file mode 100644 index 00000000000..6d356ee58fb --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/pie_chart/pie_chart.xml @@ -0,0 +1,7 @@ + + + + + + + diff --git a/awesome_dashboard/static/src/dashboard/pie_chart_card/pie_chart_card.js b/awesome_dashboard/static/src/dashboard/pie_chart_card/pie_chart_card.js new file mode 100644 index 00000000000..23aa93ec2c3 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/pie_chart_card/pie_chart_card.js @@ -0,0 +1,11 @@ +import { Component } from "@odoo/owl"; +import { PieChart } from "../pie_chart/pie_chart"; + +export class PieChartCard extends Component { + static template = "awesome_dashboard.PieChartCard"; + static components = { PieChart } + static props = { + title: String, + data: Object, + }; +} diff --git a/awesome_dashboard/static/src/dashboard/pie_chart_card/pie_chart_card.xml b/awesome_dashboard/static/src/dashboard/pie_chart_card/pie_chart_card.xml new file mode 100644 index 00000000000..f22aac77842 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/pie_chart_card/pie_chart_card.xml @@ -0,0 +1,11 @@ + + + + +
+
+ +
+
+ +
diff --git a/awesome_dashboard/static/src/dashboard/statistics_service.js b/awesome_dashboard/static/src/dashboard/statistics_service.js new file mode 100644 index 00000000000..fe717b5e350 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard/statistics_service.js @@ -0,0 +1,26 @@ +/** @odoo-module **/ +import { registry } from "@web/core/registry"; +import { rpc } from "@web/core/network/rpc"; +import { reactive } from "@odoo/owl"; + +const statisticsService = { + name: "awesome_dashboard.statistics", + + start() { + const stats = reactive({}); + + async function loadStatistics() { + const data = await rpc("/awesome_dashboard/statistics"); + Object.assign(stats, data); + } + + loadStatistics(); + setInterval(loadStatistics, 10 * 60 * 1000); + + return stats; + }, +}; + +registry + .category("services") + .add("awesome_dashboard.statistics", statisticsService); diff --git a/awesome_dashboard/static/src/dashboard_action.js b/awesome_dashboard/static/src/dashboard_action.js new file mode 100644 index 00000000000..52decad3f86 --- /dev/null +++ b/awesome_dashboard/static/src/dashboard_action.js @@ -0,0 +1,10 @@ +import { Component } from "@odoo/owl"; +import { LazyComponent } from "@web/core/assets"; +import { registry } from "@web/core/registry"; + +export class DashboardAction extends Component { + static components = { LazyComponent }; + static template = "awesome_dashboard.DashboardLoader"; +} + +registry.category("actions").add("awesome_dashboard.dashboard", DashboardAction); diff --git a/awesome_dashboard/static/src/dashboard_action.xml b/awesome_dashboard/static/src/dashboard_action.xml new file mode 100644 index 00000000000..5beaede125b --- /dev/null +++ b/awesome_dashboard/static/src/dashboard_action.xml @@ -0,0 +1,9 @@ + + + + + + diff --git a/awesome_owl/static/src/card/card.js b/awesome_owl/static/src/card/card.js new file mode 100644 index 00000000000..b159b534b05 --- /dev/null +++ b/awesome_owl/static/src/card/card.js @@ -0,0 +1,19 @@ +/** @odoo-module **/ +import { Component, useState } from "@odoo/owl"; + +export class Card extends Component { + static template = "awesome_owl.Card"; + + static props = { + title: { type: String }, + slots: { type: Object, optional: true }, + }; + + setup() { + this.state = useState({ isOpen: true }); + } + + toggleContent() { + this.state.isOpen = !this.state.isOpen; + } +} diff --git a/awesome_owl/static/src/card/card.xml b/awesome_owl/static/src/card/card.xml new file mode 100644 index 00000000000..145406a5024 --- /dev/null +++ b/awesome_owl/static/src/card/card.xml @@ -0,0 +1,33 @@ + + + +
+ +
+ + + + + + + + + + + +
+ +
+ +
+ +
+
+
diff --git a/awesome_owl/static/src/counter/counter.js b/awesome_owl/static/src/counter/counter.js new file mode 100644 index 00000000000..b509cd0672a --- /dev/null +++ b/awesome_owl/static/src/counter/counter.js @@ -0,0 +1,21 @@ +/** @odoo-module **/ +import { Component, useState } from "@odoo/owl"; + +export class Counter extends Component { + static template = "awesome_owl.Counter"; + + static props = { + onChange: { type: Function, optional: true }, + }; + + setup() { + this.state = useState({ value: 0 }); + } + + increment() { + this.state.value++; + if (this.props.onChange) { + this.props.onChange(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..eb3e6b97472 --- /dev/null +++ b/awesome_owl/static/src/counter/counter.xml @@ -0,0 +1,22 @@ + + + +
+ + + Counter: + + + + +
+
+
diff --git a/awesome_owl/static/src/playground.js b/awesome_owl/static/src/playground.js index 4ac769b0aa5..dbb268c4d18 100644 --- a/awesome_owl/static/src/playground.js +++ b/awesome_owl/static/src/playground.js @@ -1,5 +1,20 @@ -import { Component } from "@odoo/owl"; +/** @odoo-module **/ +import { Component, useState, markup } from "@odoo/owl"; +import { Counter } from "./counter/counter"; +import { Card } from "./card/card"; +import { TodoList } from "./todo_list/todo_list"; export class Playground extends Component { - static template = "awesome_owl.playground"; + static template = "awesome_owl.Playground"; + static components = { Counter, Card, TodoList }; + + setup() { + this.state = useState({ sum: 0 }); + this.safeHtml = markup("
some content
"); + this.unsafeHtml = "
some content
"; + } + + incrementSum() { + this.state.sum++; + } } diff --git a/awesome_owl/static/src/playground.xml b/awesome_owl/static/src/playground.xml index 4fb905d59f9..b06c0be9cb3 100644 --- a/awesome_owl/static/src/playground.xml +++ b/awesome_owl/static/src/playground.xml @@ -1,10 +1,32 @@ - + + + +
+ +

Hello Counter

+ + + + +

+ The sum is: +

+ +
+ + This is simple text inside card + + + + +
+ +
+ +
- -
- hello world
-
+ diff --git a/awesome_owl/static/src/todo_list/todo_item.js b/awesome_owl/static/src/todo_list/todo_item.js new file mode 100644 index 00000000000..efc3053a547 --- /dev/null +++ b/awesome_owl/static/src/todo_list/todo_item.js @@ -0,0 +1,19 @@ +/** @odoo-module **/ +import { Component } from "@odoo/owl"; + +export class TodoItem extends Component { + static template = "awesome_owl.TodoItem"; + + static props = { + todo: { + type: Object, + shape: { + id: Number, + description: String, + isCompleted: Boolean, + }, + }, + toggleState: { type: Function }, + removeTodo: { type: Function }, + }; +} diff --git a/awesome_owl/static/src/todo_list/todo_item.xml b/awesome_owl/static/src/todo_list/todo_item.xml new file mode 100644 index 00000000000..14126c9e69d --- /dev/null +++ b/awesome_owl/static/src/todo_list/todo_item.xml @@ -0,0 +1,26 @@ + + + +
+ + + + + . + + + + + +
+
+
diff --git a/awesome_owl/static/src/todo_list/todo_list.js b/awesome_owl/static/src/todo_list/todo_list.js new file mode 100644 index 00000000000..04dfb0f57c7 --- /dev/null +++ b/awesome_owl/static/src/todo_list/todo_list.js @@ -0,0 +1,45 @@ +/** @odoo-module **/ +import { Component, useState, useRef, onMounted } from "@odoo/owl"; +import { TodoItem } from "./todo_item"; + +export class TodoList extends Component { + static template = "awesome_owl.TodoList"; + static components = { TodoItem }; + + setup() { + this.todos = useState([]); + this.nextId = 1; + this.inputRef = useRef("todoInput"); + + onMounted(() => { + this.inputRef.el.focus(); + }); + } + + addTodo(ev) { + if (ev.keyCode === 13) { + const description = ev.target.value.trim(); + if (!description) return; + this.todos.push({ + id: this.nextId++, + description: description, + isCompleted: false, + }); + ev.target.value = ""; + } + } + + toggleState(todoId) { + const todo = this.todos.find(t => t.id === todoId); + if (todo) { + todo.isCompleted = !todo.isCompleted; + } + } + + removeTodo(todoId) { + const index = this.todos.findIndex(t => t.id === todoId); + if (index >= 0) { + this.todos.splice(index, 1); + } + } +} diff --git a/awesome_owl/static/src/todo_list/todo_list.xml b/awesome_owl/static/src/todo_list/todo_list.xml new file mode 100644 index 00000000000..7b19de79303 --- /dev/null +++ b/awesome_owl/static/src/todo_list/todo_list.xml @@ -0,0 +1,26 @@ + + + +
+ +

+ Todo List +

+ + + + + + + +
+
+
diff --git a/bom_forecast/__init__.py b/bom_forecast/__init__.py new file mode 100644 index 00000000000..9186ee3ad24 --- /dev/null +++ b/bom_forecast/__init__.py @@ -0,0 +1 @@ +from . import model diff --git a/bom_forecast/__manifest__.py b/bom_forecast/__manifest__.py new file mode 100644 index 00000000000..53a96141049 --- /dev/null +++ b/bom_forecast/__manifest__.py @@ -0,0 +1,15 @@ +{ + "name": "bom_forecast", + "version": "1.0", + "depends": ["mrp", "purchase"], + "author": "soham", + "category": "Tutorials", + "license": "LGPL-3", + 'installable': True, + 'application': True, + "assets": { + "web.assets_backend": [ + "bom_forecast/static/src/**/*", + ], + }, +} diff --git a/bom_forecast/model/__init__.py b/bom_forecast/model/__init__.py new file mode 100644 index 00000000000..98974148e0e --- /dev/null +++ b/bom_forecast/model/__init__.py @@ -0,0 +1 @@ +from . import report_mrp_bom_structure diff --git a/bom_forecast/model/report_mrp_bom_structure.py b/bom_forecast/model/report_mrp_bom_structure.py new file mode 100644 index 00000000000..a406162251c --- /dev/null +++ b/bom_forecast/model/report_mrp_bom_structure.py @@ -0,0 +1,15 @@ +from odoo import _, models + + +class ReportMrpReportBomStructure(models.AbstractModel): + _inherit = "report.mrp.report_bom_structure" + + def _get_bom_data(self, *args, **kwargs): + result = super()._get_bom_data(*args, **kwargs) + if result.get("level") == 0: + qty = int(result.get("producible_qty") or 0) + if qty > 0: + result["status"] = _("%(qty)s Ready To Produce", qty=qty) + else: + result["status"] = _("No Ready To Produce") + return result diff --git a/bom_forecast/static/src/bom_overview_line.js b/bom_forecast/static/src/bom_overview_line.js new file mode 100644 index 00000000000..058723a2187 --- /dev/null +++ b/bom_forecast/static/src/bom_overview_line.js @@ -0,0 +1,28 @@ +import { patch } from "@web/core/utils/patch"; +import { BomOverviewLine } from "@mrp/components/bom_overview_line/mrp_bom_overview_line"; + +patch(BomOverviewLine.prototype, { + + get statusBackgroundColor() { + switch (this.data.availability_state) { + case "available": return "text-bg-success"; + case "expected": + case "estimated": + if (this.data.level === 0) return "text-bg-dark"; + return "border border-warning text-warning"; + case "unavailable": return "text-bg-danger"; + default: return "text-bg-dark"; + } + }, + + get statusDisplay() { + if (this.data.level > 0 && + this.data.availability_display && + this.data.availability_display.includes("Estimated")) { + return this.data.availability_display.replace( + "Estimated", "Expected" + ); + } + return this.data.availability_display; + } +}); diff --git a/bom_forecast/static/src/bom_overview_line.xml b/bom_forecast/static/src/bom_overview_line.xml new file mode 100644 index 00000000000..c3dc59fb628 --- /dev/null +++ b/bom_forecast/static/src/bom_overview_line.xml @@ -0,0 +1,38 @@ + + + + + + + +
+ + + + + + + + + + + + + + + + + + +
+
+
+
diff --git a/bom_forecast/static/src/mrp_bom_overview_table.xml b/bom_forecast/static/src/mrp_bom_overview_table.xml new file mode 100644 index 00000000000..151a81b4662 --- /dev/null +++ b/bom_forecast/static/src/mrp_bom_overview_table.xml @@ -0,0 +1,7 @@ + + + + + + + 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..5cdf7aed65f --- /dev/null +++ b/estate/__manifest__.py @@ -0,0 +1,22 @@ +{ + 'name': 'Real Estate', + 'author': 'soham', + 'license': 'LGPL-3', + 'depends': ['base', 'calendar', 'mail', 'sale'], + 'data': [ + 'security/ir.model.access.csv', + 'data/mail_template_data.xml', + 'data/ir_cron.xml', + 'views/estate_property_visit_views.xml', + 'views/estate_property_issue_views.xml', + 'views/estate_property_offer_views.xml', + 'views/estate_property_type_views.xml', + 'views/estate_property_tag_views.xml', + 'views/estate_property_wizard_views.xml', + 'views/estate_property_views.xml', + 'views/res_users_views.xml', + 'views/estate_menus.xml', + ], + 'installable': True, + 'application': True, +} diff --git a/estate/data/ir_cron.xml b/estate/data/ir_cron.xml new file mode 100644 index 00000000000..9d1cf480adc --- /dev/null +++ b/estate/data/ir_cron.xml @@ -0,0 +1,12 @@ + + + + Offer: Expired Offers Refused + + code + model._cron_automatic_refuse() + + 5 + minutes + + diff --git a/estate/data/mail_template_data.xml b/estate/data/mail_template_data.xml new file mode 100644 index 00000000000..dac0617beb4 --- /dev/null +++ b/estate/data/mail_template_data.xml @@ -0,0 +1,27 @@ + + + + + Estate: Property Sold + + {{object.name}} Sold + + {{ object.salesman_id.email_formatted }} + + +
+

Dear ,

+

Property + + has been sold.

+

Seller:

+

Selling Price:

+

Description:

+

Thank you!

+
+
+
+
+
diff --git a/estate/models/__init__.py b/estate/models/__init__.py new file mode 100644 index 00000000000..dfacdea237e --- /dev/null +++ b/estate/models/__init__.py @@ -0,0 +1,8 @@ +from . import estate_property +from . import estate_property_type +from . import estate_property_tag +from . import estate_property_offer +from . import res_users +from . import estate_property_visit +from . import estate_property_issue +from . import estate_property_wizard diff --git a/estate/models/estate_property.py b/estate/models/estate_property.py new file mode 100644 index 00000000000..41b538fe1e7 --- /dev/null +++ b/estate/models/estate_property.py @@ -0,0 +1,261 @@ +from odoo import api, Command, 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' + _inherit = ['mail.thread', 'mail.activity.mixin'] + _description = "Real Estate Property" + _order = "id desc" + + name = fields.Char(required=True) + expected_price = fields.Float(required=True) + description = fields.Text() + postcode = fields.Char(tracking=1) + date_availability = fields.Date(copy=False, default=lambda self: fields.Date.add(fields.Date.today(), months=3)) + selling_price = fields.Float(readonly=True, copy=False) + bedrooms = fields.Integer(default=2) + living_area = fields.Integer() + sold_date = fields.Date(string="Sold Date", readonly=True, copy=False) + active = fields.Boolean(default=True) + facades = fields.Integer() + garage = fields.Boolean() + garden = fields.Boolean() + garden_area = fields.Integer() + garden_orientation = fields.Selection( + string="Orientation", + selection=[('north', "North"), ('south', "South"), ('east', "East"), ('west', "West")], + help="The direction the garden faces." + ) + state = fields.Selection( + selection=[ + ('new', "New"), + ('offer_received', "Offer Received"), + ('offer_accepted', "Offer Accepted"), + ('sold', "Sold"), + ('canceled', "Cancelled"), + ], + string="Status", + required=True, + copy=False, + default='new', + ) + property_type_id = fields.Many2one( + "estate.property.type", + string="Property Type", + ) + buyer_id = fields.Many2one( + "res.partner", + string="Buyer", + copy=False, + ) + salesman_id = fields.Many2one( + "res.users", + string="Salesman", + default=lambda self: self.env.user, + ) + tag_ids = fields.Many2many( + "estate.property.tag", + string="Tags", + compute="_compute_tags", + store=True, + ) + offer_ids = fields.One2many( + "estate.property.offer", + "property_id", + string="Offers", + copy=True + ) + issue_ids = fields.One2many( + 'estate.property.issue', + 'property_id', + ) + visit_ids = fields.One2many( + "estate.property.visit", + "property_id", + string="Visits", + ) + total_area = fields.Integer( + compute="_compute_total_area", + string="Total Area (sqm)", + ) + best_price = fields.Float( + compute="_compute_best_price", + string="Best Offer", + store=True, + ) + has_suspicious_offers = fields.Boolean( + string="Has Suspicious Offers", + compute="_compute_has_suspicious_offers" + ) + visit_count = fields.Integer( + string="Visits Count", + compute="_compute_visit_count", + store=True, + ) + + _check_expected_price = models.Constraint( + 'CHECK(expected_price > 0)', + 'Expected price must be strictly positive!', + ) + _check_selling_price = models.Constraint( + 'CHECK(selling_price >= 0)', + 'Selling price must be positive!', + ) + + @api.depends("visit_ids") + def _compute_visit_count(self): + for record in self: + record.visit_count = len(record.visit_ids) + + @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_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 + + @api.depends("offer_ids.is_suspicious") + def _compute_has_suspicious_offers(self): + for record in self: + record.has_suspicious_offers = any( + offer.is_suspicious for offer in record.offer_ids + ) + + @api.depends('expected_price', 'offer_ids', 'sold_date', 'date_availability', 'state') + def _compute_tags(self): + for record in self: + tag_records = self.env['estate.property.tag'] + + if record.expected_price > 200000: + tag_records |= self._get_or_create_tag('High Value') + + if (record.state == 'sold' + and record.sold_date + and record.date_availability + and (record.sold_date - record.date_availability).days <= 10): + tag_records |= self._get_or_create_tag('Quick Sale') + + if len(record.offer_ids) < 2: + tag_records |= self._get_or_create_tag('Low Interest') + + record.tag_ids = tag_records + + @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 + + @api.constrains('selling_price', 'expected_price') + def _check_seling_price(self): + for record in self: + if float_is_zero( + record.selling_price, + precision_digits=2 + ): + continue + if float_compare( + record.selling_price, + record.expected_price * 0.9, + precision_digits=2 + ) < 0: + raise ValidationError("Selling price cannot be lower than 90%" + "of expected price!") + + def action_sold(self): + for record in self: + if record.state != 'offer_accepted': + raise UserError("Accept the property first") + if record.state == 'canceled': + raise UserError("Cancelled property cannot be sold!") + for rec in record.issue_ids: + if rec.staty != 'resolved' and rec.priority == 'high': + raise UserError( + "Cannot sell the property, please solve the issues" + ) + record.state = 'sold' + record.sold_date = fields.Date.today() + + ctx = { + 'default_model': 'estate.property', + 'default_res_ids': self.ids, + 'default_template_id': self.env.ref('estate.email_template_estate_property_sold').id, + 'default_partner_ids': [self.buyer_id.id, self.salesman_id.partner_id.id], + } + + return { + 'name': 'Sold', + 'type': 'ir.actions.act_window', + 'view_mode': 'form', + 'res_model': 'mail.compose.message', + 'views': [(False, 'form')], + 'view_id': False, + 'target': 'new', + 'context': ctx, + } + + def action_cancel(self): + for record in self: + if record.state == 'sold': + raise UserError( + "Sold property cannot be cancelled!" + ) + record.state = 'canceled' + return True + + def action_create_quotations(self): + if not self.offer_ids: + raise UserError("No offers found") + + for offer in self.offer_ids: + if offer.quotation_id: + continue + quotation = self.env["sale.order"].create({ + "partner_id": offer.partner_id.id, + "origin": self.name, + "order_line": [ + Command.create({ + "name": self.name, + "product_uom_qty": 1, + "price_unit": offer.price, + }) + ], + }) + + offer.quotation_id = quotation.id + return True + + def _search_tag(self, tag_name): + return self.env['estate.property.tag'].search( + [('name', '=', tag_name)], limit=1 + ) + + def _create_tag(self, tag_name): + return self.env['estate.property.tag'].create( + {'name': tag_name} + ) + + def _get_or_create_tag(self, tag_name): + return self._search_tag(tag_name) or self._create_tag(tag_name) + + @api.ondelete(at_uninstall=False) + def _unlink_if_new_or_canceled(self): + for record in self: + if record.state not in ('new', 'canceled'): + raise UserError( + "Only new and cancelled properties can be deleted!" + ) diff --git a/estate/models/estate_property_issue.py b/estate/models/estate_property_issue.py new file mode 100644 index 00000000000..05066253f3b --- /dev/null +++ b/estate/models/estate_property_issue.py @@ -0,0 +1,105 @@ +from datetime import timedelta +from odoo import api, fields, models + + +class EstatePropertyIssue(models.Model): + _name = "estate.property.issue" + _description = "Property Issue" + + name = fields.Char(required=True) + property_id = fields.Many2one('estate.property', required=True) + + buyer_id = fields.Many2one( + "res.partner", + string="Reported By", + copy=False, + ) + + salesman_id = fields.Many2one( + "res.users", + string="Assigned By", + ) + + issue_type = fields.Selection( + selection=[ + ('plumbing', "Plumbing"), + ('electrical', "Electrical"), + ('structural', "Structural"), + ('others', "Others"), + ], + required=True, + ) + + staty = fields.Selection( + selection=[ + ('new', "New"), + ('in_progress', "In Progress"), + ('resolved', "Resolved"), + ('canceled', "Canceled"), + ], + default='new' + ) + + priority = fields.Selection( + selection=[ + ('low', "Low"), + ('medium', "Medium"), + ('high', "High"), + ], + compute="_compute_priority", + store=True, + readonly=True, + ) + + resolved_date = fields.Datetime() + description_issue = fields.Text() + + is_overdue = fields.Boolean( + compute="_compute_is_overdue", + store=True, + string="Overdue", + ) + + @api.depends("issue_type") + def _compute_priority(self): + for rec in self: + if rec.issue_type in ('others', 'plumbing'): + rec.priority = 'low' + elif rec.issue_type == 'structural': + rec.priority = 'high' + elif rec.issue_type == 'electrical': + rec.priority = 'medium' + else: + rec.priority = False + + @api.depends('create_date', 'priority', 'resolved_date') + def _compute_is_overdue(self): + for rec in self: + if not rec.create_date: + rec.is_overdue = False + continue + + if rec.priority == 'high': + days = 2 + elif rec.priority == 'medium': + days = 5 + else: + days = 10 + + deadline = rec.create_date + timedelta(days=days) + end_time = rec.resolved_date or fields.Datetime.now() + rec.is_overdue = end_time > deadline + + @api.onchange("salesman_id") + def _onchange_salesman_id(self): + if self.salesman_id: + self.staty = 'in_progress' + + def action_resolve(self): + for rec in self: + rec.staty = 'resolved' + rec.resolved_date = fields.Datetime.now() + + def action_cancel(self): + for rec in self: + rec.staty = 'canceled' diff --git a/estate/models/estate_property_offer.py b/estate/models/estate_property_offer.py new file mode 100644 index 00000000000..e9de6781318 --- /dev/null +++ b/estate/models/estate_property_offer.py @@ -0,0 +1,149 @@ +from datetime import timedelta +from odoo import api, fields, models +from odoo.exceptions import UserError + + +class EstatePropertyOffer(models.Model): + _name = "estate.property.offer" + _description = "Real Estate Property Offer" + _order = "price desc" + + price = fields.Float() + status = fields.Selection( + selection=[ + ('accepted', "Accepted"), + ('refused', "Refused"), + ], + copy=True + ) + partner_id = fields.Many2one( + "res.partner", + string="Buyer", + required=True + ) + property_id = fields.Many2one( + "estate.property", + string="Property", + required=True + ) + property_type_id = fields.Many2one( + "estate.property.type", + related="property_id.property_type_id", + string="Property Type", + store=True + ) + validity = fields.Integer( + string="Validity (days)", + default=7, + ) + quotation_id = fields.Many2one( + "sale.order", + string="Quotation", + readonly=True, + copy=False + ) + date_deadline = fields.Date( + string="Deadline", + compute="_compute_date_deadline", + inverse="_set_date_deadline", + ) + is_suspicious = fields.Boolean( + string="Suspicious", + default=False, + readonly=True, + copy=False, + compute="_compute_is_suspicious", + store=True, + ) + + _check_price = models.Constraint( + 'CHECK(price > 0)', + 'Offer price must be strictly positive!', + ) + + @api.depends("validity", "create_date") + def _compute_date_deadline(self): + for record in self: + if record.create_date: + record.date_deadline = fields.Date.add( + record.create_date, + days=record.validity, + ) + else: + record.date_deadline = fields.Date.add( + fields.Date.today(), + days=record.validity, + ) + + def _set_date_deadline(self): + for record in self: + if record.create_date and record.date_deadline: + record.validity = ( + record.date_deadline - + record.create_date.date() + ).days + + @api.depends("partner_id", "create_date") + def _compute_is_suspicious(self): + for record in self: + if not record.create_date: + record.is_suspicious = False + continue + + start = (record.create_date - timedelta(minutes=5)) + end = (record.create_date + timedelta(minutes=5)) + recent_offers = self.search([ + ('partner_id', '=', record.partner_id.id), + ('create_date', '>=', start), + ('create_date', '<=', end), + ]) + + record.is_suspicious = len(recent_offers) >= 3 + + @api.model_create_multi + def create(self, vals_list): + for vals in vals_list: + property_id = self.env['estate.property'].browse( + vals.get('property_id', False) + ) + if property_id.offer_ids: + max_offer = max(property_id.offer_ids.mapped('price')) + if vals.get('price', 0) < max_offer: + raise UserError( + "You cannot create an offer lower " + "than an existing offer of %.2f" % max_offer + ) + + property_id.state = 'offer_received' + + return super().create(vals_list) + + def action_accept(self): + if any(offer.status == 'accepted' for offer in self.property_id.offer_ids): + raise UserError("An offer has already been accepted for this property.") + self.status = 'accepted' + self.property_id.selling_price = self.price + self.property_id.buyer_id = self.partner_id + self.property_id.state = 'offer_accepted' + + for offer in self.property_id.offer_ids: + if offer.id != self.id: + offer.status = 'refused' + return True + + def action_refuse(self): + for record in self: + record.status = 'refused' + return True + + def _cron_automatic_refuse(self): + today = fields.Datetime.now() + offers = self.search([ + ('status', "not in", ["accepted", "refused"]), + ]) + for offer in offers: + expiry_date = offer.create_date + timedelta(days=offer.validity) + if expiry_date < today: + offer.write({ + 'status': 'refused' + }) diff --git a/estate/models/estate_property_tag.py b/estate/models/estate_property_tag.py new file mode 100644 index 00000000000..eea1a456238 --- /dev/null +++ b/estate/models/estate_property_tag.py @@ -0,0 +1,15 @@ +from odoo import fields, models + + +class EstatePropertyTag(models.Model): + _name = "estate.property.tag" + _description = "Real Estate Property Tag" + _order = "name" + + name = fields.Char(required=True) + color = fields.Integer(string="Color") + + _unique_name = models.Constraint( + 'UNIQUE(name)', + 'Property tag name must be unique!', + ) diff --git a/estate/models/estate_property_type.py b/estate/models/estate_property_type.py new file mode 100644 index 00000000000..0580974d8fa --- /dev/null +++ b/estate/models/estate_property_type.py @@ -0,0 +1,40 @@ +from odoo import api, fields, models + + +class EstatePropertyType(models.Model): + _name = "estate.property.type" + _description = "Real Estate Property Type" + _order = "sequence,name" + + name = fields.Char() + sequence = fields.Integer( + string="Sequence", + default=1, + help="Used to order property types manually." + ) + + property_ids = fields.One2many( + "estate.property", + "property_type_id", + string="Properties" + ) + offer_ids = fields.One2many( + "estate.property.offer", + "property_type_id", + string="Offers" + ) + + offer_count = fields.Integer( + string="Offers Count", + compute="_compute_offer_count" + ) + + _unique_name = models.Constraint( + 'UNIQUE(name)', + 'Property type name 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/estate_property_visit.py b/estate/models/estate_property_visit.py new file mode 100644 index 00000000000..2522c0bb31d --- /dev/null +++ b/estate/models/estate_property_visit.py @@ -0,0 +1,36 @@ +from datetime import timedelta +from odoo import api, fields, models + + +class EstatePropertyVisit(models.Model): + _name = "estate.property.visit" + _description = "Property Visits" + _rec_name = "partner_id" + + property_id = fields.Many2one("estate.property", string="Property") + partner_id = fields.Many2one("res.partner", string="Customer") + visit_date = fields.Datetime(required=True) + stato = fields.Selection( + selection=[ + ('scheduled', "Scheduled"), + ('done', "Done"), + ], + default='scheduled' + ) + + _unique_date = models.Constraint( + 'UNIQUE(property_id, visit_date)', + 'Already Scheduled!!', + ) + + @api.model_create_multi + def create(self, vals_list): + records = super().create(vals_list) + + for record in records: + self.env['calendar.event'].create({ + 'name': "Visit", + 'start': record.visit_date, + 'stop': record.visit_date + timedelta(hours=1), + }) + return records diff --git a/estate/models/estate_property_wizard.py b/estate/models/estate_property_wizard.py new file mode 100644 index 00000000000..42cb0528067 --- /dev/null +++ b/estate/models/estate_property_wizard.py @@ -0,0 +1,27 @@ +from odoo import fields, models + + +class EstatePropertyWizard(models.TransientModel): + _name = "estate.property.wizard" + _description = "Create Offers" + + price = fields.Float() + + partner_id = fields.Many2one( + "res.partner", + string="Buyer", + copy=False, + ) + + def action_create_offers(self): + + properties = self.env["estate.property"].search([ + ("state", "not in", ["sold", "canceled", "offer_accepted"]) + ]) + for property in properties: + self.env["estate.property.offer"].create({ + "price": self.price, + "partner_id": self.partner_id.id, + "property_id": property.id, + }) + return True diff --git a/estate/models/res_users.py b/estate/models/res_users.py new file mode 100644 index 00000000000..f88e1bb18cc --- /dev/null +++ b/estate/models/res_users.py @@ -0,0 +1,12 @@ +from odoo import fields, models + + +class ResUsers(models.Model): + _inherit = "res.users" + + property_ids = fields.One2many( + "estate.property", + "salesman_id", + string="Assigned Properties", + domain=[('state', 'in', ['new', 'offer_received'])] + ) diff --git a/estate/security/ir.model.access.csv b/estate/security/ir.model.access.csv new file mode 100644 index 00000000000..051e7c3e2c4 --- /dev/null +++ b/estate/security/ir.model.access.csv @@ -0,0 +1,8 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_estate_property,access_estate_property,model_estate_property,base.group_user,1,1,1,1 +access_estate_property_type,access_estate_property_type,model_estate_property_type,base.group_user,1,1,1,1 +access_estate_property_tag,access_estate_property_tag,model_estate_property_tag,base.group_user,1,1,1,1 +access_estate_property_offer,access_estate_property_offer,model_estate_property_offer,base.group_user,1,1,1,1 +access_estate_property_visit,estate.property.visit,model_estate_property_visit,base.group_user,1,1,1,1 +access_estate_property_issue,estate.property.issue,model_estate_property_issue,base.group_user,1,1,1,1 +access_estate_property_wizard,estate.property.wizard,model_estate_property_wizard,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..25fad3f77c3 --- /dev/null +++ b/estate/views/estate_menus.xml @@ -0,0 +1,26 @@ + + + + + + + + + + + + + + + + + + diff --git a/estate/views/estate_property_issue_views.xml b/estate/views/estate_property_issue_views.xml new file mode 100644 index 00000000000..560e46741cf --- /dev/null +++ b/estate/views/estate_property_issue_views.xml @@ -0,0 +1,71 @@ + + + Property Issue + estate.property.issue + list,form + + + + estate.property.issue.view.list + estate.property.issue + + + + + + + + + + + + + + + estate.property.issue.view.form + estate.property.issue + +
+
+
+ +
+

+ +

+
+ + + + + + + + + + + + + + + + + + + +
+
+
+
+ + + Property Issue + estate.property.issue + list,form + [('property_id', '=', active_id)] + {'default_property_id': active_id} + +
diff --git a/estate/views/estate_property_offer_views.xml b/estate/views/estate_property_offer_views.xml new file mode 100644 index 00000000000..3dc3f36061e --- /dev/null +++ b/estate/views/estate_property_offer_views.xml @@ -0,0 +1,66 @@ + + + Property Offers + estate.property.offer + list,form + + + + estate.property.offer.view.list + 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..658f344eee8 --- /dev/null +++ b/estate/views/estate_property_views.xml @@ -0,0 +1,213 @@ + + + Properties + estate.property + kanban,list,form + {'search_default_available': 1} + + + + estate.property.view.list + estate.property + + +
+
+ + + + + + + + + + + +
+

+ +

+
+ + + + + + +
+ + This property has some suspicious offers +
+
+ + + + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +