odoo17 | 模型视图继承

前言

Odoo的强大之处在于它的模块化。模块专门用于满足业务需求,但模块也可以彼此交互。这对于扩展现有模块的功能非常有用。例如,在我们的房地产场景中,我们希望在常规用户视图中直接显示销售人员的属性列表。

但是在讨论特定的Odoo模块继承之前,让我们看看如何更改标准CRUD(创建、检索、更新或删除)方法的行为。

Python 继承

目标

  • 设定不能删除非新建或已取消的属性。
    在这里插入图片描述
  • 创建产品/服务后,属性状态应更改为“已收到产品/服务”
  • 不能创建一个比现有报价更低的报价
    在这里插入图片描述
    在我们的房地产模块中,我们不需要开发任何特定的东西来执行标准的CRUD操作。Odoo框架提供了完成这些任务所需的工具。事实上,由于经典的Python继承,这些操作已经包含在我们的模型中:
from odoo import fields, modelsclass TestModel(models.Model):_name = "test_model"_description = "Test Model"...

我们的类TestModel继承自提供create()、read()、write()unlink()功能的models.Model

这些方法(以及Model类上定义的任何其他方法)可以扩展以添加特定的业务逻辑:

from odoo import fields, modelsclass TestModel(models.Model):_name = "test_model"_description = "Test Model"...@api.modeldef create(self, vals):# 做一些业务逻辑,修改值…...# 然后调用super来执行父方法return super().create(vals)

装饰器model()对于create()方法是必需的,因为记录集本身的内容与创建上下文无关,但对于其他CRUD方法则不是必需的。

同样重要的是要注意,尽管我们可以直接覆盖**unlink()方法,但您几乎总是希望用ondelete()装饰器来编写一个新方法。带有此装饰符的方法将在unlink()期间被调用,从而避免了当unlink()**被直接覆盖时卸载模型模块时可能出现的一些问题。

Python 3中,super()等价于super(TestModel, self)。当您需要使用修改后的记录集调用父方法时,后者可能是必需的。

注意

总是调用super()以避免中断流程是非常重要的。只有一些非常特殊的情况下你用调用super()。

确保始终返回与父方法一致的数据。例如,如果父方法返回dict(),则重写也必须返回dict()。

模型继承

在我们的房地产模块中,我们希望显示与销售人员相关的房地产列表 直接在“设置”/“用户和公司”/“用户”窗体视图中。为此,我们需要将一个字段添加到 模型并调整其视图以显示它。res.users

Odoo提供了两种继承机制,以模块化方式扩展现有模型。

第一种继承机制允许模块通过以下方式修改另一个模块中定义模型的行为

  • 向模型添加字段,

  • 覆盖模型中字段的定义,

  • 向模型添加约束,

  • 向模型添加方法,

  • 重写模型中的现有方法。

第二种继承机制(委托)允许模型的每个记录链接到父模型的记录,并提供对该父记录字段的透明访问。

在这里插入图片描述

在Odoo中,第一种机制是目前使用最多的。在我们的示例中,我们希望向现有模型添加一个字段,这意味着我们将使用第一种机制。例如:

from odoo import fields, modelsclass InheritedModel(models.Model):_inherit = "inherited.model"new_field = fields.Char(string="New Field")

可以在此处找到向模型添加两个字段的实际示例。

