
本教程详细介绍了如何在odoo 16的表单视图中,根据当前记录的数据动态修改内嵌tree视图的属性,例如设置分页限制。文章深入探讨了通过重写`get_view`方法并利用`self.env.context['params'].get('id')`准确获取当前记录id的关键技术,并提供了完整的代码示例和实现步骤,帮助开发者实现高度定制化的视图行为。
Odoo提供了一套强大的视图系统,允许开发者通过XML定义界面的结构和行为。然而,在某些业务场景下,我们可能需要根据当前正在查看或编辑的记录的具体数据,动态地调整视图中某个组件的属性。一个常见的例子是在主表单中嵌入一个子Tree视图(也称为One2Many或Many2Many字段的列表视图),并希望根据主记录的某个字段值来改变这个子Tree视图的分页限制、可编辑性或列的显示。
本文将以在Odoo 16的stock.picking(调拨单)表单视图中,根据调拨单记录上定义的limit字段值,动态设置其关联的move_ids_without_package Tree视图(调拨行)的分页限制为例,详细讲解如何实现这一需求。
Odoo在加载任何视图时,都会调用相应的模型上的get_view方法。这个方法是进行视图XML架构动态修改的理想入口点。通过重写此方法,我们可以在视图数据发送到前端渲染之前,对其XML结构进行解析、修改和重新序列化。
get_view方法接收以下主要参数:
在get_view方法中,一个常见的挑战是准确获取当前正在加载的表单视图所对应的记录ID。开发者有时会尝试使用self.env.context.get('active_id'),但在get_view的上下文中,active_id可能不总是指向当前表单的记录,尤其是在通过菜单项或关联字段打开表单时。
正确的做法是利用self.env.context['params'].get('id')。 params字典通常包含更精确的视图加载参数,其中id键会可靠地提供当前正在加载的表单视图所关联的记录的数据库ID。
下面我们将通过一个完整的代码示例,演示如何在stock.picking模型中实现根据记录的limit字段动态设置move_ids_without_package Tree视图的分页限制。
首先,我们需要在stock.picking模型中添加一个用于存储分页限制的字段,并重写get_view方法。
from odoo import models, fields, api
from lxml import etree
import logging
_logger = logging.getLogger(__name__)
class StockPicking(models.Model):
_inherit = "stock.picking"
limit = fields.Integer(string="Tree 分页限制", default=0, help="设置关联Tree视图的每页显示记录数。0表示不设置限制。")
def tree_pagination_limit_apply(self):
"""
一个示例方法,可以在用户界面中调用,以触发当前视图的重新加载,
从而应用get_view中定义的动态修改。
例如,可以在表单上添加一个按钮,点击后调用此方法。
"""
return {
'type': 'ir.actions.client',
'tag': 'reload',
}
@api.model
def get_view(self, view_id=None, view_type='form', **options):
_logger.info(f"Custom get_view for StockPicking called for view_type: {view_type}")
# 1. 调用父类方法获取原始视图架构
# 这是非常重要的一步,确保Odoo的默认视图处理逻辑不被破坏
result = super(StockPicking, self).get_view(view_id=view_id, view_type=view_type, **options)
# 2. 仅在表单视图类型下进行修改,避免影响其他视图类型
if view_type == 'form':
# 3. 解析视图XML架构
# 使用lxml库解析XML字符串,便于操作
doc = etree.XML(result['arch'])
# 4. 获取当前表单记录的ID
# 关键:从self.env.context['params']中获取id,这是获取当前表单记录ID最可靠的方式
active_id = self.env.context['params'].get('id') if 'params' in self.env.context else None
if active_id:
try:
# 5. 浏览当前记录以获取其数据
current_record = self.browse(active_id)
# 检查记录是否存在且limit字段值大于0才进行设置
if current_record and current_record.limit > 0:
# 6. 定位目标Tree视图并修改属性
# 使用XPath表达式精确查找名为'move_ids_without_package'的field下的'tree'标签
for tree_node in doc.xpath("//field[@name='move_ids_without_package']/tree"):
tree_node.set('limit', str(current_record.limit))
_logger.info(f"为记录ID {active_id} (Picking Name: {current_record.name}) 设置Tree视图限制为 {current_record.limit}")
break # 假设一个field下只有一个tree,找到后即可退出循环
except Exception as e:
_logger.error(f"在设置Tree视图限制时发生错误 (记录ID: {active_id}): {e}")
# 7. 将修改后的XML架构更新回结果中
# lxml的tostring方法返回bytes,需要解码为unicode字符串
result['arch'] = etree.tostring(doc, encoding='unicode').decode('utf-8')
return result
通过重写Odoo模型的get_view方法,并结合lxml库进行XML架构操作,我们可以实现高度灵活和动态的视图定制。关键在于理解get_view的执行时机,以及如何通过self.env.context['params'].get('id')可靠地获取当前表单记录的ID。掌握这些技术,开发者可以根据业务需求,创建更加智能和用户友好的Odoo界面。在实际应用中,务必注意代码的健壮性、性能影响以及与Odoo未来版本的兼容性。
以上就是Odoo 16:在表单视图中动态修改Tree视图属性的教程的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号