odoo17 | 计算字段和更改事件

前言

模型之间的关系是任何Odoo模块的关键组成部分。它们是任何业务案例建模所必需的。然而,我们可能希望给定模型中的字段之间存在链接。有时一个字段的值是由其他字段的值决定的,而有时我们希望帮助用户进行数据输入。

这些案例得到了计算字段和onchanges概念的支持。虽然本章在技术上并不复杂,但这两个概念的语义非常重要。这也是我们第一次编写Python逻辑。到目前为止,除了类定义和字段声明之外,我们还没有编写任何内容。

计算字段

目标

  • 在房地产模型中,应计算总面积和最佳报价
    在这里插入图片描述
  • 在房地产报价模型中,应计算并更新有效期
    在这里插入图片描述
    在我们的房地产模块中,我们定义了生活区和花园区。因此,将总面积定义为这两个字段的总和是很自然的。我们将使用计算字段的概念,即给定字段的值将根据其他字段的值计算得出。

到目前为止,字段都是直接存储在数据库中并直接从数据库中检索的。字段也可以被计算。在这种情况下,字段的值不是从数据库中检索的,而是通过调用模型的方法动态计算的。

要创建计算字段,请创建一个字段并将其属性计算设置为方法的名称。计算方法应为 self 中的每个记录设置计算字段的值。

按照惯例,计算方法是私有的,这意味着它们不能从表示层调用,只能从业务层调用。私有方法有一个以下划线 _ 开头的名称。

依赖

计算字段的值通常取决于计算记录中其他字段的值。ORM希望开发人员使用装饰器 depends()在计算方法上指定这些依赖关系。给定的依赖关系由ORM使用,以便在修改了某些依赖关系时触发字段的重新计算:

from odoo import api, fields, models

class TestComputed(models.Model):
    _name = "test.computed"

    total = fields.Float(compute="_compute_total")
    amount = fields.Float()

    @api.depends("amount")
    def _compute_total(self):
        for record in self:
            record.total = 2.0 * record.amount

注意
self 是一种集合。

对象self是一个记录集,即一个有序的记录集合。它支持集合上的标准Python操作,例如len(self)和iter(self),以及额外的集合操作,如recs1|recs2

遍历self会逐个给出记录,其中每个记录本身都是大小为1的集合。您可以使用点符号
访问/分配 单个记录上的字段,例如 record.name

更多示例代码

    @api.depends('full_reconcile_id.name', 'matched_debit_ids', 'matched_credit_ids')
    def _compute_matching_number(self):
        for record in self:
            if record.full_reconcile_id:
                record.matching_number = record.full_reconcile_id.name
            elif record.matched_debit_ids or record.matched_credit_ids:
                record.matching_number = 'P'
            else:
                record.matching_number = None

    @api.depends('debit', 'credit')
    def _compute_balance(self):
        for line in self:
            line.balance = line.debit - line.credit

对于关系字段,可以使用通过字段的路径作为依赖项:

  • Many2one
description = fields.Char(compute="_compute_description")
partner_id = fields.Many2one("res.partner")

@api.depends("partner_id.name")
def _compute_description(self):
    for record in self:
        record.description = "Test for partner %s" % record.partner_id.name
  • Many2many & One2many.
line_ids = fields.One2many('account.reconcile.model.line', 'model_id')
show_decimal_separator = fields.Boolean(compute='_compute_show_decimal_separator', help="决定是否应该显示正则表达式匹配字段的小数分隔符的技术字段。")
@api.depends('line_ids.amount_type')
    def _compute_show_decimal_separator(self):
        for record in self:
            record.show_decimal_separator = any(l.amount_type == 'regex' for l in record.line_ids)

Inverse 反函数/逆向方法

您可能已经注意到,默认情况下计算字段是只读的。这是意料之中的,因为用户不应该设置值。

在某些情况下,仍然可以直接设置一个值。在我们的房地产例子中,我们可以为报价定义一个有效期并设置一个有效日期。我们希望能够设置一个影响另一个的持续时间或日期。

为了支持这一点,Odoo提供了使用Inverse 反函数的能力。

from odoo import api, fields, models