# -*- coding: utf-8 -*-
# Part of Odoo. See LICENSE file for full copyright and licensing details.from odoo import models, fields, api, _class AccountMove(models.Model):_inherit = 'account.move'def _post(self, soft=True):vendor_bill_service = self.env.ref('account_fleet.data_fleet_service_type_vendor_bill', raise_if_not_found=False)if not vendor_bill_service:return super()._post(soft)val_list = []log_list = []not_posted_before = self.filtered(lambda r: not r.posted_before)posted = super()._post(soft)  # We need the move name to be set, but we also need to know which move are posted for the first time.for line in (not_posted_before & posted).line_ids.filtered(lambda ml: ml.vehicle_id):val = {'service_type_id': vendor_bill_service.id,'vehicle_id': line.vehicle_id.id,'amount': line.price_subtotal,'vendor_id': line.partner_id.id,'description': line.name,}log = _('Service Vendor Bill: <a href=# data-oe-model=account.move data-oe-id={move_id}>{move_name}</a>').format(move_id=line.move_id.id,move_name=line.move_id.name,)val_list.append(val)log_list.append(log)log_service_ids = self.env['fleet.vehicle.log.services'].create(val_list)for log_service_id, log in zip(log_service_ids, log_list):log_service_id.message_post(body=log)return postedclass AccountMoveLine(models.Model):_inherit = 'account.move.line'vehicle_id = fields.Many2one('fleet.vehicle', string='Vehicle')need_vehicle = fields.Boolean(compute='_compute_need_vehicle',help="Technical field to decide whether the vehicle_id field is editable")def _compute_need_vehicle(self):self.need_vehicle = False

按照惯例,每个继承的模型都在自己的Python文件中定义。在我们的例子中,它将是models/inherited_model.py。

视图继承

目标

  • 链接到销售人员的可用属性列表应显示在其用户窗体视图中
    在这里插入图片描述
    Odoo不是对现有视图进行修改(通过覆盖它们),而是提供视图继承,其中子“扩展”视图应用于根视图之上。这些扩展既可以添加也可以从父视图删除内容。

扩展视图使用 inherit_id 字段引用其父视图。与其单个视图不同,其 arch 字段包含多个 xpath 元素,用于选择和更改其父视图的内容

<record id="inherited_model_view_form" model="ir.ui.view"><field name="name">inherited.model.form.inherit.test</field><field name="model">inherited.model</field><field name="inherit_id" ref="inherited.inherited_model_view_form"/><field name="arch" type="xml"><!-- find field description and add the fieldnew_field after it --><xpath expr="//field[@name='description']" position="after"><field name="new_field"/></xpath></field>
</record>

expr 表达式

一个XPath表达式,用于选择父视图中的一个元素。如果它没有匹配任何元素或匹配多个元素,则引发错误

position

应用于匹配元素的运算:

inside 在…内

将xpath的body附加到匹配元素的末尾

replace 代替

用xpath的body替换匹配的元素,将新body中出现的任何$0节点替换为原始元素

before 之前

将xpath的body作为同级插入到匹配的元素之前

after 之后

将xpath的body作为匹配元素之后的同级元素插入

attributes 属性,特征

使用XPath主体中的特殊属性元素更改匹配元素的属性

当匹配单个元素时,可以直接在要查找的元素上设置position属性。下面的两个继承具有相同的结果。

<xpath expr="//field[@name='description']" position="after"><field name="idea_ids" />
</xpath><field name="description" position="after"><field name="idea_ids" />
</field>

可以在这里找到一个视图继承扩展的示例。

<?xml version='1.0' encoding='utf-8'?>
<odoo><record id="view_move_form" model="ir.ui.view"><field name="name">account.move.form</field><field name="model">account.move</field><field name="inherit_id" ref="account.view_move_form"/><field name="arch" type="xml"><xpath expr="//field[@name='line_ids']//field[@name='account_id']" position="after"><field name='need_vehicle' invisible='1'/><field name='vehicle_id' attrs="{'required': [('need_vehicle', '=', True), ('parent.move_type', '=', 'in_invoice')], 'column_invisible': [('parent.move_type', '!=', 'in_invoice')]}" optional='hidden'/></xpath><xpath expr="//field[@name='invoice_line_ids']//field[@name='account_id']" position="after"><field name='need_vehicle' invisible='1'/><field name='vehicle_id' attrs="{'required': [('need_vehicle', '=', True), ('parent.move_type', '=', 'in_invoice')], 'column_invisible': [('parent.move_type', '!=', 'in_invoice')]}" optional='hidden'/></xpath></field></record>
</odoo>

在下一章中,我们将学习如何 与其他模块交互。

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

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

相关文章

Joplin配合teracloud进行多版本客户端分别笔记同步

