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