diff --git a/mobile_service_shop/README.rst b/mobile_service_shop/README.rst new file mode 100644 index 000000000..94afe0acf --- /dev/null +++ b/mobile_service_shop/README.rst @@ -0,0 +1,34 @@ +Mobile Service Management +========================= +Module for mobile service shop daily activities + +Configuration +============= +* No additional configurations needed + +Company +------- +* `Cybrosys Techno Solutions `__ + +Credits +------- +* Developers: Milind Mohan @ Cybrosys, odoo@cybrosys.com + Mohammed Shahil M P @cybrosys, odoo@cybrosys.com + +Bug Tracker +----------- +Bugs are tracked on GitHub Issues. In case of trouble, please check there if your issue has already been reported. + +Maintainer +========== +.. image:: https://cybrosys.com/images/logo.png + :target: https://cybrosys.com + +This module is maintained by Cybrosys Technologies. + +For support and more information, please visit `Our Website `__ + +Further information +=================== +HTML Description: ``__ + diff --git a/mobile_service_shop/__init__.py b/mobile_service_shop/__init__.py new file mode 100644 index 000000000..52d4e9e57 --- /dev/null +++ b/mobile_service_shop/__init__.py @@ -0,0 +1,24 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU LESSER +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# + +from . import models +from . import wizard diff --git a/mobile_service_shop/__manifest__.py b/mobile_service_shop/__manifest__.py new file mode 100644 index 000000000..8b7f3ff20 --- /dev/null +++ b/mobile_service_shop/__manifest__.py @@ -0,0 +1,50 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU AFFERO +# GENERAL PUBLIC LICENSE (LGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU LESSER GENERAL PUBLIC LICENSE (LGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# +{ + 'name': 'Mobile Service Management', + 'version': '13.0.1.0.0', + 'summary': 'Module for managing mobile service shop daily activities.', + 'category': 'Industries', + 'author': 'Cybrosys Techno Solutions', + 'company': 'Cybrosys Techno Solutions', + 'website': 'https://www.cybrosys.com', + 'depends': ['base', 'stock_account', 'mail', 'product', 'account'], + 'data': ['security/security.xml', + 'security/ir.model.access.csv', + 'views/mobile_service_views.xml', + 'views/product_template.xml', + 'views/terms_and_condition.xml', + 'views/complaint_template.xml', + 'views/complaint_type.xml', + 'views/brand_models.xml', + 'views/brand.xml', + 'wizard/mobile_create_invoice_views.xml', + 'reports/mobile_service_ticket.xml', + 'reports/service_ticket_template.xml', + 'data/mobile_service_data.xml', + 'data/mobile_service_email_template.xml'], + 'images': ['static/description/banner.jpg'], + 'installable': True, + 'application': True, + 'auto_install': False, + 'license': 'LGPL-3', +} \ No newline at end of file diff --git a/mobile_service_shop/data/mobile_service_data.xml b/mobile_service_shop/data/mobile_service_data.xml new file mode 100644 index 000000000..27e3edf1f --- /dev/null +++ b/mobile_service_shop/data/mobile_service_data.xml @@ -0,0 +1,24 @@ + + + + + + Mobile Service + sale + SERV + TRUE + + + Mobile Service Charge + service + order + + + Mobile Service Advance + service + order + + + + + \ No newline at end of file diff --git a/mobile_service_shop/data/mobile_service_email_template.xml b/mobile_service_shop/data/mobile_service_email_template.xml new file mode 100644 index 000000000..99e7894a4 --- /dev/null +++ b/mobile_service_shop/data/mobile_service_email_template.xml @@ -0,0 +1,24 @@ + + + + + + + Service Status Email + + ${(object.user_id.email |safe} + ${object.person_name.email |safe} + Ref ${object.name or 'n/a' } + + ${object.person_name.lang} + Dear ${object.person_name.name},

+

Your service request for ${object.brand_name.brand_name or 'n/a' } (${object.model_name.mobile_brand_models or 'n/a' }) with reference no ${object.name or 'n/a' } has been processed to ${object.service_state or 'n/a' } stage.

+

If you have any questions, please let us know.

+

