@ -0,0 +1,55 @@ |
|||||
|
.. image:: https://img.shields.io/badge/license-AGPL--3-blue.svg |
||||
|
:target: https://www.gnu.org/licenses/agpl-3.0-standalone.html |
||||
|
:alt: License: AGPL-3 |
||||
|
|
||||
|
Multi Product Return From Website |
||||
|
====================== |
||||
|
This module allows to create and manage sale order return from website. |
||||
|
The logged user can create multiple product return from website. |
||||
|
The Authorize persons can manage the return orders. |
||||
|
And allow the portal users can view their return orders |
||||
|
from the account portal view. |
||||
|
|
||||
|
Configuration |
||||
|
============= |
||||
|
No additional configuration required |
||||
|
|
||||
|
Company |
||||
|
------- |
||||
|
* `Cybrosys Techno Solutions <https://cybrosys.com/>`__ |
||||
|
|
||||
|
License |
||||
|
------- |
||||
|
GNU Affero General Public License v3.0 (AGPL v3) |
||||
|
(http://www.gnu.org/licenses/agpl-3.0-standalone.html) |
||||
|
|
||||
|
Credits |
||||
|
------- |
||||
|
Developer: (V15) Bhagyadev, |
||||
|
(V16) Bhagyadev, |
||||
|
(V17) Ranjith R, |
||||
|
(V18) Safa Faheem PE, |
||||
|
Contact: odoo@cybrosys.com |
||||
|
|
||||
|
Contacts |
||||
|
-------- |
||||
|
* Mail Contact : odoo@cybrosys.com |
||||
|
* Website : https://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 <https://cybrosys.com/>`__ |
||||
|
|
||||
|
Further information |
||||
|
=================== |
||||
|
HTML Description: `<static/description/index.html>`__ |
@ -0,0 +1,24 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
from . import controllers |
||||
|
from . import models |
||||
|
from . import wizard |
@ -0,0 +1,57 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
{ |
||||
|
'name': 'Multi Product Return From Website', |
||||
|
'version': '18.0.1.0.0', |
||||
|
'category': 'Website', |
||||
|
'summary': 'Sale order multi product return management from website', |
||||
|
'description': "Streamline your website with advanced Multi-product Return " |
||||
|
"Order Management. Easily manage returns, RMA, and order " |
||||
|
"return processes directly from your website.", |
||||
|
'author': 'Cybrosys Techno Solutions', |
||||
|
'company': 'Cybrosys Techno Solutions', |
||||
|
'maintainer': 'Cybrosys Techno Solutions', |
||||
|
'website': 'https://www.cybrosys.com', |
||||
|
'depends': ['website_sale', 'stock', 'sale_management'], |
||||
|
'data': [ |
||||
|
'security/ir.model.access.csv', |
||||
|
'data/ir_sequence_data.xml', |
||||
|
'views/website_thankyou_template.xml', |
||||
|
'views/sale_order_portal_templates.xml', |
||||
|
'views/sale_return_views.xml', |
||||
|
'views/sale_order_views.xml', |
||||
|
'views/res_partner_views.xml', |
||||
|
'views/stock_picking_views.xml', |
||||
|
'report/sale_return_templates.xml', |
||||
|
'report/sale_return_reports.xml', |
||||
|
], |
||||
|
'assets': { |
||||
|
'web.assets_frontend': [ |
||||
|
'website_multi_product_return_management/static/src/js/sale_return.js', |
||||
|
], |
||||
|
}, |
||||
|
'images': ['static/description/banner.jpg'], |
||||
|
'license': 'AGPL-3', |
||||
|
'installable': True, |
||||
|
'auto_install': False, |
||||
|
'application': False, |
||||
|
} |
@ -0,0 +1,23 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
from . import website_multi_product_return_management |
||||
|
from . import portal |
@ -0,0 +1,158 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
import base64 |
||||
|
from collections import OrderedDict |
||||
|
from odoo import http |
||||
|
from odoo.exceptions import AccessError, MissingError |
||||
|
from odoo.http import request |
||||
|
from odoo.tools import image_process |
||||
|
from odoo.tools.translate import _ |
||||
|
from odoo.addons.portal.controllers.portal import CustomerPortal |
||||
|
from odoo.addons.web.controllers.binary import Binary |
||||
|
|
||||
|
|
||||
|
class ReturnCustomerPortal(CustomerPortal): |
||||
|
"""Customized Customer Portal for managing sale returns.""" |
||||
|
|
||||
|
def _prepare_home_portal_values(self, counters): |
||||
|
"""Prepare values for the home portal.""" |
||||
|
values = super()._prepare_home_portal_values(counters) |
||||
|
partner = request.env.user.partner_id |
||||
|
if 'return_count' in counters: |
||||
|
if request.env.user.has_group('base.group_portal'): |
||||
|
values['return_count'] = request.env[ |
||||
|
'sale.return'].search_count( |
||||
|
[('state', 'in', ['draft', 'confirm', 'done', 'cancel']), |
||||
|
('partner_id', '=', partner.id)]) |
||||
|
elif request.env.user.has_group('base.group_system'): |
||||
|
values['return_count'] = request.env[ |
||||
|
'sale.return'].search_count( |
||||
|
[('state', 'in', ['draft', 'confirm', 'done', 'cancel'])]) |
||||
|
else: |
||||
|
values['return_count'] = request.env[ |
||||
|
'sale.return'].search_count( |
||||
|
['|', ('partner_id', '=', partner.id), |
||||
|
('user_id', '=', request.env.user.id)]) |
||||
|
return values |
||||
|
|
||||
|
@http.route(['/my/return_orders', '/my/return_orders/page/<int:page>'], |
||||
|
type='http', auth="user", website=True) |
||||
|
def portal_my_sale_return(self, page=1, date_begin=None, date_end=None, |
||||
|
sortby=None, filterby=None, **kw): |
||||
|
"""Handle requests for displaying sale return orders in the portal.""" |
||||
|
values = self._prepare_portal_layout_values() |
||||
|
sale_return = request.env['sale.return'] |
||||
|
partner = request.env.user.partner_id |
||||
|
if request.env.user.has_group('base.group_portal'): |
||||
|
domain = [ |
||||
|
('partner_id', '=', partner.id) |
||||
|
] |
||||
|
elif request.env.user.has_group('base.group_system'): |
||||
|
domain = [] |
||||
|
else: |
||||
|
domain = ['|', ('partner_id', '=', partner.id), |
||||
|
('user_id', '=', request.env.user.id)] |
||||
|
searchbar_sortings = { |
||||
|
'date': {'label': _('Newest'), 'order': 'create_date desc'}, |
||||
|
'name': {'label': _('Name'), 'order': 'name'}, |
||||
|
'sale': {'label': _('Sale Order'), 'order': 'sale_order'}, |
||||
|
} |
||||
|
if not sortby: |
||||
|
sortby = 'date' |
||||
|
order = searchbar_sortings[sortby]['order'] |
||||
|
if date_begin and date_end: |
||||
|
domain += [('create_date', '>', date_begin), |
||||
|
('create_date', '<=', date_end)] |
||||
|
searchbar_filters = { |
||||
|
'all': {'label': _('All'), 'domain': [ |
||||
|
('state', 'in', ['draft', 'confirm', 'done', 'cancel'])]}, |
||||
|
'confirm': {'label': _('Confirmed'), |
||||
|
'domain': [('state', '=', 'confirm')]}, |
||||
|
'cancel': {'label': _('Cancelled'), |
||||
|
'domain': [('state', '=', 'cancel')]}, |
||||
|
'done': {'label': _('Done'), 'domain': [('state', '=', 'done')]}, |
||||
|
} |
||||
|
if not filterby: |
||||
|
filterby = 'all' |
||||
|
domain += searchbar_filters[filterby]['domain'] |
||||
|
return_count = sale_return.sudo().search_count(domain) |
||||
|
pager = request.website.pager( |
||||
|
url="/my/return_orders", |
||||
|
url_args={'date_begin': date_begin, 'date_end': date_end, |
||||
|
'sortby': sortby}, |
||||
|
total=return_count, |
||||
|
page=page, |
||||
|
step=self._items_per_page |
||||
|
) |
||||
|
orders = sale_return.sudo().search(domain, order=order, |
||||
|
limit=self._items_per_page, |
||||
|
offset=pager['offset']) |
||||
|
request.session['my_return_history'] = orders.ids[:100] |
||||
|
values.update({ |
||||
|
'date': date_begin, |
||||
|
'orders': orders.sudo(), |
||||
|
'page_name': 'Sale_Return', |
||||
|
'default_url': '/my/return_orders', |
||||
|
'pager': pager, |
||||
|
'searchbar_filters': OrderedDict(sorted(searchbar_filters.items())), |
||||
|
'searchbar_sortings': searchbar_sortings, |
||||
|
'sortby': sortby, |
||||
|
}) |
||||
|
return request.render( |
||||
|
"website_multi_product_return_management.portal_my_returns", values) |
||||
|
|
||||
|
@http.route(['/my/return_orders/<int:order_id>'], type='http', |
||||
|
auth="public", website=True) |
||||
|
def portal_my_return_detail(self, order_id=None, access_token=None, |
||||
|
report_type=None, download=False, **kw): |
||||
|
"""Handle requests for displaying details of a specific sale |
||||
|
return order.""" |
||||
|
try: |
||||
|
order_sudo = self._document_check_access('sale.return', order_id, |
||||
|
access_token) |
||||
|
except (AccessError, MissingError): |
||||
|
return request.redirect('/my') |
||||
|
if report_type in ('html', 'pdf', 'text'): |
||||
|
return self._show_report( |
||||
|
model=order_sudo, report_type=report_type, |
||||
|
report_ref='website_multi_product_return_management.' |
||||
|
'report_sale_returns', |
||||
|
download=download) |
||||
|
values = self._sale_return_get_page_view_values(order_sudo, |
||||
|
access_token, **kw) |
||||
|
return request.render( |
||||
|
"website_multi_product_return_management.portal_sale_return_page", |
||||
|
values) |
||||
|
|
||||
|
def _sale_return_get_page_view_values(self, order, access_token, **kwargs): |
||||
|
"""Get values for rendering the sale return page view.""" |
||||
|
def resize_to_48(b64source): |
||||
|
"""Resizing image as needed""" |
||||
|
if not b64source: |
||||
|
b64source = base64.b64encode(Binary.placeholder()) |
||||
|
return image_process(b64source, size=(48, 48)) |
||||
|
values = { |
||||
|
'orders': order, |
||||
|
'resize_to_48': resize_to_48, |
||||
|
} |
||||
|
return self._get_page_view_values(order, access_token, values, |
||||
|
'my_return_history', False, **kwargs) |
@ -0,0 +1,74 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
from datetime import datetime |
||||
|
from odoo import http |
||||
|
from odoo.http import request |
||||
|
|
||||
|
|
||||
|
class CustomerRegistration(http.Controller): |
||||
|
@http.route('/sale_return', type='json', csrf=False, auth="public", |
||||
|
website=True) |
||||
|
def sale_return(self, **kwargs, ): |
||||
|
"""Controller to create return order""" |
||||
|
if kwargs.get('vals'): |
||||
|
lines = [] |
||||
|
so_id = False |
||||
|
for line in kwargs.get('vals'): |
||||
|
so_id = request.env['sale.order'].sudo().browse(int(line['order_id'])) |
||||
|
qty = int(line['quantity']) |
||||
|
d_qty = int(line['deli_qty']) |
||||
|
reason = line['reason'] |
||||
|
if qty > 0: |
||||
|
lines.append((0, 0, {'product_id': line['product_id'], |
||||
|
'received_qty': qty, |
||||
|
'quantity': d_qty, |
||||
|
'reason': reason |
||||
|
})) |
||||
|
if so_id: |
||||
|
values = { |
||||
|
'partner_id': so_id.partner_id.id, |
||||
|
'sale_order_id': so_id.id, |
||||
|
'user_id': request.env.uid, |
||||
|
'create_date': datetime.now(), |
||||
|
'return_line_ids': lines |
||||
|
} |
||||
|
stock_picks = request.env['stock.picking'].sudo().search( |
||||
|
[('origin', '=', so_id.name)]) |
||||
|
moves = stock_picks.mapped( |
||||
|
'move_ids_without_package').sudo().filtered( |
||||
|
lambda p: p.product_id.id == line['product_id']) |
||||
|
if moves: |
||||
|
moves = moves.sorted('product_uom_qty', reverse=True) |
||||
|
values.update({'state': 'draft'}) |
||||
|
ret_order = request.env['sale.return'].sudo().create(values) |
||||
|
moves[0].picking_id.return_order_id = ret_order.id |
||||
|
moves[0].picking_id.return_order_picking = False |
||||
|
return True |
||||
|
return False |
||||
|
|
||||
|
@http.route('/my/request-thank-you', website=True, page=True, |
||||
|
auth='public', csrf=False) |
||||
|
def maintenance_request_thanks(self): |
||||
|
"""Controller for return the thanks page for return confirmation.""" |
||||
|
return request.render( |
||||
|
'website_multi_product_return_management.' |
||||
|
'customers_request_thank_page') |
@ -0,0 +1,12 @@ |
|||||
|
<?xml version="1.0" encoding="UTF-8"?> |
||||
|
<odoo> |
||||
|
<data> |
||||
|
<record id="sale_return_ir_sequence" model="ir.sequence"> |
||||
|
<field name="name">Sale Return</field> |
||||
|
<field name="code">sale.return</field> |
||||
|
<field name="prefix">RET</field> |
||||
|
<field name="padding">5</field> |
||||
|
<field name="company_id" eval="False"/> |
||||
|
</record> |
||||
|
</data> |
||||
|
</odoo> |
@ -0,0 +1,7 @@ |
|||||
|
|
||||
|
## Module <website_multi_product_return_management> |
||||
|
|
||||
|
#### 06.05.2025 |
||||
|
#### Version 18.0.1.0.0 |
||||
|
#### ADD |
||||
|
- Initial Commit for Multi Product Return From Website |
@ -0,0 +1,25 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
from . import res_partner |
||||
|
from . import sale_order |
||||
|
from . import sale_return |
||||
|
from . import stock_picking |
@ -0,0 +1,63 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
from odoo import fields, models |
||||
|
|
||||
|
|
||||
|
class ResPartner(models.Model): |
||||
|
""" |
||||
|
Extends the base 'res.partner' model to include additional fields |
||||
|
and functionality. |
||||
|
""" |
||||
|
_inherit = 'res.partner' |
||||
|
|
||||
|
return_order_count = fields.Integer( |
||||
|
compute="_compute_return_order_count", string='Return Orders', |
||||
|
help="Count of order returned by the customer") |
||||
|
|
||||
|
def _compute_return_order_count(self): |
||||
|
"""Function to calculate the return count""" |
||||
|
all_partners = self.with_context(active_test=False).sudo().search( |
||||
|
[('id', 'child_of', self.ids)]) |
||||
|
all_partners.read(['parent_id']) |
||||
|
sale_return_groups = self.env['sale.return'].sudo().read_group( |
||||
|
domain=[('partner_id', 'in', all_partners.ids)], |
||||
|
fields=['partner_id'], groupby=['partner_id']) |
||||
|
for partner in self: |
||||
|
if sale_return_groups != []: |
||||
|
partner.return_order_count = int( |
||||
|
sale_return_groups[0]['partner_id_count']) |
||||
|
else: |
||||
|
partner.return_order_count = 0 |
||||
|
|
||||
|
def action_open_returns(self): |
||||
|
"""This function returns an action that displays the sale return |
||||
|
orders from partner.""" |
||||
|
action = self.env['ir.actions.act_window']._for_xml_id( |
||||
|
'website_multi_product_return_management.sale_return_action') |
||||
|
domain = [] |
||||
|
if self.is_company: |
||||
|
domain.append(('partner_id.commercial_partner_id.id', '=', self.id)) |
||||
|
else: |
||||
|
domain.append(('partner_id.id', '=', self.id)) |
||||
|
action['domain'] = domain |
||||
|
action['context'] = {'search_default_customer': 1} |
||||
|
return action |
@ -0,0 +1,65 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
from odoo import fields, models |
||||
|
|
||||
|
|
||||
|
class SaleOrder(models.Model): |
||||
|
_inherit = 'sale.order' |
||||
|
|
||||
|
return_order_count = fields.Integer( |
||||
|
compute="_compute_return_order_count", string='Return Orders', |
||||
|
help="Count of product returned on this order") |
||||
|
|
||||
|
def _compute_return_order_count(self): |
||||
|
"""method to compute return count""" |
||||
|
sale_return_groups = self.env['sale.return'].sudo().read_group( |
||||
|
domain=[('sale_order_id', 'in', self.ids)], |
||||
|
fields=['sale_order_id'], groupby=['sale_order_id']) |
||||
|
orders = self.sudo().browse() |
||||
|
for group in sale_return_groups: |
||||
|
sale_order = self.sudo().browse(group['sale_order_id'][0]) |
||||
|
while sale_order: |
||||
|
if sale_order in self: |
||||
|
print(group) |
||||
|
sale_order.return_order_count = 0 |
||||
|
sale_order.return_order_count += group['sale_order_id_count'] |
||||
|
orders |= sale_order |
||||
|
sale_order = False |
||||
|
(self - orders).return_order_count = 0 |
||||
|
|
||||
|
def action_open_returns(self): |
||||
|
"""This function returns an action that displays the sale return |
||||
|
orders from sale order""" |
||||
|
action = self.env['ir.actions.act_window']._for_xml_id( |
||||
|
'website_multi_product_return_management.sale_return_action') |
||||
|
domain = [('sale_order_id', '=', self.id)] |
||||
|
action['domain'] = domain |
||||
|
action['context'] = {'search_default_order': 1} |
||||
|
return action |
||||
|
|
||||
|
class SaleOrderLine(models.Model): |
||||
|
"""Extend the Sale Order line model to include a new field.""" |
||||
|
_inherit = 'sale.order.line' |
||||
|
|
||||
|
return_order_line_count = fields.Integer( |
||||
|
string='Return Orders', default=0, |
||||
|
help="Count of product returned in this order line") |
@ -0,0 +1,321 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
from odoo import api, fields, models |
||||
|
|
||||
|
|
||||
|
class SaleReturn(models.Model): |
||||
|
"""Model for managing return orders.""" |
||||
|
_name = 'sale.return' |
||||
|
_inherit = ['portal.mixin', 'mail.thread', 'mail.activity.mixin'] |
||||
|
_order = "name" |
||||
|
_description = "Return Order" |
||||
|
|
||||
|
@api.model |
||||
|
def _get_default_name(self): |
||||
|
"""Get the default name for a new return order.""" |
||||
|
return self.env['ir.sequence'].get('sale.return') |
||||
|
|
||||
|
active = fields.Boolean(string='Active', default=True, |
||||
|
help="Indicates if the return order is active.") |
||||
|
name = fields.Char(string="Name", default=_get_default_name, |
||||
|
help="Name of the return order.") |
||||
|
sale_order_id = fields.Many2one( |
||||
|
'sale.order', string="Sale Order", |
||||
|
required=True, |
||||
|
help="Reference to the original sale order.") |
||||
|
partner_id = fields.Many2one( |
||||
|
'res.partner', string="Customer", |
||||
|
help="Customer associated with the return order.") |
||||
|
user_id = fields.Many2one('res.users', string="Responsible", |
||||
|
default=lambda self: self.env.user, |
||||
|
help="User responsible for the return order.") |
||||
|
create_date = fields.Datetime( |
||||
|
string="Create Date", |
||||
|
help="Date when the return order was created.") |
||||
|
stock_picking_ids = fields.One2many( |
||||
|
'stock.picking', 'return_order_pick_id', |
||||
|
domain="[('return_order_id','=',False)]", |
||||
|
string="Return Picking", |
||||
|
help="Shows the return picking of the corresponding return order.") |
||||
|
picking_count = fields.Integer(compute="_compute_picking_count", |
||||
|
string='Picking Order', copy=False, |
||||
|
default=0, store=True, |
||||
|
help="Count of associated picking orders.") |
||||
|
delivery_count = fields.Integer(compute="_compute_delivery", |
||||
|
string='Delivery Order', copy=False, |
||||
|
default=0, store=True, |
||||
|
help="Count of associated delivery orders.") |
||||
|
state = fields.Selection( |
||||
|
[('draft', 'Draft'), ('confirm', 'Confirm'), ('done', 'Done'), |
||||
|
('cancel', 'Canceled')], string='Status', readonly=True, |
||||
|
default='draft', help="Current state of the return order.") |
||||
|
source_pick_ids = fields.One2many( |
||||
|
'stock.picking', 'return_order_id', |
||||
|
string="Source Delivery", |
||||
|
domain="[('return_order_pick_id','=',False)]", |
||||
|
help="Shows the delivery orders of the corresponding return order.") |
||||
|
note = fields.Text( |
||||
|
string="Note", |
||||
|
help="Additional notes or comments for the return order.") |
||||
|
|
||||
|
return_line_ids = fields.One2many( |
||||
|
'return.order.line', 'order_id', |
||||
|
string="Return Lines", |
||||
|
help="Lines associated with the return order.") |
||||
|
|
||||
|
def action_return_confirm(self): |
||||
|
"""Confirm the sale return""" |
||||
|
if not self.source_pick_ids: |
||||
|
stock_picks = self.env['stock.picking'].sudo().search( |
||||
|
[('origin', '=', self.sale_order_id.name)]) |
||||
|
moves = stock_picks.mapped( |
||||
|
'move_ids_without_package').sudo().filtered( |
||||
|
lambda p: p.product_id.id in self.return_line_ids.mapped( |
||||
|
'product_id').ids) |
||||
|
else: |
||||
|
moves = self.source_pick_ids.mapped( |
||||
|
'move_ids_without_package').sudo().filtered( |
||||
|
lambda p: p.product_id.id in self.return_line_ids.mapped( |
||||
|
'product_id').ids) |
||||
|
|
||||
|
if moves: |
||||
|
moves = moves.sorted('product_uom_qty', reverse=True) |
||||
|
pick = moves[0].picking_id |
||||
|
vals = {'picking_id': pick.id} |
||||
|
return_pick_wizard = self.env['stock.return.picking'].sudo().create( |
||||
|
vals) |
||||
|
return_pick_wizard._compute_moves_locations() |
||||
|
return_pick_wizard.product_return_moves.unlink() |
||||
|
lines = [] |
||||
|
reason = '' |
||||
|
for line in self.return_line_ids: |
||||
|
so_line_id = self.sale_order_id.order_line.filtered( |
||||
|
lambda m: m.product_template_id == |
||||
|
line.product_id.product_tmpl_id) |
||||
|
so_line_id.return_order_line_count = line.received_qty |
||||
|
move = moves.filtered(lambda m: m.product_id == line.product_id) |
||||
|
lines.append( |
||||
|
{'product_id': line.product_id.id, |
||||
|
"quantity": line.received_qty, |
||||
|
'move_id': move.id, 'to_refund': line.to_refund}) |
||||
|
reason = reason + '\n' + line.reason |
||||
|
lines = self.env['stock.return.picking.line'].create(lines) |
||||
|
return_pick_wizard.write({ |
||||
|
'product_return_moves': [(6, 0, lines.ids)] |
||||
|
}) |
||||
|
return_pick = return_pick_wizard._create_return() |
||||
|
if return_pick: |
||||
|
return_pick = self.env['stock.picking'].sudo().browse( |
||||
|
return_pick[0].id) |
||||
|
return_pick.update({'note': reason}) |
||||
|
return_pick.write( |
||||
|
{'return_order_id': False, 'return_order_pick_id': self.id, |
||||
|
'return_order_picking': True}) |
||||
|
self.write({'state': 'confirm'}) |
||||
|
|
||||
|
def action_return_cancel(self): |
||||
|
"""Cancel the return""" |
||||
|
self.write({'state': 'cancel'}) |
||||
|
if self.stock_picking_ids: |
||||
|
for rec in self.stock_picking_ids.sudo().filtered( |
||||
|
lambda s: s.state not in ['done', 'cancel']): |
||||
|
rec.action_cancel() |
||||
|
|
||||
|
def _get_report_base_filename(self): |
||||
|
"""Get the base filename for the return order report.""" |
||||
|
self.ensure_one() |
||||
|
return 'Sale Return - %s' % (self.name) |
||||
|
|
||||
|
def _compute_access_url(self): |
||||
|
"""Compute the access URL for the return order.""" |
||||
|
super(SaleReturn, self)._compute_access_url() |
||||
|
for order in self: |
||||
|
order.access_url = '/my/return_orders/%s' % order.id |
||||
|
|
||||
|
@api.depends('stock_picking_ids', 'state') |
||||
|
def _compute_delivery(self): |
||||
|
"""Function to compute picking and delivery counts""" |
||||
|
for rec in self: |
||||
|
rec.delivery_count = 0 |
||||
|
rec.picking_count = 0 |
||||
|
if rec.source_pick_ids: |
||||
|
rec.delivery_count = len(rec.source_pick_ids) |
||||
|
else: |
||||
|
rec.delivery_count = self.env[ |
||||
|
'stock.picking'].sudo().search_count( |
||||
|
[('return_order_id', 'in', self.ids), |
||||
|
('return_order_picking', '=', False)]) |
||||
|
if rec.stock_picking_ids: |
||||
|
rec.picking_count = len(rec.stock_picking_ids) |
||||
|
else: |
||||
|
rec.picking_count = self.env[ |
||||
|
'stock.picking'].sudo().search_count( |
||||
|
[('return_order_pick_id', 'in', self.ids), |
||||
|
('return_order_picking', '=', True)]) |
||||
|
|
||||
|
def action_view_picking(self): |
||||
|
"""Function to view the stock picking transfers""" |
||||
|
action = self.env["ir.actions.actions"]._for_xml_id( |
||||
|
"stock.action_picking_tree_all") |
||||
|
|
||||
|
pickings = self.mapped('stock_picking_ids') |
||||
|
if not self.stock_picking_ids: |
||||
|
pickings = self.env['stock.picking'].sudo().search( |
||||
|
[('return_order_pick_id', '=', self.id), |
||||
|
('return_order_picking', '=', True)]) |
||||
|
if len(pickings) > 1: |
||||
|
action['domain'] = [('id', 'in', pickings.ids)] |
||||
|
elif pickings: |
||||
|
form_view = [(self.env.ref('stock.view_picking_form').id, 'form')] |
||||
|
if 'views' in action: |
||||
|
action['views'] = form_view + [(state, view) for state, view in |
||||
|
action['views'] if |
||||
|
view != 'form'] |
||||
|
else: |
||||
|
action['views'] = form_view |
||||
|
action['res_id'] = pickings.id |
||||
|
# Prepare the context. |
||||
|
picking_id = pickings.sudo().filtered( |
||||
|
lambda l: l.picking_type_id.code == 'outgoing') |
||||
|
if picking_id: |
||||
|
picking_id = picking_id[0] |
||||
|
else: |
||||
|
picking_id = pickings[0] |
||||
|
action['context'] = dict( |
||||
|
self._context, |
||||
|
default_partner_id=self.partner_id.id, |
||||
|
default_picking_type_id=picking_id.picking_type_id.id) |
||||
|
return action |
||||
|
|
||||
|
def action_view_delivery(self): |
||||
|
"""Function to view the delivery transfers""" |
||||
|
action = self.env["ir.actions.actions"]._for_xml_id( |
||||
|
"stock.action_picking_tree_all") |
||||
|
|
||||
|
pickings = self.mapped('source_pick_ids') |
||||
|
if not self.source_pick_ids: |
||||
|
pickings = self.env['stock.picking'].sudo().search( |
||||
|
[('return_order_id', '=', self.id), |
||||
|
('return_order_picking', '=', False)]) |
||||
|
if len(pickings) > 1: |
||||
|
action['domain'] = [('id', 'in', pickings.ids)] |
||||
|
elif pickings: |
||||
|
form_view = [(self.env.ref('stock.view_picking_form').id, 'form')] |
||||
|
if 'views' in action: |
||||
|
action['views'] = form_view + [(state, view) for state, view in |
||||
|
action['views'] if |
||||
|
view != 'form'] |
||||
|
else: |
||||
|
action['views'] = form_view |
||||
|
action['res_id'] = pickings.id |
||||
|
# Prepare the context. |
||||
|
picking_id = pickings.filtered( |
||||
|
lambda l: l.picking_type_id.code == 'outgoing') |
||||
|
if picking_id: |
||||
|
picking_id = picking_id[0] |
||||
|
else: |
||||
|
picking_id = pickings[0] |
||||
|
action['context'] = dict( |
||||
|
self._context, |
||||
|
default_partner_id=self.partner_id.id, |
||||
|
default_picking_type_id=picking_id.picking_type_id.id) |
||||
|
return action |
||||
|
|
||||
|
@api.onchange('sale_order_id', 'source_pick_ids') |
||||
|
def onchange_sale_order_id_source_pick_ids(self): |
||||
|
"""All the fields are updated according to the sale order""" |
||||
|
delivery = None |
||||
|
if self.sale_order_id: |
||||
|
self.partner_id = self.sale_order_id.partner_id |
||||
|
|
||||
|
delivery = self.env['stock.picking'].sudo().search( |
||||
|
[('origin', '=', self.sale_order_id.name)]) |
||||
|
if self.source_pick_ids: |
||||
|
delivery = self.source_pick_ids |
||||
|
if delivery: |
||||
|
product_ids = delivery.move_ids_without_package.mapped( |
||||
|
'product_id').ids |
||||
|
delivery = delivery.ids |
||||
|
else: |
||||
|
product_ids = self.sale_order_id.order_line.mapped('product_id').ids |
||||
|
|
||||
|
return {'domain': {'source_pick_ids': [('id', 'in', delivery)], |
||||
|
'product_id': [('id', 'in', product_ids)]}} |
||||
|
|
||||
|
@api.onchange('product_id') |
||||
|
def onchange_product_id(self): |
||||
|
"""Handle changes in the product ID field. and find received quantity""" |
||||
|
if self.product_id and self.source_pick_ids: |
||||
|
moves = self.source_pick_ids.mapped( |
||||
|
'move_ids_without_package').sudo().filtered( |
||||
|
lambda p: p.product_id == self.product_id) |
||||
|
if moves: |
||||
|
self.received_qty = sum(moves.mapped('quantity_done')) |
||||
|
|
||||
|
|
||||
|
class ReturnOrderLine(models.Model): |
||||
|
_name = 'return.order.line' |
||||
|
_description = "Return Products Details" |
||||
|
|
||||
|
order_id = fields.Many2one( |
||||
|
"sale.return", string="Order", |
||||
|
help="Reference to the associated sale return order.") |
||||
|
product_id = fields.Many2one( |
||||
|
'product.product', string="Product Variant", |
||||
|
required=True, |
||||
|
help="Defines the product variant that needs to be returned.") |
||||
|
product_tmpl_id = fields.Many2one('product.template', |
||||
|
related="product_id.product_tmpl_id", |
||||
|
store=True, string="Product") |
||||
|
quantity = fields.Float( |
||||
|
string="Delivered Quantity", store=True, |
||||
|
help="Quantity originally delivered in the associated sale order.") |
||||
|
received_qty = fields.Float( |
||||
|
string="Received Quantity", |
||||
|
help="Quantity received for the return order.") |
||||
|
reason = fields.Text( |
||||
|
"Reason", help="Reason for returning the product.") |
||||
|
to_refund = fields.Boolean( |
||||
|
string='Update SO/PO Quantity', |
||||
|
help='Trigger a decrease of the delivered/received quantity in the' |
||||
|
' associated Sale Order/Purchase Order.') |
||||
|
sale_order_id = fields.Many2one( |
||||
|
"sale.order", string="Sale Order", |
||||
|
related="order_id.sale_order_id", |
||||
|
help="Reference to the associated sale order.") |
||||
|
|
||||
|
@api.onchange('product_id') |
||||
|
def onchange_product_id(self): |
||||
|
""" setting up domain for products as per the products in the |
||||
|
vendor price list""" |
||||
|
self.ensure_one() |
||||
|
if self._context.get('order_id'): |
||||
|
order_id = self.env['sale.order'].browse( |
||||
|
self._context.get('order_id')) |
||||
|
products = order_id.order_line.mapped('product_id').ids |
||||
|
if self.product_id: |
||||
|
qty = sum( |
||||
|
order_id.order_line.filtered( |
||||
|
lambda p: p.product_id == self.product_id).mapped( |
||||
|
'qty_delivered')) |
||||
|
self.quantity = qty |
||||
|
return {'domain': {'product_id': [('id', 'in', products)]}} |
@ -0,0 +1,52 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################### |
||||
|
# |
||||
|
# Cybrosys Technologies Pvt. Ltd. |
||||
|
# |
||||
|
# Copyright (C) 2025-TODAY Cybrosys Technologies(<https://www.cybrosys.com>) |
||||
|
# Author: Cybrosys Techno Solutions (<https://www.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 <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################### |
||||
|
from odoo import fields, models |
||||
|
|
||||
|
|
||||
|
class StockPicking(models.Model): |
||||
|
"""Extend the stock.picking model to include return order information.""" |
||||
|
|
||||
|
_inherit = 'stock.picking' |
||||
|
|
||||
|
return_order_id = fields.Many2one( |
||||
|
'sale.return', string='Return Order', |
||||
|
help="Shows the return order of the current transfer.") |
||||
|
return_order_pick_id = fields.Many2one( |
||||
|
'sale.return', |
||||
|
string='Return Order Pick', |
||||
|
help="Shows the return order picking of the current return order.") |
||||
|
return_order_picking = fields.Boolean( |
||||
|
string='Return Order Picking', |
||||
|
help="Helps to identify delivery and return picking. If true, the " |
||||
|
"transfer is a return picking; else, it's a delivery.") |
||||
|
|
||||
|
def button_validate(self): |
||||
|
"""Override the button_validate method to update return order state.""" |
||||
|
res = super(StockPicking, self).button_validate() |
||||
|
for rec in self: |
||||
|
if rec.return_order_pick_id: |
||||
|
if any(line.state != 'done' for line in |
||||
|
rec.return_order_pick_id.stock_picking_ids): |
||||
|
return res |
||||
|
else: |
||||
|
rec.return_order_pick_id.write({'state': 'done'}) |
||||
|
return res |
@ -0,0 +1,14 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Define a report action for the Sale Return report --> |
||||
|
<record id="report_sale_returns" model="ir.actions.report"> |
||||
|
<field name="name">Sale Return</field> |
||||
|
<field name="model">sale.return</field> |
||||
|
<field name="report_type">qweb-pdf</field> |
||||
|
<field name="report_name">website_multi_product_return_management.report_salereturn</field> |
||||
|
<field name="report_file">website_multi_product_return_management.report_salereturn</field> |
||||
|
<field name="print_report_name">'Sale Retun - %s' % object.name</field> |
||||
|
<field name="binding_model_id" ref="model_sale_return"/> |
||||
|
<field name="binding_type">report</field> |
||||
|
</record> |
||||
|
</odoo> |
@ -0,0 +1,89 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
<!-- Define a report action for the Sale Return report --> |
||||
|
<record id="report_sale_returns" model="ir.actions.report"> |
||||
|
<field name="name">Sale Return</field> |
||||
|
<field name="model">sale.return</field> |
||||
|
<field name="report_type">qweb-pdf</field> |
||||
|
<field name="report_name">website_multi_product_return_management.report_salereturn</field> |
||||
|
<field name="report_file">website_multi_product_return_management.report_salereturn</field> |
||||
|
<field name="print_report_name">'Sale Retun - %s' % object.name</field> |
||||
|
<field name="binding_model_id" ref="model_sale_return"/> |
||||
|
<field name="binding_type">report</field> |
||||
|
</record> |
||||
|
<!-- Define a template for the Sale Return report --> |
||||
|
<template id="report_salereturn"> |
||||
|
<t t-call="web.html_container"> |
||||
|
<t t-foreach="docs" t-as="doc"> |
||||
|
<t t-call="website_multi_product_return_management.report_sale_return" |
||||
|
t-lang="doc.partner_id.lang"/> |
||||
|
</t> |
||||
|
</t> |
||||
|
</template> |
||||
|
<!-- Define the structure of the Sale Return report --> |
||||
|
<template id="report_sale_return"> |
||||
|
<t t-call="web.external_layout"> |
||||
|
<t t-set="doc" |
||||
|
t-value="doc.with_context(lang=doc.partner_id.lang)"/> |
||||
|
<div class="page"> |
||||
|
<div class="oe_structure"/> |
||||
|
<h2 class="mt16"> |
||||
|
<span>Return Order #</span> |
||||
|
<span t-field="doc.name"/> |
||||
|
</h2> |
||||
|
<div class="row mt32 mb32" id="informations"> |
||||
|
<div t-if="doc.partner_id" |
||||
|
class="col-auto col-3 mw-100 mb-2"> |
||||
|
<strong>Customer:</strong> |
||||
|
<p class="m-0" t-field="doc.partner_id"/> |
||||
|
</div> |
||||
|
<div t-if="doc.sale_order_id" |
||||
|
class="col-auto col-3 mw-100 mb-2"> |
||||
|
<strong>Sale Order:</strong> |
||||
|
<p class="m-0" t-field="doc.sale_order_id"/> |
||||
|
</div> |
||||
|
<div t-if="doc.create_date" |
||||
|
class="col-auto col-3 mw-100 mb-2"> |
||||
|
<strong>Creation Date:</strong> |
||||
|
<p class="m-0" t-field="doc.create_date" |
||||
|
t-options='{"widget": "date"}'/> |
||||
|
</div> |
||||
|
<div t-if="doc.user_id" |
||||
|
class="col-auto col-3 mw-100 mb-2" |
||||
|
name="expiration_date"> |
||||
|
<strong>Responsible:</strong> |
||||
|
<p class="m-0" t-field="doc.user_id"/> |
||||
|
</div> |
||||
|
</div> |
||||
|
<table class="table table-sm o_main_table"> |
||||
|
<!-- In case we want to repeat the header, remove "display: table-row-group" --> |
||||
|
<thead style="display: table-row-group"> |
||||
|
<tr> |
||||
|
<th name="th_description" class="text-left"> |
||||
|
Product |
||||
|
</th> |
||||
|
<th name="th_quantity" class="text-left">Quantity |
||||
|
</th> |
||||
|
<th name="th_quantity" class="text-left">Reason</th> |
||||
|
</tr> |
||||
|
</thead> |
||||
|
<tbody class="sale_tbody"> |
||||
|
<tr t-foreach="doc.return_line_ids" t-as="line" |
||||
|
t-att-class="'bg-200 font-weight-bold'"> |
||||
|
<td name="td_product"> |
||||
|
<span t-field="line.product_id"/> |
||||
|
</td> |
||||
|
<td name="td_quantity" class="text-left"> |
||||
|
<span t-esc="line.received_qty"/> |
||||
|
<span t-field="line.product_id.uom_id"/> |
||||
|
</td> |
||||
|
<td name="td_quantity" class="text-left"> |
||||
|
<span t-field="line.reason"/> |
||||
|
</td> |
||||
|
</tr> |
||||
|
</tbody> |
||||
|
</table> |
||||
|
</div> |
||||
|
</t> |
||||
|
</template> |
||||
|
</odoo> |
|
After Width: | Height: | Size: 2.2 KiB |
After Width: | Height: | Size: 28 KiB |
After Width: | Height: | Size: 628 KiB |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 210 KiB |
After Width: | Height: | Size: 209 KiB |
After Width: | Height: | Size: 109 KiB |
After Width: | Height: | Size: 495 B |
After Width: | Height: | Size: 1.0 KiB |
After Width: | Height: | Size: 624 B |
After Width: | Height: | Size: 136 KiB |
After Width: | Height: | Size: 214 KiB |
After Width: | Height: | Size: 36 KiB |
After Width: | Height: | Size: 3.6 KiB |
After Width: | Height: | Size: 310 B |
After Width: | Height: | Size: 929 B |
After Width: | Height: | Size: 1.3 KiB |
After Width: | Height: | Size: 3.3 KiB |
After Width: | Height: | Size: 1.4 KiB |
After Width: | Height: | Size: 17 KiB |
After Width: | Height: | Size: 542 B |
After Width: | Height: | Size: 576 B |
After Width: | Height: | Size: 733 B |
After Width: | Height: | Size: 4.3 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 4.0 KiB |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 2.2 MiB |
After Width: | Height: | Size: 2.2 KiB |
After Width: | Height: | Size: 911 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 600 B |
After Width: | Height: | Size: 673 B |
After Width: | Height: | Size: 2.0 KiB |
After Width: | Height: | Size: 462 B |
After Width: | Height: | Size: 2.1 KiB |
After Width: | Height: | Size: 926 B |
After Width: | Height: | Size: 9.0 KiB |
After Width: | Height: | Size: 23 KiB |
After Width: | Height: | Size: 7.0 KiB |
After Width: | Height: | Size: 878 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 653 B |
After Width: | Height: | Size: 800 B |
After Width: | Height: | Size: 905 B |
After Width: | Height: | Size: 189 KiB |
After Width: | Height: | Size: 4.3 KiB |
After Width: | Height: | Size: 839 B |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 5.9 KiB |
After Width: | Height: | Size: 1.6 KiB |
After Width: | Height: | Size: 34 KiB |
After Width: | Height: | Size: 26 KiB |
After Width: | Height: | Size: 3.8 KiB |
After Width: | Height: | Size: 23 KiB |
After Width: | Height: | Size: 1.9 KiB |
After Width: | Height: | Size: 2.3 KiB |
After Width: | Height: | Size: 427 B |
After Width: | Height: | Size: 627 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 988 B |
After Width: | Height: | Size: 3.7 KiB |
After Width: | Height: | Size: 5.0 KiB |
After Width: | Height: | Size: 875 B |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 86 KiB |
After Width: | Height: | Size: 90 KiB |
After Width: | Height: | Size: 118 KiB |
After Width: | Height: | Size: 131 KiB |
After Width: | Height: | Size: 138 KiB |
After Width: | Height: | Size: 85 KiB |
After Width: | Height: | Size: 159 KiB |
After Width: | Height: | Size: 114 KiB |
After Width: | Height: | Size: 138 KiB |
After Width: | Height: | Size: 99 KiB |
After Width: | Height: | Size: 106 KiB |
After Width: | Height: | Size: 38 KiB |
After Width: | Height: | Size: 72 KiB |
After Width: | Height: | Size: 59 KiB |
After Width: | Height: | Size: 72 KiB |
After Width: | Height: | Size: 71 KiB |