class TestComputed(models.Model):
    _name = "test.computed"

    total = fields.Float(compute="_compute_total", inverse="_inverse_total")
    amount = fields.Float()

    @api.depends("amount")
    def _compute_total(self):
        for record in self:
            record.total = 2.0 * record.amount

    def _inverse_total(self):
        for record in self:
            record.amount = record.total / 2.0

计算方法设置字段,而反函数/逆向方法设置字段的依赖关系。

请注意,在保存记录时调用反函数/逆向方法,而在每次更改其依赖关系时调用计算方法。

其他信息

默认情况下,计算字段不会存储在数据库中。因此,除非定义了搜索方法,否则无法在计算字段上进行搜索。此主题超出了本培训的范围,因此我们不会涉及。可以在此处找到示例。

 is_ongoing = fields.Boolean('Is Ongoing', compute='_compute_is_ongoing', search='_search_is_ongoing')

另一种解决方案是使用store=True属性存储字段。虽然这通常很方便,但要注意给模型增加的潜在计算负载。让我们重新使用我们的例子

description = fields.Char(compute="_compute_description", store=True)
partner_id = fields.Many2one("res.partner")

@api.depends("partner_id.name")
def _compute_description(self):
    for record in self:
        record.description = "Test for partner %s" % record.partner_id.name

每次更改合作伙伴名称时,都会自动重新计算所有引用它的记录的描述!当数百万条记录需要重新计算时,这很快就会变得难以实现。

值得注意的是,一个计算字段可以依赖于另一个计算字段。ORM足够智能,可以按正确的顺序重新计算所有依赖关系……但有时会以性能下降为代价。

一般来说,在定义计算字段时,必须始终牢记性能。计算字段越复杂(例如,具有大量依赖关系或计算字段依赖于其他计算字段),计算所需的时间就越长。始终花一些时间提前评估计算字段的成本。大多数时候,只有当你的代码到达生产服务器时,你才会意识到它减慢了整个过程。

Onchanges 更改

目标:

  • 启用花园将设置默认区域 10 和 朝向北方。
    在这里插入图片描述
    在我们的房地产模块中,我们还想帮助用户进行数据输入。当“garden”字段被设置时,我们想为garden area和orientation提供一个默认值。此外,当“garden”字段未设置时,我们希望garden area重置为零,方向被删除。在这种情况下,给定字段的值会修改其他字段的值。

“onchange”机制为客户端界面提供了一种更新表单的方法,当用户填写字段值时,无需将任何内容保存到数据库。为此,我们定义了一个方法,其中self表示表单视图中的记录,并用onchange()装饰它,以指定它是由哪个字段触发的。您对self所做的任何更改都会反映在表单上:

from odoo import api, fields, models

class TestOnchange(models.Model):
    _name = "test.onchange"

    name = fields.Char(string="Name")
    description = fields.Char(string="Description")
    partner_id = fields.Many2one("res.partner", string="Partner")

    @api.onchange("partner_id")
    def _onchange_partner_id(self):
        self.name = "Document for %s" % (self.partner_id.name)
        self.description = "Default description for %s" % (self.partner_id.name)

在这个例子中,更改合作伙伴也会更改名称和描述值。之后是否更改名称和描述值由用户决定。另请注意,我们不会在self上循环,这是因为该方法仅在表单视图中触发,其中self始终是单个记录。

其他信息

Onchanges 方法还可以返回非阻塞警告消息

    @api.onchange('provider', 'check_validity')
    def onchange_check_validity(self):
        if self.provider == 'authorize' and self.check_validity:
            self.check_validity = False
            return {'warning': {
                'title': _("Warning"),
                'message': ('This option is not supported for Authorize.net')}}

如何使用它们?

对于计算字段和onchanges的使用没有严格的规定。

在许多情况下,计算字段和onchanges都可以用来实现相同的结果。总是首选计算字段,因为它们也在表单视图上下文之外触发。永远不要使用onchanges向模型添加业务逻辑。这是一个非常糟糕的想法,因为通过编程方式创建记录时,onchanges不会自动触发;它们只在表单视图中触发。