Best regards,

]]> +
+
+ +
+ +
\ No newline at end of file diff --git a/mobile_service_shop/doc/RELEASE_NOTES.md b/mobile_service_shop/doc/RELEASE_NOTES.md new file mode 100755 index 000000000..c5459858f --- /dev/null +++ b/mobile_service_shop/doc/RELEASE_NOTES.md @@ -0,0 +1,7 @@ +## Module + +#### 27.11.2019 +#### Version 13.0.1.0.0 +#### ADD + +Initial Commit. diff --git a/mobile_service_shop/models/__init__.py b/mobile_service_shop/models/__init__.py new file mode 100644 index 000000000..be71dbc02 --- /dev/null +++ b/mobile_service_shop/models/__init__.py @@ -0,0 +1,25 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU AFFERO +# GENERAL PUBLIC LICENSE (AGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# +from . import mobile_service +from . import service_ticket +from . import product_order_line +from . import product \ No newline at end of file diff --git a/mobile_service_shop/models/mobile_service.py b/mobile_service_shop/models/mobile_service.py new file mode 100644 index 000000000..952ddd972 --- /dev/null +++ b/mobile_service_shop/models/mobile_service.py @@ -0,0 +1,374 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU AFFERO +# GENERAL PUBLIC LICENSE (AGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# +############################################################################# +from datetime import datetime, date, timedelta +from odoo import models, fields, api, _ +from odoo.exceptions import Warning, UserError +import pytz + + +class MobileServiceShop(models.Model): + _name = 'mobile.service' + _rec_name = 'name' + _inherit = ['mail.thread', 'mail.activity.mixin'] + + name = fields.Char(string='Service Number', copy=False, default="New") + person_name = fields.Many2one('res.partner', string="Customer Name", required=True) + contact_no = fields.Char(related='person_name.mobile', string="Contact Number") + email_id = fields.Char(related='person_name.email', string="Email") + + street = fields.Char(related='person_name.street', string="Address") + street2 = fields.Char(related='person_name.street2', string="Address") + city = fields.Char(related='person_name.city', string="Address") + state_id = fields.Many2one(related='person_name.state_id', string="Address") + zip = fields.Char(related='person_name.zip', string="Address") + country_id = fields.Many2one(related='person_name.country_id', string="Address") + + brand_name = fields.Many2one('mobile.brand', string="Mobile Brand") + is_in_warranty = fields.Boolean( + 'In Warranty', default=False, + help="Specify if the product is in warranty.") + + warranty_number = fields.Char(string="Warranty No ", help="warranty details") + + re_repair = fields.Boolean( + 'Re-repair', default=False, + help="Re-repairing.") + + imei_no = fields.Char(string="IMEI Number") + + model_name = fields.Many2one('brand.model', string="Model", domain="[('mobile_brand_name','=',brand_name)]") + image_medium = fields.Binary(related='model_name.image_medium', store=True, attachment=True) + date_request = fields.Date(string="Requested date", default=fields.Date.context_today) + return_date = fields.Date(string="Return date", required=True) + technician_name = fields.Many2one('res.users', string="Technician Name", + default=lambda self: self.env.user, required=True) + service_state = fields.Selection([('draft', 'Draft'), ('assigned', 'Assigned'), + ('completed', 'Completed'), ('returned', 'Returned'), + ('not_solved', 'Not solved')], + string='Service Status', default='draft', track_visibility='always') + + complaints_tree = fields.One2many('mobile.complaint.tree', 'complaint_id', string='Complaints Tree') + + product_order_line = fields.One2many('product.order.line', 'product_order_id', string='Parts Order Lines') + + internal_notes = fields.Text(string="Internal notes") + invoice_count = fields.Integer(compute='_invoice_count', string='# Invoice', copy=False) + invoice_ids = fields.Many2many("account.move", string='Invoices', compute="_get_invoiced", readonly=True, + copy=False) + + first_payment_inv = fields.Many2one('account.move', copy=False) + + first_invoice_created = fields.Boolean(string="First Invoice Created", invisible=True, copy=False) + + journal_type = fields.Many2one('account.journal', 'Journal', invisible=True, + default=lambda self: self.env['account.journal'].search([('code', '=', 'SERV')])) + + company_id = fields.Many2one('res.company', 'Company', + default=lambda self: self.env['res.company']._company_default_get('mobile.service')) + + @api.model + def _default_picking_transfer(self): + type_obj = self.env['stock.picking.type'] + company_id = self.env.context.get('company_id') or self.env.user.company_id.id + types = type_obj.search([('code', '=', 'outgoing'), ('warehouse_id.company_id', '=', company_id)], limit=1) + if not types: + types = type_obj.search([('code', '=', 'outgoing'), ('warehouse_id', '=', False)]) + return types[:4] + + stock_picking_id = fields.Many2one('stock.picking', string="Picking Id") + + picking_transfer_id = fields.Many2one('stock.picking.type', 'Deliver To', required=True, + default=_default_picking_transfer, + help="This will determine picking type of outgoing shipment") + + @api.onchange('return_date') + def check_date(self): + if self.return_date != False: + return_date_string = datetime.strptime(str(self.return_date), "%Y-%m-%d") + request_date_string = datetime.strptime(str(self.date_request), "%Y-%m-%d") + if return_date_string < request_date_string: + raise UserError("Return date should be greater than requested date") + + def approve(self): + self.service_state = 'assigned' + + def complete(self): + self.service_state = 'completed' + + def return_to(self): + self.service_state = 'returned' + + def not_solved(self): + self.service_state = 'not_solved' + + def action_send_mail(self): + ''' + This function opens a window to compose an email, with the edi sale template message loaded by default + ''' + self.ensure_one() + ir_model_data = self.env['ir.model.data'] + try: + template_id = ir_model_data.get_object_reference('mobile_service_shop', 'email_template_mobile_service')[1] + except ValueError: + template_id = False + try: + compose_form_id = ir_model_data.get_object_reference('mail', 'email_compose_message_wizard_form')[1] + except ValueError: + compose_form_id = False + ctx = { + 'default_model': 'mobile.service', + 'default_res_id': self.ids[0], + 'default_use_template': bool(template_id), + 'default_template_id': template_id, + 'default_composition_mode': 'comment', + } + return { + 'name': _('Compose Email'), + 'type': 'ir.actions.act_window', + 'view_mode': 'form', + 'res_model': 'mail.compose.message', + 'views': [(compose_form_id, 'form')], + 'view_id': compose_form_id, + 'target': 'new', + 'context': ctx, + } + + def return_advance(self): + inv_obj = self.env['account.move'].search([('invoice_origin', '=', self.name)]) + inv_ids = [] + for each in inv_obj: + inv_ids.append(each.id) + view_id = self.env.ref('account.view_move_form').id + if inv_ids: + if len(inv_ids) <= 1: + value = { + 'view_mode': 'form', + 'res_model': 'account.move', + 'view_id': view_id, + 'type': 'ir.actions.act_window', + 'name': 'Invoice', + 'res_id': inv_ids and inv_ids[0] + } + else: + value = { + 'domain': str([('id', 'in', inv_ids)]), + 'view_mode': 'tree,form', + 'res_model': 'account.move', + 'view_id': False, + 'type': 'ir.actions.act_window', + 'name': 'Invoice', + 'res_id': inv_ids + } + + return value + else: + raise UserError("No invoice created") + + def _invoice_count(self): + invoice_ids = self.env['account.move'].search([('invoice_origin', '=', self.name)]) + self.invoice_count = len(invoice_ids) + + @api.model + def create(self, vals): + print(self.env.user.company_id) + if 'company_id' in vals: + vals['name'] = self.env['ir.sequence'].with_context(force_company=self.env.user.company_id.id).next_by_code( + 'mobile.service') or _('New') + else: + vals['name'] = self.env['ir.sequence'].next_by_code('mobile.service') or _('New') + vals['service_state'] = 'draft' + return super(MobileServiceShop, self).create(vals) + + def unlink(self): + for i in self: + if i.service_state != 'draft': + raise UserError(_('You cannot delete an assigned service request')) + return super(MobileServiceShop, self).unlink() + + def action_invoice_create_wizard(self): + + return { + 'name': _('Create Invoice'), + 'view_mode': 'form', + 'res_model': 'mobile.invoice', + 'type': 'ir.actions.act_window', + 'target': 'new' + } + + def action_post_stock(self): + flag = 0 + for order in self.product_order_line: + if order.product_uom_qty > order.qty_stock_move: + flag = 1 + pick = { + 'picking_type_id': self.picking_transfer_id.id, + 'partner_id': self.person_name.id, + 'origin': self.name, + 'location_dest_id': self.person_name.property_stock_customer.id, + 'location_id': self.picking_transfer_id.default_location_src_id.id, + } + + picking = self.env['stock.picking'].create(pick) + self.stock_picking_id = picking.id + self.picking_count = len(picking) + moves = order.filtered( + lambda r: r.product_id.type in ['product', 'consu'])._create_stock_moves_transfer(picking) + move_ids = moves._action_confirm() + move_ids._action_assign() + if order.product_uom_qty < order.qty_stock_move: + raise UserError(_('Used quantity is less than quantity stock move posted. ')) + if flag != 1: + raise UserError(_('Nothing to post stock move')) + if flag != 1: + raise UserError(_('Nothing to post stock move')) + + def action_view_invoice(self): + inv_obj = self.env['account.move'].search([('invoice_origin', '=', self.name)]) + inv_ids = [] + for each in inv_obj: + inv_ids.append(each.id) + view_id = self.env.ref('account.view_move_form').id + ctx = dict( + create=False, + ) + if inv_ids: + if len(inv_ids) <= 1: + value = { + 'view_mode': 'form', + 'res_model': 'account.move', + 'view_id': view_id, + 'type': 'ir.actions.act_window', + 'name': 'Invoice', + 'context': ctx, + 'res_id': inv_ids and inv_ids[0] + } + else: + value = { + 'domain': str([('id', 'in', inv_ids)]), + 'view_mode': 'tree,form', + 'res_model': 'account.move', + 'view_id': False, + 'type': 'ir.actions.act_window', + 'context': ctx, + 'name': 'Invoice', + 'res_id': inv_ids + } + + return value + + def get_ticket(self): + self.ensure_one() + user = self.env['res.users'].browse(self.env.uid) + if user.tz: + tz = pytz.timezone(user.tz) + time = pytz.utc.localize(datetime.now()).astimezone(tz) + date_today = time.strftime("%Y-%m-%d %H:%M %p") + else: + date_today = datetime.strftime(datetime.now(), "%Y-%m-%d %I:%M:%S %p") + complaint_text = "" + description_text = "" + complaint_id = self.env['mobile.complaint.tree'].search([('complaint_id', '=', self.id)]) + if complaint_id: + for obj in complaint_id: + complaint = obj.complaint_type_tree + description = obj.description_tree + complaint_text = complaint.complaint_type + ", " + complaint_text + if description.description: + description_text = description.description + ", " + description_text + else: + for obj in complaint_id: + complaint = obj.complaint_type_tree + complaint_text = complaint.complaint_type + ", " + complaint_text + data = { + 'ids': self.ids, + 'model': self._name, + 'date_today': date_today, + 'date_request': self.date_request, + 'date_return': self.return_date, + 'sev_id': self.name, + 'warranty': self.is_in_warranty, + 'customer_name': self.person_name.name, + 'imei_no': self.imei_no, + 'technician': self.technician_name.name, + 'complaint_types': complaint_text, + 'complaint_description': description_text, + 'mobile_brand': self.brand_name.brand_name, + 'model_name': self.model_name.mobile_brand_models, + + } + return self.env.ref('mobile_service_shop.mobile_service_ticket').report_action(self, data=data) + + +class MobileBrand(models.Model): + _name = 'mobile.brand' + _rec_name = 'brand_name' + + brand_name = fields.Char(string="Mobile Brand", required=True) + + +class MobileComplaintType(models.Model): + _name = 'mobile.complaint' + _rec_name = 'complaint_type' + + complaint_type = fields.Char(string="Complaint Type", required=True) + + +class MobileComplaintTypeTemplate(models.Model): + _name = 'mobile.complaint.description' + _rec_name = 'description' + + complaint_type_template = fields.Many2one('mobile.complaint', string="Complaint Type Template", required=True) + description = fields.Text(string="Complaint Description") + + +class MobileComplaintTree(models.Model): + _name = 'mobile.complaint.tree' + _rec_name = 'complaint_type_tree' + + complaint_id = fields.Many2one('mobile.service') + + complaint_type_tree = fields.Many2one('mobile.complaint', string="Category", required=True) + description_tree = fields.Many2one('mobile.complaint.description', string="Description", + domain="[('complaint_type_template','=',complaint_type_tree)]") + + +class MobileBrandModels(models.Model): + _name = 'brand.model' + _rec_name = 'mobile_brand_models' + + mobile_brand_name = fields.Many2one('mobile.brand', string="Mobile Brand", required=True) + mobile_brand_models = fields.Char(string="Model Name", required=True) + image_medium = fields.Binary(string='image', store=True, attachment=True) + + +class MobileServiceTermsAndConditions(models.Model): + _name = 'terms.conditions' + _rec_name = 'terms_id' + + terms_id = fields.Char(String="Terms and condition", compute="_find_id") + terms_conditions = fields.Text(string="Terms and Conditions") + + def _find_id(self): + self.terms_id = self.id or '' + + diff --git a/mobile_service_shop/models/product.py b/mobile_service_shop/models/product.py new file mode 100644 index 000000000..ca57029fe --- /dev/null +++ b/mobile_service_shop/models/product.py @@ -0,0 +1,36 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU AFFERO +# GENERAL PUBLIC LICENSE (AGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# +from odoo import models, fields, api, _ + + +class ProductProduct(models.Model): + _inherit = 'product.template' + + is_a_parts = fields.Boolean( + 'Is a Mobile Part', default=False, + help="Specify if the product is a mobile part or not.") + + brand_name = fields.Many2one('mobile.brand', String="Brand", help="Select a mobile brand for the part") + model_name = fields.Many2one('brand.model', String="Model Name", domain="[('mobile_brand_name','=',brand_name)]", + help="Select a model for the part") + model_colour = fields.Char(string="Colour", help="colour for the part") + extra_descriptions = fields.Text(string="Note") \ No newline at end of file diff --git a/mobile_service_shop/models/product_order_line.py b/mobile_service_shop/models/product_order_line.py new file mode 100644 index 000000000..e17959f72 --- /dev/null +++ b/mobile_service_shop/models/product_order_line.py @@ -0,0 +1,92 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU AFFERO +# GENERAL PUBLIC LICENSE (AGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# +from odoo import models, fields, api, _ + + +class ProductOrderLine(models.Model): + + _name = 'product.order.line' + + product_order_id = fields.Many2one('mobile.service') + + product_id = fields.Many2one('product.product', string='Product', + domain="[('is_a_parts','=', True)]", required=True) + product_uom_qty = fields.Float(string='Used Quantity', default=1.0, required=True) + price_unit = fields.Float(string='Unit Price', default=0.0, required=True) + qty_invoiced = fields.Float(string='Invoiced qty', readonly=True) + qty_stock_move = fields.Float(string='Stock Move Posted Qty', readonly=True) + part_price = fields.Char(compute='_compute_amount', string='Price', readonly=True, store=True) + product_uom = fields.Char(string='Unit of Measure', required=True) + + @api.onchange('product_id') + def change_prod(self): + self.ensure_one() + if self.product_id: + product_template_obj = self.product_id.product_tmpl_id + self.price_unit = product_template_obj.list_price + self.product_uom = product_template_obj.uom_id.name + + @api.depends('product_uom_qty', 'product_id') + def _compute_amount(self): + """ + Compute the amount + """ + for line in self: + price = line.price_unit * line.product_uom_qty + + line.update({ + 'part_price': price, + }) + + def _create_stock_moves_transfer(self, picking): + moves = self.env['stock.move'] + done = self.env['stock.move'].browse() + if self.product_id.product_tmpl_id.type != 'service': + price_unit = self.price_unit + template = { + 'name': self.product_id.product_tmpl_id.name or '', + 'product_id': self.product_id.id, + 'product_uom': self.product_id.product_tmpl_id.uom_id.id, + 'location_id': picking.picking_type_id.default_location_src_id.id, + 'location_dest_id': self.product_order_id.person_name.property_stock_customer.id, + 'picking_id': picking.id, + 'move_dest_id': False, + 'state': 'draft', + 'company_id': self.product_order_id.company_id.id, + 'price_unit': price_unit, + 'picking_type_id': picking.picking_type_id.id, + 'procurement_id': False, + 'route_ids': 1 and [ + (6, 0, [x.id for x in self.env['stock.location.route'].search([('id', 'in', (2, 3))])])] or [], + 'warehouse_id': picking.picking_type_id.warehouse_id.id, + } + qty = self.product_uom_qty - self.qty_stock_move + print(qty) + diff_quantity = qty + tmp = template.copy() + tmp.update({ + 'product_uom_qty': diff_quantity, + }) + template['product_uom_qty'] = diff_quantity + done += moves.create(template) + self.qty_stock_move = self.qty_stock_move + qty + return done \ No newline at end of file diff --git a/mobile_service_shop/models/service_ticket.py b/mobile_service_shop/models/service_ticket.py new file mode 100755 index 000000000..e02817d8e --- /dev/null +++ b/mobile_service_shop/models/service_ticket.py @@ -0,0 +1,47 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU AFFERO +# GENERAL PUBLIC LICENSE (AGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# +from odoo import models, api + + +class StockMoveReport(models.AbstractModel): + + _name = 'report.mobile_service_shop.mobile_service_ticket_template' + + @api.model + def _get_report_values(self, docids, data): + terms = self.env['terms.conditions'].search([]) + return { + 'date_today': data['date_today'], + 'date_request': data['date_request'], + 'date_return': data['date_return'], + 'sev_id': data['sev_id'], + 'imei_no': data['imei_no'], + 'technician': data['technician'], + 'complaint_types': data['complaint_types'], + 'complaint_description': data['complaint_description'], + 'mobile_brand': data['mobile_brand'], + 'model_name': data['model_name'], + 'customer_name': data['customer_name'], + 'warranty': data['warranty'], + 'terms': terms, + } + diff --git a/mobile_service_shop/reports/mobile_service_ticket.xml b/mobile_service_shop/reports/mobile_service_ticket.xml new file mode 100644 index 000000000..e249cddd9 --- /dev/null +++ b/mobile_service_shop/reports/mobile_service_ticket.xml @@ -0,0 +1,12 @@ + + + + \ No newline at end of file diff --git a/mobile_service_shop/reports/service_ticket_template.xml b/mobile_service_shop/reports/service_ticket_template.xml new file mode 100644 index 000000000..71e0f56b0 --- /dev/null +++ b/mobile_service_shop/reports/service_ticket_template.xml @@ -0,0 +1,83 @@ + + + + + \ No newline at end of file diff --git a/mobile_service_shop/security/ir.model.access.csv b/mobile_service_shop/security/ir.model.access.csv new file mode 100644 index 000000000..ca693432e --- /dev/null +++ b/mobile_service_shop/security/ir.model.access.csv @@ -0,0 +1,15 @@ +id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink +access_mobile_service_shop_mobile_service_shop,access.mobile_service_shop.mobile_service_shop,model_mobile_service,mobile_service_group_executer,1,1,0,0 +access_mobile_service_shop_mobile_service_shop11,access.mobile_service_shop.mobile_service_shop11,model_mobile_complaint,mobile_service_group_executer,1,0,0,0 +access_mobile_service_shop_mobile_service_shop22,access.mobile_service_shop.mobile_service_shop22,model_mobile_service,mobile_service_group_manager,1,1,1,1 +access_mobile_service_shop_mobile_service_shop33,access.mobile_service_shop.mobile_service_shop33,model_mobile_brand,mobile_service_group_manager,1,1,1,1 +access_mobile_service_shop_mobile_service_shop44,access.mobile_service_shop.mobile_service_shop44,model_brand_model,mobile_service_group_manager,1,1,1,1 +access_mobile_service_shop_mobile_service_shop66,access.mobile_service_shop.mobile_service_shop66,model_mobile_complaint,mobile_service_group_manager,1,1,1,1 +access_mobile_service_shop_mobile_service_shop88,access.mobile_service_shop.mobile_service_shop88,model_product_template,mobile_service_group_manager,1,1,1,1 +access_mobile_service_shop_mobile_service_shop99,access.mobile_service_shop.mobile_service_shop99,model_mobile_complaint_tree,mobile_service_group_executer,1,1,1,0 +access_mobile_service_shop_mobile_service_shop111,access.mobile_service_shop.mobile_service_shop111,model_mobile_complaint_description,mobile_service_group_manager,1,1,1,1 +access_mobile_service_shop_mobile_service_shop112,access.mobile_service_shop.mobile_service_shop112,model_terms_conditions,mobile_service_group_manager,1,1,1,1 +access_mobile_service_shop_mobile_service_shop113,access.mobile_service_shop.mobile_service_shop113,model_mobile_complaint_tree,mobile_service_group_manager,1,1,1,1 +access_mobile_service_shop_mobile_service_shop114,access.mobile_service_shop.mobile_service_shop114,model_mobile_complaint_description,mobile_service_group_executer,1,1,0,0 +access_mobile_service_shop_mobile_service_shop115,access.mobile_service_shop.mobile_service_shop115,model_product_order_line,mobile_service_group_executer,1,1,1,1 +access_mobile_service_shop_mobile_service_shop116,access.mobile_service_shop.mobile_service_shop116,model_mobile_invoice,mobile_service_group_manager,1,1,1,1 diff --git a/mobile_service_shop/security/security.xml b/mobile_service_shop/security/security.xml new file mode 100644 index 000000000..c52e40263 --- /dev/null +++ b/mobile_service_shop/security/security.xml @@ -0,0 +1,36 @@ + + + + + Mobile Service + 5 + + + + Mobile Technician + + + + + + Manager + + + + + + + Mobile Service rule + + [('technician_name','=',user.id),('service_state','!=','draft'), ('company_id','=',user.company_id.id)] + + + + + Mobile Service Manager rule + + [('company_id','=',user.company_id.id)] + + + + \ No newline at end of file diff --git a/mobile_service_shop/static/description/banner.jpg b/mobile_service_shop/static/description/banner.jpg new file mode 100644 index 000000000..68b2a4eb4 Binary files /dev/null and b/mobile_service_shop/static/description/banner.jpg differ diff --git a/mobile_service_shop/static/description/icon.png b/mobile_service_shop/static/description/icon.png new file mode 100644 index 000000000..2725a213f Binary files /dev/null and b/mobile_service_shop/static/description/icon.png differ diff --git a/mobile_service_shop/static/description/index.html b/mobile_service_shop/static/description/index.html new file mode 100755 index 000000000..ec5c3dbbc --- /dev/null +++ b/mobile_service_shop/static/description/index.html @@ -0,0 +1,578 @@ + +
+
+

