我们经常在程序中遇到这样的场景,在一个作业界面点击某个按钮或者进行某类操作,然后触发到另外一个作业界面的逻辑,此时我们有时不仅仅
需要将特定的数据库字段传过去,还可能将第一个界面的某些信息传递到跳转的界面。举个例子:
学生选课的时候,在课程界面点击选课,然后会将课程的相关信息传递到学生界面,但是我们还需要将课程界面的操作人员(可能不是学生本人)传递过去,
此时就需要用到context了。
# -*- coding: utf-8 -*-
from odoo import models, fields, api
from odoo.exceptions import UserError, ValidationError
from odoo.modules.module import get_module_resource
from odoo import tools, _
import random
import time
class course(models.Model):
_name = 'xksystem.course'
_description = 'xksystem.course'
name = fields.Char(string=u'课程名')
code = fields.Char(string=u'课程编号')
studentlimit = fields.Integer(string=u'学生上限')
havestudent = fields.Integer(string=u'已选课学生数', compute='_compute_have_student')
teachers = fields.Many2many('xksystem.teacher', 'xksystem_course_xksystem_teacher_rel',
'course_id', 'teacher_id', string='授课教师')
course_score = fields.Float(string='课程学分')
remarks = fields.Html()
@api.constrains('course_score')
def check_course_score(self):
for record in self:
if record.course_score > 2:
raise ValidationError("The Score is too High: %s" % record.course_score)
#--重写name_get方法,同时返回课程名称和代码,这样别人在调用的时候就会明确的知道课程名称和课程代码
@api.multi
def name_get(self):
result = []
for record in self:
result.append((record.id, "%s(%s)" % (record.name, record.code)))
return result
@api.multi
def _compute_have_student(self):
for recode in self:
l_sql = "select course_id,count(*) from xksystem_studentcourseline " \
"where course_id = %s group by course_id" % (recode.id)
self.env.cr.execute(l_sql)
dicts = self.env.cr.dictfetchall()
#print(dicts)
if len(dicts) > 0:
recode.havestudent = dicts[0]['count']
else:
recode.havestudent = 0
@api.multi
def xk_btn(self):
# --点击选课按钮,然后创建一笔学生课程资料
res = self.env['res.users'].search([('id', '=', self.env.uid)]) # 获取当前用户的ID
print(res.login)
code = res.login # 获取当前用户的学号
res = self.env['xksystem.student'].search([('code', '=', code)]) # 以学号获取当前学生头表信息
if res.id:
# 检索是否已经选过此门课程
res_course = self.env['xksystem.studentcourseline'].search(['&', ('student_id', '=', res.id),
('coursecode', '=', self.id)])
if res_course:
print('此门课程已被选过了,不能重复选择!')
raise UserError(('你已经选过了这门课,不能重复选择!'))
else:
# 防止大量并发选课,每位学生随机停止0-1秒
sleep_time = random.random()
print(sleep_time)
time.sleep(sleep_time)
# 检索课程是否已经被选光
l_sql = "select course_id,count(*) from xksystem_studentcourseline " \
"where course_id = %s group by course_id" % (self.id)
self.env.cr.execute(l_sql)
dicts = self.env.cr.dictfetchall()
if len(dicts)==0:
havastudent_count = 0
else:
havastudent_count = dicts[0]['count']
if havastudent_count >= self.studentlimit:
raise UserError(('选课学生人数已经超过上限,请选择其他课程!'))
# 合规,系统进行选课
#在课程类中组学生课程信息,同时将当前操作用户传递过去,然后在student类中判断当前用户的合法性
context = {'用户名':code}
print('context:',context)
vals = {'linenumber': self.env['ir.sequence'].next_by_code('seq.test'), 'student_id': res.id,
'course_id': self.id, 'coursecode': self.code,}
self.with_context(context).env['xksystem.studentcourseline'].sudo().create(vals)
else:
raise UserError(('此账户不是学生账户,不能选课!'))
return True
class StudentCourseLine(models.Model):
@api.model_create_multi
def create(self, vals_list):
limit_student = self.env.context.get('用户名',False) #从上下文中获取对应的学生,限制其选课
if limit_student == 'S081001':
raise UserError(('此学生被限制选课,请联系学院管理员!'))
for rec in self:
print('----context test-----')
print(rec.context_get())
print('----context test-----')
return super(StudentCourseLine, self).create(vals_list)
这个例子就是从一个类跳转到另外一个类中,将第一个类中的信息用with_context形式传递到下游,然后下游的类再用self.env.context.get('用户名',False)获取对应的
键值,进而逻辑判断