计算字段和onchanges的常见陷阱是试图通过添加过多的逻辑来变得“过于智能”。这可能会产生与预期相反的结果:最终用户对所有自动化感到困惑。

计算字段往往更容易调试:这样的字段是由给定的方法设置的,因此很容易跟踪值的设置时间。另一方面,onchanges可能会令人困惑:很难知道onchanges的范围。由于几个onchanges方法可能会设置相同的字段,因此很容易难以跟踪值的来源。

使用存储的计算字段时,请密切关注依赖关系。当计算字段依赖于其他计算字段时,更改一个值可能会触发大量的重新计算。这会导致性能不佳。

在下一章中,我们将看到如何在单击按钮时触发一些业务逻辑。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/291441.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

pycharm远程开发调试(remote development)踩坑记录2

在一次我清理了服务器上一些老的pycharm版本之后 打算重新装3.2版本,就全部给清理了。结果坏了事了,新版的装不上了。 试了公司和中科院的服务器都出现这样的问题,100%复现。md。 一直在这一步循环: Downloading the IDE Backen…

航天航空线束工艺3D虚拟展馆支持多人异地参观漫游

为了满足汽车线束企业员工工作需要,让新老员工了解到更先进、规范的线束工艺设计技术,华锐视点基于VR虚拟仿真、web3d开发和图形图像技术制作了一款汽车线束工艺设计VR虚拟仿真模拟展示系统。 汽车线束工艺设计VR虚拟仿真模拟展示系统共分为pc电脑端和VR…

在Google Colab中调用Gemini的API实现智能问答

一、引言 Google终于放出大招,在2023年12月6日正式推出规模最大、功能最强大的人工智能模型Gemini,对标ChatGPT,甚至有要赶超ChatGPT-4.0的节奏。 相比之前的Bard,Gemini的文本理解能力、图片识别能力和语义抽取能力大大增强&am…

图像分割实战-系列教程1:语义分割与实例分割概述

🍁🍁🍁图像分割实战-系列教程 总目录 有任何问题欢迎在下面留言 本篇文章的代码运行界面均在Pycharm中进行 本篇文章配套的代码资源已经上传 下篇内容: Unet系列算法 1、图像分割任务概述 1.1 图像分割 分割任务就是在原始图像…

php学习06-魔术常量

有九个魔术常量它们的值随着它们在代码中的位置改变而改变。例如 LINE 的值就依赖于它在脚本中所处的行来决定。这些特殊的常量不区分大小写,如下: 参考

小红书12月内容趋势分析

为洞察小红书平台的内容创作趋势及品牌营销策略,新红推出12月月度榜单,从创作者、品牌、热搜词多方面入手,解析月榜数据,为从业者提供参考。 以下为12月部分榜单解析,想要查看更多行业榜单,创作优质内容&am…

【智慧零售】东胜物联蓝牙网关硬件解决方案,促进零售门店数字化管理

依托物联网(IoT)、大数据、人工智能(AI)等快速发展,数字化和智能化已成为零售企业的核心竞争力。更多的企业通过引入人工智能、大数据等先进技术手段,提高门店运营效率和服务质量。 某连锁咖啡企业牢牢抓住…

UCharts配置个性化图表:折柱混合、条状图、渐变