+ Mobile Service Management +

+

+ This module helps in managing mobile service shop daily activities. +

+
+ Cybrosys Technologies +
+ +
+ cybrosys technologies
+
+
+
+ +
+
+

+ Overview +

+

+ This module is exclusively designed for the mobile service shop for managing their daily activities + inside the service center. The module comes integrated with accounting and inventory modules maximizing + the efficiency. It comes embedded with features such as adding complaint templates, + making it a much user-friendly application. The module also helps in generating service tickets in the workplace. +

+
+
+

+ Access Rights +

+

+ + Manager :- Manager has the complete access over the mobile shop management.
+ + Mobile Technician :- Mobile Technician can read and write service requests and also update the parts usage. +

+
+
+ +
+
+

+ Features +

+

+ + Service request creation. +

+

+ + Assigning service request to technicians.

+ + Mobile service ticket generation. +

+

+ + Tracking the service status.

+

+ + Integrated with accounting and sales module.

+

+ + Multi-level access rights.

+

+ + Mobile complaint templates.

+

+ + Invoice for parts usage and service charges.

+

+ + Email notifications to customer.

+

+ + Parts inventory.

+

+ + Flexible for further customization.

+
+
+
+
+

+ Mobile Service Shop +

+
+

+ + When you install the module, an extra menu named Mobile Service is created. + Here you can see the service requests and thereafter configure your mobile service shop. +

