from django.db import models
|
from exam.models import Exam, GeneratedPaper
|
from datetime import datetime
|
# Create your models here.
|
import numpy as np
|
|
class Scan(models.Model):
|
exam=models.ForeignKey(Exam, on_delete=models.CASCADE)
|
paper=models.ForeignKey(GeneratedPaper, on_delete=models.CASCADE)
|
page_no=models.IntegerField(default=1)
|
student_id=models.CharField(max_length=18, null=True, default=None, blank=True)
|
scan_date=models.DateTimeField(default=datetime.now)
|
answer_matrix=models.TextField(null=True, default='[]', blank=True)
|
coordinates=models.TextField(null=True, default=None, blank=True)
|
ocr_debug=models.TextField(null=True, default=None, blank=True)
|
grade=models.FloatField(default=-1)
|
percentage=models.FloatField(default=-1)
|
scan_image=models.ImageField(upload_to='scans/', blank=True)
|
|
def wrong_correct_matrices(self):
|
try:
|
all_pages=Scan.objects.all().filter(paper=self.paper).order_by('page_no')
|
except:
|
return None, None, None
|
if type(self.paper.answer_matrix)==type(""):
|
cmatrix=eval(self.paper.answer_matrix)
|
else:
|
cmatrix=self.paper.answer_matrix
|
|
amatrix=[]
|
curr_page=1
|
for p in all_pages:
|
if(p.page_no!=curr_page): #check if the page number is correct. If not return None matrices
|
return None, None, None
|
curr_page+=1
|
if type(p.answer_matrix)==type(""):
|
page_amatrix=eval(p.answer_matrix)
|
else:
|
page_amatrix=eval(p.answer_matrix)
|
amatrix=amatrix+page_amatrix
|
correct=(np.array(cmatrix)=='1')*1
|
answers=(np.zeros(np.shape(correct))==0)*0
|
for i,row in enumerate(amatrix):
|
for j, col in enumerate(row):
|
answers[i][j]=col
|
#print(correct, answers)
|
c=correct&answers
|
w=answers&~correct
|
return c,w,correct
|
|
def grade(self):
|
c,w,correct=self.wrong_correct_matrices()
|
if(c is None):
|
return -1000, -1000
|
grad_positive=np.sum(c*self.exam.mark_positive)
|
grad_negative=np.sum(w*self.exam.mark_negative)
|
for kv in range(0, self.exam.kvizek):
|
if(grad_negative!=0):
|
grad_negative-=self.exam.mark_negative
|
sum_grading=float(grad_positive+grad_negative)
|
grading_percent=np.ceil(sum_grading/(len(correct)*self.exam.mark_positive)*100)
|
return grading_percent, sum_grading
|
|
|
def __str__(self):
|
return "{} {} {} {}".format(self.exam, self.paper, self.page_no, self.student_id)
|
|
|
|
class ProblemReport(models.Model):
|
scan=models.ForeignKey(Scan, on_delete=models.CASCADE)
|
message=models.TextField(null=True, blank=True)
|
date=models.DateTimeField(default=datetime.now)
|
|
def __str__(self):
|
return "{}". format(self.message)
|