UCharts配置个性化图表:折柱混合、条状图、渐变 折线圆滑折线柱状图饼图条状图折柱混合渐变 折线 效果图: 配置: const opts {color: ["#B7E55D", "#78DAE6", "#FF432A", "#FF9641", "#FFD…

Spring的IOC解决程序耦合

pom.xml: <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.apache.org/POM/4.0.0 …

SpringBoot-搭建集成Mybatis的项目

本文介绍了如何在IntelliJ IDEA中使用SpringBoot和Mybatis构建Java Web应用程序。通过本文的学习&#xff0c;读者将了解如何使用IntelliJ IDEA快速搭建一个基于SpringBoot和Mybatis的Java Web应用程序&#xff0c;提高开发效率。IntelliJ IDEA是一款功能强大的Java集成开发环境…

acwing 1358. 约数个数和(莫比乌斯函数)

设 d(x)&#xfffd;(&#xfffd;) 为 x&#xfffd; 的约数个数&#xff0c;给定 N,M&#xfffd;,&#xfffd;&#xff0c;求 ∑i1N∑j1Md(ij)∑&#xfffd;1&#xfffd;∑&#xfffd;1&#xfffd;&#xfffd;(&#xfffd;&#xfffd;) 输入格式 输入多组测试数据…

Spark内核解析-内存管理7(六)

1、Spark内存管理 Spark 作为一个基于内存的分布式计算引擎&#xff0c;其内存管理模块在整个系统中扮演着非常重要的角色。理解 Spark 内存管理的基本原理&#xff0c;有助于更好地开发 Spark 应用程序和进行性能调优。本文中阐述的原理基于 Spark 2.1 版本。 在执行 Spark 的…

VMware17 下载安装教程

VMware17 下载安装ubuntu22.04虚拟机安装 一、VM安装 1.打开官方下载地址&#xff1a;https://www.vmware.com/products/workstation-pro/workstation-pro-evaluation.html 跳转页面后&#xff0c;点击左边的Download oad now&#xff0c;下载的就是最新版的 Workstation 17 …

stm32实战之su-03t语音模块固件的制作与烧录

目录 su-03t简介 管脚定义 ​​智能公元语音固件制作​​ 账号注册 创建产品 产品配置 唤醒词自定义 命令词自定义 发音人配置 其他配置 生成和下载语音固件 固件烧录 下载SDK固件烧录工具 SU-03T驱动分享 su-03t简介 SU-03T 是一款低成本、低功耗、小体积的离线…

平衡二叉树,力扣

目录 前序遍历与后续遍历 题目地址&#xff1a; 题目&#xff1a; 我们直接看题解吧&#xff1a; 审题目事例提示&#xff1a; 解题方法&#xff1a; 难度分析&#xff1a; 解题方法分析&#xff1a; 解题分析&#xff1a; 解题思路&#xff1a; 代码实现&#xff1a; 补充说明…

idea 社区版 Database Navigator插件 列显示顺序错乱解决办法

idea 社区版 Database Navigator插件 列显示顺序错乱 影响&#xff1a;MyBatisCodeHelperPro插件生成代码字段顺序错乱 解决办法&#xff1a;将COLUMN 的排序方式由Name改为Position方式之后&#xff0c;reload即可&#xff01;

Spring Security 6.x 系列(14)—— 会话管理之源码分析

一、前言 在上篇 Spring Security 6.x 系列(13)—— 会话管理之会话概念及常用配置 Spring Security 6.x 系列(14)—— 会话管理之会话固定攻击防护及Session共享 中了清晰了协议和会话的概念、对 Spring Security 中的常用会话配置进行了说明,并了解会话固定攻击防护…

SCT52240Q双路 4A/4A 高速MOSFET/IGBT栅极驱动器, 可并联输出,替代UCC27524

• 4.5-24V宽供电电压 • 4A 峰值驱动拉电流和灌电流 • 双通道并联输出&#xff0c;增强驱动能力 • 低至-5V负压输入 • 支持TTL低压逻辑输入 • 13ns传输延迟 • 快速上升下降时间&#xff08;典型值8ns&#xff09; • 双通道1ns典型值延迟匹配时间 • 55uA静态功耗 • 输入…

巨杉数据库荣登2023胡润全球猎豹企业榜

胡润研究院与广州南沙联合发布《2023胡润全球猎豹企业榜》&#xff0c;这是胡润研究院首次发布“全球猎豹企业”。榜单列出了全球成立于2000年后&#xff0c;五年内最有可能达到独角兽级十亿美金估值的高成长性企业。巨杉数据库凭借在分布式文档型数据库领域的创新突破&#xf…

SpringMVC-视图

SpringMVC中的视图实现了View接口&#xff0c;作用是渲染数据&#xff0c;将Model中的数据展示给用户。render是渲染方法&#xff0c;可以看到渲染的视图是一个View类型的对象。 SpringMVC视图的种类有很多&#xff0c;默认有转发视图和重定向视图。 如果配置了Thymeleaf视图解…