+
+
+ +
+
+

Mobile Service -> Service Request -> Create

+

+ + This is the service request creation form. Service request can be created according to the device brand and model. + One can assign the technician, service request, provide with warranty details, customer details, IMEI no, request and return dates etc. +

+
+
+ +
+

+ Service Tickets +

+
+

Generating tickets for service request.

+

+ + Here you can create tickets for service requests by using the 'Print Ticket' button. +

+
+
+ +
+

+ + Service ticket will be generated as follows, +

+
+
+ +
+

+ Parts Inventory +

+
+

Mobile parts inventory

+

+ + Here you see the mobile parts used for service processing. +

+
+
+ +
+

+ + Product will be only visible in the parts inventory if 'is a mobile part' is checked. +
You can provide the brand, model and color details from here. +

+ +
+ +
+

+ Service Invoice +

+
+

Invoice for service requests

+

+ + Here you can create invoices for service requests. +

+
+
+ +
+

+ + Invoice can be created based on advance or full amount. +

+ +
+ +
+

+ + The parts used during the service process will be automatically added to the invoice line. +

+
+ +
+

+ Post Stock moves +

+
+

+ + No need to create stock moves for parts usage manually. 'Post Stock Moves' will generate stock for the parts usage. +

+
+ +
+

+ Email Notification +

+
+