最近瞎搜索joplin&#xff0c;意外在github上搜到plugins&#xff0c;插件仓库&#xff0c;里面有一个思维导图的插件我还是蛮喜欢的&#xff0c;结果下载后安装发现&#xff0c;我当前的Jopin的版本如下 &#xff08;Joplin 2.7.15 (prod, win32) 同步版本: 3 配置文件版本: 4…

关于无人机上层控制的PID算法的思考

一、前言 背景介绍&#xff1a;PID虽然出现了很多年&#xff0c;但是目前工业界还是把PID作为主流的控制算法&#xff08;尽管学术界有很多非常时尚的控制算法&#xff0c;包括鲁邦控制&#xff0c;神经网络控制等等&#xff09;&#xff0c;PID的算法在于其不需要对系统进行复…

CSS基础笔记-03选择器

CSS基础笔记系列 《CSS基础笔记-01CSS概述》《CSS基础笔记-02动画》 前言 在前面两篇博客中&#xff0c;我实际上已经使用过了选择器。但到底什么是选择器、有什么作用&#xff0c;我反而不能表达出来。因此&#xff0c;决定记录了我的学习和思考。 什么是选择器 selector…

阿里云系统盘测评ESSD、SSD和高效云盘IOPS、吞吐量性能参数表

阿里云服务器系统盘或数据盘支持多种云盘类型&#xff0c;如高效云盘、ESSD Entry云盘、SSD云盘、ESSD云盘、ESSD PL-X云盘及ESSD AutoPL云盘等&#xff0c;阿里云百科aliyunbaike.com详细介绍不同云盘说明及单盘容量、最大/最小IOPS、最大/最小吞吐量、单路随机写平均时延等性…

Arduino开发实例-欧姆龙E3Z-D61光电传感器

欧姆龙E3Z-D61光电传感器 文章目录 欧姆龙E3Z-D61光电传感器1、E3Z-D61光电传感器介绍2、硬件准备及接线3、代码实现1、E3Z-D61光电传感器介绍 Omran 光电传感器可用于检测 5 至 100 毫米距离内的障碍物和物体。 传感器上有一个 LED,它始终熄灭,并在检测到障碍物时亮起。 您…

构建自己的私人GPT

创作不易&#xff0c;请大家多鼓励支持。 在现实生活中&#xff0c;很多人的资料是不愿意公布在互联网上的&#xff0c;但是我们又要使用人工智能的能力帮我们处理文件、做决策、执行命令那怎么办呢&#xff1f;于是我们构建自己或公司的私人GPT变得非常重要。 一、本地部署…

企业大文件传输慢的原因与解决方案

数据已经成为当今信息化社会的重要资源&#xff0c;而数据的流动和共享则是数据价值的实现。在很多情况下&#xff0c;企业需要传输大型文件&#xff0c;比如设计方案、视频内容、项目文档等。然而&#xff0c;企业大型文件传输缓慢的问题却让很多用户苦恼&#xff0c;不仅拖慢…

域名流量被劫持怎么办?如何避免域名流量劫持?

随着互联网不断发展&#xff0c;流量成为线上世界的巨大财富。然而一种叫做域名流量劫持的网络攻击&#xff0c;将会在不经授权的情况下控制或重定向一个域名的DNS记录&#xff0c;导致用户在访问一个网站时&#xff0c;被引导到另一个不相关的网站&#xff0c;从而劫持走原网站…

聊天Demo

文章目录 参考链接使用前端界面消息窗口平滑滚动至底部vue使用watch监听vuex中的变量变化 参考链接 vue.js实现带表情评论功能前后端实现&#xff08;仿B站评论&#xff09; vue.js实现带表情评论仿bilibili&#xff08;滚动加载效果&#xff09; vue.js支持表情输入 vue.js表…

DataFunSummit:2023年知识图谱在线峰会-核心PPT资料下载

一、峰会简介 AIGC&#xff0c;ChatGPT以及发布的GPT-4相信已经给大家带来足够的冲击&#xff0c;那么对于知识图谱的应用产生哪些变化和变革&#xff1f;知识图谱在其中如何发挥作用呢&#xff1f;通过LLM是否有可能辅助创建通用大规模知识图谱&#xff1f;AIGC时代下行业知识…

