odoo17 | 计算字段和更改事件

前言

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

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

计算字段

目标

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

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

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

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

依赖

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

from odoo import api, fields, modelsclass 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.nameelif 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, modelsclass 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.amountdef _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, modelsclass 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 = Falsereturn {'warning': {'title': _("Warning"),'message': ('This option is not supported for Authorize.net')}}

如何使用它们?

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

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

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

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

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

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

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JavaScript 基础四 函数使用、传参、返回值、作用域、匿名函数

JavaScript 基础四 函数 1.1 函数1.2 函数使用函数名命名规范函数的调用 1.3 函数传参(1)声明语法(2)调用有参函数(3)形参和实参(4)参数默认值 1.4 函数返回值(1&#xf…

HTML如何设置多图片上传,并限制格式类型

在HTML如何设置多图片上传&#xff0c;并限制格式类型为jpg和png格式。 <input type"file" name"fileInput" id"fileInput"> 上面这行代码&#xff0c;只支持单个文件上传&#xff0c;且不支持文件类型过滤&#xff0c;在实际开发过程中&…

Mybatis Plus 条件构造器的简单介绍 以及IService 的简单使用

文章目录 Mybatis Plus条件构造器LambdaWrapper自定义SQLService接口 Mybatis Plus 条件构造器 简单案例演示 特殊案例更新 UpdateWrapper 更新几个用户的余额减少200LambdaWrapper 因为上方为硬编码&#xff0c;所以不建议 而是使用反射&#xff08;Lambda&#xff09;这样…

AI的突破与融合:2024年中国智能技术的新纪元_光点科技

随着人工智能领域的不断突破&#xff0c;2024年注定将成为中国智能技术发展的一个新纪元。当下&#xff0c;AI技术不仅在理论研究上取得了重大进展&#xff0c;其在商业应用、社会服务等领域的融合也日益深入。本文将结合近期网络上的AI热点&#xff0c;展望中国在AI技术方面的…

Windows关闭自动更新

winR 打开运行界面&#xff0c;输入taskschd.msc 在任务计划程序中找到/Microsoft/Windows/WindowsUpdate&#xff0c;并禁用该任务。

告别复杂排版:Markdown语法指南

导语&#xff1a;Markdown作为一种轻量级的标记语言&#xff0c;以其简洁、易学的语法和强大的兼容性赢得了广泛的应用。本文将为您详细介绍Markdown的起源、基本语法及其在写作、博客、项目管理等场景的应用&#xff0c;带您领略这一简洁高效的文本编写工具的无穷魅力。 Mark…

超实用!人力资源都在找的工资核算管理系统,内含公式,个税自动计算!

——“有没有推荐的工资核算管理系统&#xff1f;” ——“你要做工资核算&#xff0c;就不能只做工资核算。” 人事的工作一般分为“六大模块”&#xff0c;即人力资源规划、招聘与配置、培训与开发、绩效管理、薪酬福利管理、劳动关系管理&#xff0c;六大模块分工协作并有…

JRT打印元素绘制协议之-A4Double

以前打印相信很多人因为A4打印两个报告头大过&#xff0c;M要把一堆报告既有A4的也有A5的还有微生物的&#xff0c;可能输出Page还不那么严谨。要么换页不对叠加了、要么多空白页、中间夹杂A4报告就更加头大。也有的人为了打印页码的共几页而头大。 借助新设计的优势&#xff…

搜维尔科技:【简报】第九届元宇宙数字人设计大赛,报名已经进入白热化阶段!

随着元宇宙时代的来临&#xff0c;数字人设计成为了创新前沿领域之一。为了提高大学生元宇宙虚拟人角色策划与美术设计的专业核心能力&#xff0c;我们特别举办了这场元宇宙数字人设计赛道&#xff0c;赛道主题为「AI人工智能科技」 &#xff0c;只要与「AI人工智能科技」相关的…

Docker·1(简介、安装、基本使用)

目录 1. 什么是Docker1.1. 什么是容器1.2. 什么是Docker 2. 安装Docker3. 镜像操作3.1. 拉取镜像3.2. 卸载镜像/容器3.3. 使用镜像/容器 4. 相关指令说明 1. 什么是Docker 1.1. 什么是容器 虚拟机&#xff1a; 操作系统是一个很笨重的程序&#xff0c;即是啥都不干&#xff0c…

MAC 中多显示器的设置(Parallels Desktop)

目录 一、硬件列表&#xff1a; 二、线路连接&#xff1a; 三、软件设置&#xff1a; 1. 设置显示器排列位置及显示参数 2. 分别设置外接显示器为&#xff1a;扩展显示器&#xff0c;内建显示器为主显示器 3. 设置Parallels Desktop屏幕参数 四、结果 一、硬件列表&a…

部署KVM虚拟化平台

一、KVM简介&#xff1a; KVM是Kernel Virtual Machine 的简写&#xff0c;目前Linux发行版必须在64位的系统环境才能运行KVM,同时硬件需要支持VT技术。KVM自Linux 2,6.20版本后就直接整合到Linux内核.它依托CPU虚拟化指令集&#xff08;如intel-VT.AMD-V&#xff09;实现高性…

RTSP/Onvif安防平台EasyNVR接入EasyNVS显示服务不存在的原因及解决办法

EasyNVS云管理平台具备汇聚与管理EasyGBS、EasyNVR等平台的能力&#xff0c;可以将接入的视频资源实现统一的视频能力输出&#xff0c;支持远程可视化运维等管理功能&#xff0c;还能解决设备现场没有固定公网IP却需要在公网直播的需求。 有用户在现场部署EasyNVR&#xff0c;…

HTTP模块

一、Web服务器 当应用程序&#xff08;客户端&#xff09;需要某一个资源时&#xff0c;可以向一台服务器&#xff0c;通过Http请求获取到这个资源&#xff1b;提供资源的这个服务器&#xff0c;就是一个Web服务器 目前有很多开源的Web服务器&#xff1a;Nginx、Apache&#xf…

设计领域不可缺少的10款PS插件,2023年必知!

即时设计 即时设计是一个更快更简单的产品协作设计平台。即时设计主要用户人群涵盖产品经理&#xff0c;设计师和工程师&#xff0c;其核心功能点包括&#xff1a;智能标注、一键切图、多样批注、快速交互、全貌画板、团队管理。从产品到开发&#xff0c;真正实现了只要一个文…

MySQL之表的记录操作

前言 存数据不是目的&#xff0c;目的是能够将存起来的数据取出来或者查出来&#xff0c;并且能够对数据进行增删改查操作&#xff0c;本文将详细介绍表中记录的增删改查操作。对记录的操作属于DML数据库操作语言&#xff0c;可以通过SQL实现对数据的操作&#xff0c;包括实现向…

MSE Serverless 正式商用,构建低成本高弹性的微服务架构

作者&#xff1a;问思 微服务架构充分提升了研发效率&#xff0c;解决了复杂业务系统的快速迭代问题。但随着业务及技术演进&#xff0c;各种微服务组件也愈发复杂。如何实现更敏捷的开发&#xff0c;降低微服务开发运维成本&#xff0c;做到全链路的弹性&#xff0c;保障整个…

RT-Thread 14. GD32F330RBT6 Keil4移植RT-Thread

1.增加rt-thread-v4.1.0源码 rt-thread-v4.1.0\bsp\gd32350r-eval复制重命名为gd32f330_v1 2.文件组织结构 Usr&#xff1a;存放App任务应用&#xff0c;属于应用层&#xff0c;完全脱离硬件 CMSIS&#xff1a;硬件层&#xff0c;启动文件、系统文件 Driver&#xff1a;硬件外…

【无标题】idea的lombok插件支持@SuperBuilder注解啦

在我的博客阅读本文 1. 前言 今早进公司打开idea&#xff0c;弹出更新提示&#xff0c;简单看了下&#xff0c;原来是idea的lombok插件更新了&#xff0c;惊喜的发现update log上写着Add support for SuperBuilder。 为什么说是惊喜呢&#xff1f;因为之前也有用到这个的场景…

树莓派外设开发综述及WiringPi库安装

树莓派常用的接口 IO&#xff1a; input&#xff1a; output&#xff1a; PWM&#xff1a; IIC spi uart WiringPi库 树莓派一个非常重要的 WiringPi库&#xff08;Linux下 动态库 ”.so“ 静态库”.a"&#xff09; &#xff08;是一个特定平台&#xff0c;特…