+ + The system will send email notification to the customer for each stage in the service request.
+
Note:- You should configure outgoing and incoming e-mail settings from your Odoo for email service. +

+
+
+ +
+ +
+

+ Brands Configuration +

+

+ + Create mobile brands which is used in service request creation. +
Mobile Service -> Configuration -> Brands +

+
+ +
+
+

+ Models Configuration +

+

+ + You can also create models for different mobile brands from +
Mobile Service -> Configuration -> Models +

+
+ +
+

+ Complaint Types +

+

+ + You can create different complaint types from +
Mobile Service -> Configuration -> Complaint Types +

+
+ +
+

+ Complaint Templates +

+

+ + Complaint Templates can be created from +
Mobile Service -> Configuration -> Complaint Types +

+
+ +
+

+ + You can select different complaint templates from the 'Complaints' page while creating service requests. +

+
+ +
+

+ Terms and conditions +

+

+ + Terms and conditions for service request can be added from, +
Mobile Service -> Configuration -> Terms and Conditions + which will be added to the service tickets. +

+
+ +
+

+ Mobile Service Management Pro +

+
+

An extended version of the module is also available in odoo apps.

+
+ Features +
+

+ + Device details from IMEI number. +

+

+ + Customized pivot report. +

+

+ + Mobile service request report +

+

+ + Parts usage report. +

+

+ + Complaint Type report. +

+

+ Mobile Service Management Pro is an extended version of the module 'Mobile Service Management' + Using the module application, manager can seamlessly acquire the brand, model and manufacturing details of the device using the IMEI number. + In addition, it helps in saving your time and analyzing the service shop via varied generated reports. The reports include service report, + parts usage report, complaint type report and customized pivot report. +

You can download the pro version of this app from Mobile Service Management Pro +

+
+ +
+
+ +
+
+ cybrosys technologies +
+
+
+
+

+ Our Services +

+
+ + + +
+ +
+ + + +
+

+ + Odoo Support +

+ +
+ +
+
+
+
+
+

+ Our Industries +

+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Trading + +

+

+ Easily procure and sell your products. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Manufacturing +

+

+ Plan, track and schedule your operations. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Restaurant +

+

+ Run your bar or restaurant methodical. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + POS +

+

+ Easy configuring and convivial selling. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + E-commerce & Website +

+

+ Mobile friendly, awe-inspiring product pages. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Hotel Management +

+

+ An all-inclusive hotel management application. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Education +

+

+ A Collaborative platform for educational management. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Service Management +

+

+ Keep track of services and invoice accordingly. +

+
+
+
+
+
+
+ +
+ diff --git a/mobile_service_shop/static/description/mobile-service-shop1.png b/mobile_service_shop/static/description/mobile-service-shop1.png new file mode 100644 index 000000000..9e0437c55 Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop1.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop10.png b/mobile_service_shop/static/description/mobile-service-shop10.png new file mode 100644 index 000000000..e0931e968 Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop10.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop11.png b/mobile_service_shop/static/description/mobile-service-shop11.png new file mode 100644 index 000000000..c8fc40025 Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop11.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop12.png b/mobile_service_shop/static/description/mobile-service-shop12.png new file mode 100644 index 000000000..db42078d9 Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop12.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop13.png b/mobile_service_shop/static/description/mobile-service-shop13.png new file mode 100644 index 000000000..f608a88ae Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop13.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop14.png b/mobile_service_shop/static/description/mobile-service-shop14.png new file mode 100644 index 000000000..7f625f9ff Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop14.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop15.png b/mobile_service_shop/static/description/mobile-service-shop15.png new file mode 100644 index 000000000..5b7925f2b Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop15.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop16.png b/mobile_service_shop/static/description/mobile-service-shop16.png new file mode 100644 index 000000000..3319c237f Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop16.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop17.png b/mobile_service_shop/static/description/mobile-service-shop17.png new file mode 100644 index 000000000..17b3f75f5 Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop17.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop18.png b/mobile_service_shop/static/description/mobile-service-shop18.png new file mode 100644 index 000000000..6a36f7272 Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop18.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop2.png b/mobile_service_shop/static/description/mobile-service-shop2.png new file mode 100644 index 000000000..ceb39681e Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop2.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop3.png b/mobile_service_shop/static/description/mobile-service-shop3.png new file mode 100644 index 000000000..db12f7dea Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop3.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop4.png b/mobile_service_shop/static/description/mobile-service-shop4.png new file mode 100644 index 000000000..affa01676 Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop4.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop5.png b/mobile_service_shop/static/description/mobile-service-shop5.png new file mode 100644 index 000000000..1457b276b Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop5.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop6.png b/mobile_service_shop/static/description/mobile-service-shop6.png new file mode 100644 index 000000000..b85ce82fe Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop6.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop7.png b/mobile_service_shop/static/description/mobile-service-shop7.png new file mode 100644 index 000000000..7b745c855 Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop7.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop8.png b/mobile_service_shop/static/description/mobile-service-shop8.png new file mode 100644 index 000000000..dc66b979c Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop8.png differ diff --git a/mobile_service_shop/static/description/mobile-service-shop9.png b/mobile_service_shop/static/description/mobile-service-shop9.png new file mode 100644 index 000000000..8c3ed17ca Binary files /dev/null and b/mobile_service_shop/static/description/mobile-service-shop9.png differ diff --git a/mobile_service_shop/static/description/web-icon.png b/mobile_service_shop/static/description/web-icon.png new file mode 100644 index 000000000..e56b3cbc1 Binary files /dev/null and b/mobile_service_shop/static/description/web-icon.png differ diff --git a/mobile_service_shop/views/brand.xml b/mobile_service_shop/views/brand.xml new file mode 100644 index 000000000..0c1792256 --- /dev/null +++ b/mobile_service_shop/views/brand.xml @@ -0,0 +1,45 @@ + + + + + mobile.brand.form + mobile.brand + +
+ + + + + + + +
+
+
+ + + mobile.brand.tree + mobile.brand + + + + + + + + + Brand + mobile.brand + tree,form + + +

+ Click to Create a New Record. +

+
+
+ + +
+
\ No newline at end of file diff --git a/mobile_service_shop/views/brand_models.xml b/mobile_service_shop/views/brand_models.xml new file mode 100644 index 000000000..59ece8bf5 --- /dev/null +++ b/mobile_service_shop/views/brand_models.xml @@ -0,0 +1,82 @@ + + + + + mobile.brand.model.form + brand.model + +
+ + + + + + + + + +
+
+
+ + + mobile.service.kanban + brand.model + + + + + + + +
+
+ Product +
+
+ + + +
+
+
    +
  • model:
  • +
+
+
+ + + + + + + + + mobile.brand.model.tree + brand.model + + + + + + + + + + + + Models + brand.model + kanban,tree,form + +

+ Click to Create a New Record. +

+
+
+ + + + \ No newline at end of file diff --git a/mobile_service_shop/views/complaint_template.xml b/mobile_service_shop/views/complaint_template.xml new file mode 100644 index 000000000..22a5383b7 --- /dev/null +++ b/mobile_service_shop/views/complaint_template.xml @@ -0,0 +1,45 @@ + + + + + mobile.complaint.template.form + mobile.complaint.description + +
+ + + + + + +
+
+
+ + + mobile.complaint.type.tree + mobile.complaint.description + + + + + + + + + + Complaint Types + mobile.complaint.description + tree,form + + +

+ Click to Create a New Record. +

