Odoo View视图默认是不认识attrs中的uid的,其原因在于后台将xml转化为html的过程中对attrs调用了python的eval方法,而对于eval函数来说,我们传入的形如[('user.id','=',uid)]中的uid是没有声明的变量,因此,日志的报错信息才会为"name 'xxx' is not defined".
有时候我们确实希望attrs能够认识我们的uid,以方便我们动态地控制view中页面的显示内容。虽然我们也可以通过其他的方式,例如,field_view_get方法的重载,来曲线地达到我们的目的,但这样的方法不仅繁琐复杂,而且不具有普遍的适应性。而假如我们通过某种方式让Odoo认识了attrs中的uid那么不仅我们可以将之用到button上,也可以用到field,page上,要比field_view_get方法灵活的多。
那么,究竟如何才能让odoo认识attrs中的uid呢?
我们首先要来了解xml中的attrs是如何被识别转化html的。Odoo中的xml都是通过ir.ui.view对象进行处理后,解析为前端页面展示出来。在ir.ui.view对象中有一个postprocess方法用来处理诸如xml中节点的显示,group限制等等功能,在该方法里内嵌了一个处理node的transfer_node_to_modifiers方法,该方法定义在orm中,而我们要实现我们的目的就是要通过重载这个方法来实现。
def transfer_node_to_modifiers(node, modifiers, context=None, in_tree_view=False):
if node.get('attrs'):
modifiers.update(eval(node.get('attrs'))) if node.get('states'):
if 'invisible' in modifiers and isinstance(modifiers['invisible'], list):
# TODO combine with AND or OR, use implicit AND for now.
modifiers['invisible'].append(('state', 'not in', node.get('states').split(',')))
else:
modifiers['invisible'] = [('state', 'not in', node.get('states').split(','))] for a in ('invisible', 'readonly', 'required'):
if node.get(a):
v = bool(eval(node.get(a), {'context': context or {}}))
if in_tree_view and a == 'invisible':
# Invisible in a tree view has a specific meaning, make it a
# new key in the modifiers attribute.
modifiers['tree_invisible'] = v
elif v or (a not in modifiers or not isinstance(modifiers[a], list)):
# Don't set the attribute to False if a dynamic value was
# provided (i.e. a domain from attrs or states).
modifiers[a] = v
从上面的方法中,我们可以加入对uid的处理,如果发现attrs中有uid,就将之替换为我们当前登录用户的id:
if 'uid' in node.get('attrs'):
uid = 1#为啥是1,这是个秘密
这样我们就能在xml中从容地将uid用作我们的判断条件了。
等等,为啥上面的代码中uid=1?如何在orm文件中获取到当前登录用户的id呢?读者不如自己思考一下(其实我是给伸手党们挖了个坑)。
实在想不出来的同学可以联系我索取答案:mail:kfx2007@163.com (付费阅读500RMB,支付宝账号同邮箱)