前言
对于引用字段,首先我们需要决定关联的目标模型(或comodel)。但有时我们会让用户来做决定,首先选定我们所要的模型然后记录想要关联的记录。
实现
ref_doc_id = fields.Reference(
selection='_referencable_models',
string='Reference Document')
@api.model
def _referencable_models(self):
models = self.env['ir.model'].search([
('field_id.name', '=', 'message_ids')])
return [(x.model, x.name) for x in models]
运行原理
引用字段类似于many-to-one字段,不同的是它们允许用户选择要关联的模型。
目标模型可通过由selection属性提供的列表进行选择。selection属性应是一个包含两个元素的元组,第一个元素是模型的内部标识符,第二个是它的文件描述。
例如:
[('res.users', 'User'), ('res.partner', 'Partner')]
但是,不需要提供一个固定的列表,我们可以使用最通用的模型。为进行简化,我们使用带有消息功能的所有模型。使用_referencable_models方法,我们动态地提供了一个模型列表。
本节一开始提供了一个函数来浏览所有模型记录,可供动态引用来创建用于提供给selection属性的列表。虽然两种形式都允许,我们在引号内声明了函数名,而不是不加引号直接引用函数。这更为灵活,比如它允许所引用的函数可以在代码的后面进行定义,在使用直接引用时则不能这么做。
该函数需要一个@api.model装饰器,因为它在模型级别而非记录集级别上进行操作。
虽然这个功能看起来很棒,它运行的开销会很大。使用引用字段显示大量记录(如在列表视图中)会带来很重的数据库负载,因为每个值都需在一个单独的查询中进行查找。它也不能像常规关联字段那样利用数据库的引用一致性。