+
+
+ + +
+
\ No newline at end of file diff --git a/mobile_service_shop/views/complaint_type.xml b/mobile_service_shop/views/complaint_type.xml new file mode 100644 index 000000000..1a3815bf1 --- /dev/null +++ b/mobile_service_shop/views/complaint_type.xml @@ -0,0 +1,47 @@ + + + + + mobile.complaint.type.form + mobile.complaint + +
+ + + + + + + +
+
+
+ + + mobile.complaint.type.tree + mobile.complaint + + + + + + + + + Complaint Types + mobile.complaint + tree,form + + +

+ Click to Create a New Record. +

+
+ +
+ + +
+
\ No newline at end of file diff --git a/mobile_service_shop/views/mobile_service_views.xml b/mobile_service_shop/views/mobile_service_views.xml new file mode 100644 index 000000000..63357ad66 --- /dev/null +++ b/mobile_service_shop/views/mobile_service_views.xml @@ -0,0 +1,306 @@ + + + + + + Service Code + mobile.service + SERV/ + + + + + + + service.request.search1 + mobile.service + + + + + + + + + + + + + + + + + + + + + + + + mobile.service.request.form + mobile.service + +
+
+
+ +
+ +
+
+

+ +

+
+ +
+ +
+
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
+ + + + +
+ +
+
+
+ + +
+ +
+
+ + + mobile.service.kanban + mobile.service + + + + + + +
+
+ +
+
+
+ + + +
+
+ +
+ +
+
+
+
+
+
+
+
+
+ + + mobile.service.graph + mobile.service + + + + + + + + + mobile.service.request.tree + mobile.service + + + + + + + + + + + + + + + + + + Service Request + mobile.service + tree,form,kanban,pivot,graph + + + +

+ Click to Create a New Record. +

+
+
+ + + + + + + + + + + +
+
\ No newline at end of file diff --git a/mobile_service_shop/views/product_template.xml b/mobile_service_shop/views/product_template.xml new file mode 100644 index 000000000..1250e0b2b --- /dev/null +++ b/mobile_service_shop/views/product_template.xml @@ -0,0 +1,68 @@ + + + + + + product.parts.form.view + product.template + + + + + + + + + + + + + + + + + + + + + + + + + + product.template.product.kanban + product.template + + + + + + + + + + + + + + Products + ir.actions.act_window + product.template + kanban,tree,form + {"search_default_consumable":1, 'default_type': 'product'} + [('is_a_parts', '=', True)] + +

+ Click to define a new product. +

+
+
+ + + + +
+
\ No newline at end of file diff --git a/mobile_service_shop/views/terms_and_condition.xml b/mobile_service_shop/views/terms_and_condition.xml new file mode 100644 index 000000000..59c42d9b4 --- /dev/null +++ b/mobile_service_shop/views/terms_and_condition.xml @@ -0,0 +1,44 @@ + + + + + mobile.terms.and.conditions.form + terms.conditions + +
+ + + + + +
+
+
+ + + mobile.terms.and.conditions.tree + terms.conditions + + + + + + + + + Terms and Conditions + terms.conditions + tree,form + + +

+ Click to Create a New Record. +

+
+
+ + +
+
\ No newline at end of file diff --git a/mobile_service_shop/wizard/__init__.py b/mobile_service_shop/wizard/__init__.py new file mode 100644 index 000000000..b42d7c3d2 --- /dev/null +++ b/mobile_service_shop/wizard/__init__.py @@ -0,0 +1,23 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU AFFERO +# GENERAL PUBLIC LICENSE (AGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# +from . import mobile_create_invoice + diff --git a/mobile_service_shop/wizard/mobile_create_invoice.py b/mobile_service_shop/wizard/mobile_create_invoice.py new file mode 100644 index 000000000..a89ee39fd --- /dev/null +++ b/mobile_service_shop/wizard/mobile_create_invoice.py @@ -0,0 +1,169 @@ +# -*- coding: utf-8 -*- +############################################################################# +# +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2019-TODAY Cybrosys Technologies(). +# Author: Milind Mohan @ Cybrosys, (odoo@cybrosys.com) +# Mohammed Shahil M P @ Cybrosys, (odoo@cybrosys.com) +# You can modify it under the terms of the GNU AFFERO +# GENERAL PUBLIC LICENSE (AGPL v3), Version 3. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. +# +# You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE +# (AGPL v3) along with this program. +# If not, see . +# +############################################################################# +from odoo import api, fields, models, _ +from odoo.exceptions import Warning, UserError +from odoo.exceptions import UserError, ValidationError + + +class MobileServiceInvoice(models.Model): + + _name = 'mobile.invoice' + + advance_payment_method = fields.Selection([('advance', 'Advance'), ('full_amount', 'Full amount')], + string='Invoice method', default='advance') + amount = fields.Integer(string='Amount') + number = fields.Char(string='Service Id') + + @api.model + def create(self, vals): + print(vals) + if 'amount' in vals: + vals['name'] = self.env['ir.sequence'].next_by_code('account.payment.customer.invoice') or _('New') + else: + vals['name'] = self.env['ir.sequence'].next_by_code('account.payment.customer.invoice') or _('New') + vals['service_state'] = 'draft' + return super(MobileServiceInvoice, self).create(vals) + + def action_invoice_create(self): + active_id = self._context.get('active_id') + service_id = self.env['mobile.service'].search([('id', '=', active_id)]) + if not service_id.env['product.product'].search([("name", "=", "Mobile Service Advance")]): + vals = self._prepare_advance_product() + self.env['product.product'].create(vals) + + if not service_id.env['product.product'].search([("name", "=", "Mobile Service Charge")]): + vals1 = self._prepare_service_product() + self.env['product.product'].create(vals1) + + service_id.first_invoice_created = True + inv_obj = self.env['account.move'] + inv_line_obj = self.env['account.move.line'] + supplier = service_id.person_name + inv_data = { + 'type': 'out_invoice', + 'reference': supplier.name, + 'account_id': supplier.property_account_receivable_id.id, + 'partner_id': supplier.id, + 'currency_id': service_id.company_id.currency_id.id, + 'journal_id': service_id.journal_type.id, + 'invoice_origin': service_id.name, + 'company_id': service_id.company_id.id, + 'date_due': service_id.return_date, + } + inv_id = inv_obj.create(inv_data) + service_id.first_payment_inv = inv_id.id + self.number = service_id.name + if self.advance_payment_method != 'advance': + product_id = service_id.env['product.product'].search([("name", "=", "Mobile Service Charge")]) + else: + product_id = service_id.env['product.product'].search([("name", "=", "Mobile Service Advance")]) + + if product_id.property_account_income_id.id: + income_account = product_id.property_account_income_id.id + elif product_id.categ_id.property_account_income_categ_id.id: + income_account = product_id.categ_id.property_account_income_categ_id.id + else: + raise UserError('Please define income account for this product: "%s" (id:%d).' % + (product_id.name, product_id.id)) + flag = 0 + if self.amount: + flag = 1 + inv_line_data = [(0, 0, { + 'name': product_id.name, + 'price_unit': self.amount, + 'quantity': 1, + 'credit': self.amount, + 'debit': 0, + 'account_id': income_account, + 'product_id': product_id.id, + 'move_id': inv_id.id, + })] + inv_id.write({ + 'invoice_line_ids': inv_line_data}) + inv_id._recompute_payment_terms_lines() + + sale_order_product = self.env['product.order.line'].search([('product_order_id', '=', service_id.name)]) + for line_data in sale_order_product: + qty = line_data.product_uom_qty - line_data.qty_invoiced + if line_data.product_uom_qty < line_data.qty_invoiced: + raise UserError(_('Used quantity is less than invoiced quantity')) + uom_id = line_data.product_id.product_tmpl_id.uom_id + if qty > 0: + flag = 1 + price = line_data.product_id.list_price + inv_line_data = [(0, 0, { + 'name': line_data.product_id.name, + 'price_unit': price, + 'quantity': qty, + 'product_uom_id': uom_id.id, + 'credit': price, + 'debit': 0, + 'account_id': income_account, + 'product_id': line_data.product_id.id, + 'move_id': inv_id.id, + })] + inv_id.write({ + 'invoice_line_ids': inv_line_data}) + line_data.qty_invoiced = line_data.qty_invoiced + qty + inv_id._recompute_payment_terms_lines() + + inv_id.post() + if flag != 1: + raise UserError(_('Nothing to create invoice')) + imd = service_id.env['ir.model.data'] + action = imd.xmlid_to_object('account.action_move_out_invoice_type') + list_view_id = imd.xmlid_to_res_id('account.view_move_tree') + form_view_id = imd.xmlid_to_res_id('account.view_move_form') + result = { + 'name': action.name, + 'help': action.help, + 'type': 'ir.actions.act_window', + 'views': [[list_view_id, 'tree'], [form_view_id, 'form'], [False, 'graph'], [False, 'kanban'], + [False, 'calendar'], [False, 'pivot']], + 'target': action.target, + 'context': action.context, + 'res_model': 'account.move', + } + if len(inv_id) > 1: + result['domain'] = "[('id','in',%s)]" % inv_id.ids + elif len(inv_id) == 1: + result['views'] = [(form_view_id, 'form')] + result['res_id'] = inv_id.ids[0] + else: + result = {'type': 'ir.actions.act_window_close'} + return result + + def _prepare_advance_product(self): + return { + 'name': 'Mobile Service Advance', + 'type': 'service', + 'invoice_policy': 'order', + 'company_id': False, + } + + def _prepare_service_product(self): + return { + 'name': 'Mobile Service Charge', + 'type': 'service', + 'invoice_policy': 'order', + 'company_id': False, + } diff --git a/mobile_service_shop/wizard/mobile_create_invoice_views.xml b/mobile_service_shop/wizard/mobile_create_invoice_views.xml new file mode 100644 index 000000000..23b033929 --- /dev/null +++ b/mobile_service_shop/wizard/mobile_create_invoice_views.xml @@ -0,0 +1,38 @@ + + + Create Invoice + mobile.invoice + +
+