负载均衡概述

负载均衡 负载均衡 建立在现有网络结构之上&#xff0c;它提供了一种廉价有效透明的方法扩展网络设备和服务器的带宽、增加吞吐量、加强网络数据处理能力、提高网络的灵活性和可用性。 四层负载均衡 vs 七层负载均衡 四层负载均衡&#xff08;目标地址和端口交换&#xff09;…

高校电力能耗监测精细化管理系统,提升能源利用效率的利器

电力是高校不可离开的重要能源&#xff0c;为学校相关管理人员提供在线用能查询统计等服务。通过对学校照明用电、空调用电等数据的采集、监控、分析&#xff0c;为学校电能管理制定合理的能源政策提供参考。同时&#xff0c;也可以培养学生的节能意识&#xff0c;学校后勤电力…

通信触发流程

该示例方案主要介绍如何通过建立的Modbus或TCP通信来实现触发方案、协议解析、发送事件和以及响应配置等功能。 需求&#xff1a;使用Modbus通信触发指定流程运行。 搭建思路&#xff1a;在接收事件中使用协议组装&#xff0c;比较规则选择上升沿&#xff0c;当接收到的值从其…

Kali Linux——设置中文

【问题现象】 从下图可以看到&#xff0c;菜单全是英文的。对于英文不好的同学&#xff0c;使用起来很难受。 【解决方法】 1、获取root权限 su root 2、进入语言设置 dpkg-reconfigure locales 3、选择zh_CN.UTF-8 UTF-8 4、设置默认 5、安装完成 6、重启虚拟机 reboot…

locust 快速入门--一次接口压测

背景&#xff1a; 使用locust&#xff0c;借助webUI&#xff0c;完成一次接口压测 实现步骤&#xff1a; 完成locust环境配置 准备一个locustfile&#xff08;current_limiting_test.py&#xff09; from locust import HttpUser, task, events from locust.env import Envi…

PiflowX组件-JDBCWrite

JDBCWrite组件 组件说明 使用JDBC驱动向任意类型的关系型数据库写入数据。 计算引擎 flink 有界性 Sink: Batch Sink: Streaming Append & Upsert Mode 组件分组 Jdbc 端口 Inport&#xff1a;默认端口 outport&#xff1a;默认端口 组件属性 名称展示名称默…

秋招复习之堆

目录 前言 堆 堆的常用操作 堆的实现&#xff08;大根堆&#xff09; 1. 堆的存储与表示 2. 访问堆顶元素 3. 元素入堆 4. 堆顶元素出堆 Top-k 问题 方法一&#xff1a;遍历选择 方法二&#xff1a;排序 方法三&#xff1a;堆 总结 前言 秋招复习之堆。 堆 「堆 heap…

2024年如何借用电商新零售破局?新型商业模式——乐享甄选竞拍模式

2024年如何借用电商新零售破局&#xff1f;新型商业模式——乐享甄选竞拍模式 背景&#xff1a;经历疫情三年的黑天鹅&#xff0c;消费者对未来收入预期和不自信等悲观情绪&#xff0c;从而使得“勒紧腰带&#xff0c;少消费&#xff0c;不消费”&#xff0c;以简单实用成为了新…

航空公司管理系统(迷你版12306)

要求 今天分享一个之前辅导留学生的作业&#xff0c;作业要求如下&#xff1a; Project E: Airways Management System Overall description: Your team is employed by an Airways company for the implementation of a computer system responsible for a large part of th…

PyTorch常用工具(1)数据处理

文章目录 前言1 数据处理1.1 Dataset1.2 DataLoader 前言 在训练神经网络的过程中需要用到很多的工具&#xff0c;最重要的是数据处理、可视化和GPU加速。本章主要介绍PyTorch在这些方面常用的工具模块&#xff0c;合理使用这些工具可以极大地提高编程效率。 由于内容较多&am…