+ Invoices will be created in draft so that you can review + them before validation. +

+ + + +
+
+
+
+
+ + + + Create Invoice + ir.actions.act_window + mobile.invoice + form + new + + + +
diff --git a/task_deadline_reminder/README.rst b/task_deadline_reminder/README.rst new file mode 100644 index 000000000..c0e852b72 --- /dev/null +++ b/task_deadline_reminder/README.rst @@ -0,0 +1,59 @@ +Task DeadLine Reminder v13 +========================== +This module extends the functionality of project module to allow to send deadline reminder emails on task deadline day. + +Configuration +============= + +By default, a cron job named "Task DeadLine Reminder" will be created while installing this module. +This cron job can be found in: + + **Settings > Technical > Automation > Scheduled Actions** + +This job runs daily by default. + +Usage +===== + +To use this functionality, you need to: + +#. Create a project to which the new tasks will be related. +#. Add a name, deadline date, who the task will be assigned to, etc... +#. In order to send email reminder to responsible user,you have to set reminder box (Project > Task > Reminder ) +#. Go to the Scheduled Action(Settings > Technical > Automation > Scheduled Action) and edit the time at which Deadline Reminder Action is to be done + +The cron job will do the rest. + +Installation +============ +- www.odoo.com/documentation/11.0/setup/install.html +- Install our custom addon + +Bug Tracker +=========== +Bugs are tracked on GitHub Issues. In case of trouble, please check there if your issue has already been reported. + +Credits +======= +Cybrosys Techno Solutions + +Author +------ +* Developer v9: Saritha @ cybrosys +* Developer v10, v11: Niyas Raphy @ cybrosys +* V13: Nimisha + +Maintainer +---------- + +This module is maintained by Cybrosys Technologies. + +For support and more information, please visit https://www.cybrosys.com. + + + + + + + + diff --git a/task_deadline_reminder/__init__.py b/task_deadline_reminder/__init__.py new file mode 100644 index 000000000..167c5926c --- /dev/null +++ b/task_deadline_reminder/__init__.py @@ -0,0 +1,23 @@ +# -*- coding: utf-8 -*- +################################################################################### +# Cybrosys Technologies Pvt. Ltd. +# Copyright (C) 2018-TODAY Cybrosys Technologies ().# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU Affero General Public License for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################### + +from . import models + + + diff --git a/task_deadline_reminder/__manifest__.py b/task_deadline_reminder/__manifest__.py new file mode 100755 index 000000000..1fa4b60cc --- /dev/null +++ b/task_deadline_reminder/__manifest__.py @@ -0,0 +1,41 @@ +# -*- coding: utf-8 -*- +################################################################################### +# Cybrosys Technologies Pvt. Ltd. +# +# Copyright (C) 2018-TODAY Cybrosys Technologies ().# +# This program is free software: you can modify +# it under the terms of the GNU Affero General Public License (AGPL) as +# published by the Free Software Foundation, either version 3 of the +# License, or (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU Affero General Public License for more details. +# +# You should have received a copy of the GNU Affero General Public License +# along with this program. If not, see . +# +################################################################################### + +{ + 'name': "Task Deadline Reminder", + 'version': "13.0.1.0.0", + 'author': 'Cybrosys Techno Solutions', + 'company': 'Cybrosys Techno Solutions', + 'maintainer': 'Cybrosys Techno Solutions', + 'website': 'https://www.cybrosys.com', + 'summary': '''Automatically Send Mail To Responsible User if Deadline Of Task is Today''', + 'description': '''Automatically Send Mail To Responsible User if Deadline Of Task is Today''', + 'category': "Project", + 'depends': ['project'], + 'license': 'AGPL-3', + 'data': [ + 'views/deadline_reminder_view.xml', + 'views/deadline_reminder_cron.xml', + 'data/deadline_reminder_action_data.xml' + ], + 'images': ['static/description/banner.jpg'], + 'installable': True, + 'auto_install': False +} diff --git a/task_deadline_reminder/data/deadline_reminder_action_data.xml b/task_deadline_reminder/data/deadline_reminder_action_data.xml new file mode 100644 index 000000000..37358154d --- /dev/null +++ b/task_deadline_reminder/data/deadline_reminder_action_data.xml @@ -0,0 +1,46 @@ + + + + + + Deadline Reminder...!! + ${object.company_id.name}<${object.company_id.email}> + ${object.user_id.email} + Today Due Task -${object.date_deadline or 'n/a' } + + + +
+

Hello ${object.user_id.name},

+

This Email Is To Remind You That You Have Task As Below Listed Which Are Due On Today.

+
+
+ + + + + + + + + + + + + + + + + + + +
TaskProjectDeadlineAssigned ToLink
${object.name}${object.project_id.name}${object.date_deadline}${object.user_id.name}View Now
+
+
+ ]]> + + + + + diff --git a/task_deadline_reminder/doc/RELEASE_NOTES.md b/task_deadline_reminder/doc/RELEASE_NOTES.md new file mode 100644 index 000000000..8a013932c --- /dev/null +++ b/task_deadline_reminder/doc/RELEASE_NOTES.md @@ -0,0 +1,6 @@ +## Module + +#### 16.11.2019 +#### Version 13.0.1.0.0 +#### ADD +Initial Commit diff --git a/task_deadline_reminder/models/__init__.py b/task_deadline_reminder/models/__init__.py new file mode 100644 index 000000000..804dd48f7 --- /dev/null +++ b/task_deadline_reminder/models/__init__.py @@ -0,0 +1,5 @@ +# -*- coding: utf-8 -*- + +from . import deadline_reminder + + diff --git a/task_deadline_reminder/models/deadline_reminder.py b/task_deadline_reminder/models/deadline_reminder.py new file mode 100644 index 000000000..133ef283c --- /dev/null +++ b/task_deadline_reminder/models/deadline_reminder.py @@ -0,0 +1,40 @@ +# -*- coding: utf-8 -*- + +import datetime +from datetime import datetime +from odoo import SUPERUSER_ID +from odoo import api, fields, models, _ + + +class DeadLineReminder(models.Model): + _inherit = "project.task" + + task_reminder = fields.Boolean("Reminder") + + @api.model + def _cron_deadline_reminder(self): + print("test") + su_id = self.env['res.partner'].browse(SUPERUSER_ID) + print(su_id) + for task in self.env['project.task'].search([('date_deadline', '!=', None), + ('task_reminder', '=', True), ('user_id', '!=', None)]): + print(task, "task") + reminder_date = task.date_deadline + today = datetime.now().date() + if reminder_date == today and task: + # print("kkkkkkkkkkkkkkk") + template_id = self.env['ir.model.data'].get_object_reference( + 'task_deadline_reminder', + 'email_template_edi_deadline_reminder')[1] + if template_id: + # print("template_id", template_id) + email_template_obj = self.env['mail.template'].browse(template_id) + # print("email_template_obj", email_template_obj) + values = email_template_obj.generate_email(task.id, fields=None) + msg_id = self.env['mail.mail'].create(values) + if msg_id: + msg_id._send() + + return True + + diff --git a/task_deadline_reminder/static/description/banner.jpg b/task_deadline_reminder/static/description/banner.jpg new file mode 100644 index 000000000..998679818 Binary files /dev/null and b/task_deadline_reminder/static/description/banner.jpg differ diff --git a/task_deadline_reminder/static/description/cybro_logo.png b/task_deadline_reminder/static/description/cybro_logo.png new file mode 100644 index 000000000..bb309114c Binary files /dev/null and b/task_deadline_reminder/static/description/cybro_logo.png differ diff --git a/task_deadline_reminder/static/description/cybrosys-task-reminder-1.jpg b/task_deadline_reminder/static/description/cybrosys-task-reminder-1.jpg new file mode 100644 index 000000000..5ab2a2ed0 Binary files /dev/null and b/task_deadline_reminder/static/description/cybrosys-task-reminder-1.jpg differ diff --git a/task_deadline_reminder/static/description/cybrosys-task-reminder-2.png b/task_deadline_reminder/static/description/cybrosys-task-reminder-2.png new file mode 100644 index 000000000..c7eba88c4 Binary files /dev/null and b/task_deadline_reminder/static/description/cybrosys-task-reminder-2.png differ diff --git a/task_deadline_reminder/static/description/cybrosys-task-reminder-3.png b/task_deadline_reminder/static/description/cybrosys-task-reminder-3.png new file mode 100644 index 000000000..7555a00ac Binary files /dev/null and b/task_deadline_reminder/static/description/cybrosys-task-reminder-3.png differ diff --git a/task_deadline_reminder/static/description/icon.png b/task_deadline_reminder/static/description/icon.png new file mode 100644 index 000000000..1ff90030a Binary files /dev/null and b/task_deadline_reminder/static/description/icon.png differ diff --git a/task_deadline_reminder/static/description/index.html b/task_deadline_reminder/static/description/index.html new file mode 100644 index 000000000..4c8038f3b --- /dev/null +++ b/task_deadline_reminder/static/description/index.html @@ -0,0 +1,353 @@ +
+
+

+ Task DeadLine Reminder +

+

+ Helps to remind the deadline of various tasks. +

+
+ Cybrosys Technologies +
+ +
+ cybrosys technologies +
+
+
+
+ +
+
+

+ Overview +

+

+ This module send auto reminder to responsible user of task if deadline = Today. Cron job will + run everyday and search for task which due today and send reminder email to employee. +

+
+
+
+
+

+ Features +

+

+ + Helps the HR manager to know the deadline of various tasks. +

+

+ + HR can send reminder mail to the particular employee whose tasks is going to end today +

+

+ + HR can take actions and further processes based on the deadline +

+
+
+
+
+

+ Screenshots +

+

+ + Project Task Form - Configuration of Task Deadline Reminder +

+
    +
  • + If set this box then only this task will be consider for reminder. +
  • +
+
+ +
+

+ + Automation Scheduled Action - Cron Job +

+
+ +
+

+ + Now link will allow user to jump to related task directly. +

+

+ + This email will group all tasks which are deadline today for that user and send summary table to user/employee by email. +

+
+ +
+
+
+ + +
+
+ cybrosys technologies +
+
+
+
+

+ Our Services +

+
+ + + +
+ +
+ + + +
+

+ + Odoo Support +

+ +
+ +
+
+
+
+
+

+ Our Industries +

+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Trading + +

+

+ Easily procure and sell your products. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Manufacturing +

+

+ Plan, track and schedule your operations. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Restaurant +

+

+ Run your bar or restaurant methodical. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + POS +

+

+ Easy configuring and convivial selling. +

+
+ +
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + E-commerce & Website +

+

+ Mobile friendly, awe-inspiring product pages. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Hotel Management +

+

+ An all-inclusive hotel management application. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Education +

+

+ A Collaborative platform for educational management. +

+
+
+
+ +
+
+ + Odoo Industry + +
+
+
+

+ + Service Management +

+

+ Keep track of services and invoice accordingly. +

+
+
+
+
+
+
+ +
diff --git a/task_deadline_reminder/static/description/mail.png b/task_deadline_reminder/static/description/mail.png new file mode 100644 index 000000000..7555a00ac Binary files /dev/null and b/task_deadline_reminder/static/description/mail.png differ diff --git a/task_deadline_reminder/static/description/project_task_form.jpg b/task_deadline_reminder/static/description/project_task_form.jpg new file mode 100644 index 000000000..5ab2a2ed0 Binary files /dev/null and b/task_deadline_reminder/static/description/project_task_form.jpg differ diff --git a/task_deadline_reminder/static/description/scheduled_action_form.png b/task_deadline_reminder/static/description/scheduled_action_form.png new file mode 100644 index 000000000..c7eba88c4 Binary files /dev/null and b/task_deadline_reminder/static/description/scheduled_action_form.png differ diff --git a/task_deadline_reminder/views/deadline_reminder_cron.xml b/task_deadline_reminder/views/deadline_reminder_cron.xml new file mode 100644 index 000000000..ede641ed2 --- /dev/null +++ b/task_deadline_reminder/views/deadline_reminder_cron.xml @@ -0,0 +1,16 @@ + + + + + Task DeadLine Reminder + + code + model._cron_deadline_reminder() + + 1 + days + -1 + + + + diff --git a/task_deadline_reminder/views/deadline_reminder_view.xml b/task_deadline_reminder/views/deadline_reminder_view.xml new file mode 100644 index 000000000..3c150738e --- /dev/null +++ b/task_deadline_reminder/views/deadline_reminder_view.xml @@ -0,0 +1,17 @@ + + + + + + ProjectTaskRemainder + project.task + + + + + + + + + + \ No newline at end of file