Exam.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445
  1. <?php
  2. namespace addons\yexam\service;
  3. use app\admin\model\yexam\Answer;
  4. use app\admin\model\yexam\ExamConfig;
  5. use app\admin\model\yexam\ExamUser;
  6. use app\admin\model\yexam\ExamUserLog;
  7. use app\admin\model\yexam\QuestionLog;
  8. use think\Exception;
  9. class Exam
  10. {
  11. public $model;
  12. public $error;
  13. public function __construct()
  14. {
  15. $this->model = new \app\admin\model\yexam\Exam();
  16. }
  17. /**
  18. * 获取考试列表
  19. * @param $page
  20. * @param $limit
  21. * @throws \think\Exception
  22. */
  23. public function getExamList($where,$page,$limit){
  24. $count = $this->model->where($where)->count();
  25. if($page){
  26. $data = $this->model->where($where)->page($page,$limit)->order("sort asc")->select();
  27. }else{
  28. $data = $this->model->where($where)->order("sort asc")->select();
  29. }
  30. return ['total'=>$count,'data'=>$data];
  31. }
  32. /**
  33. * 获取考试记录(已交卷)
  34. * @param $user_id
  35. * @param $page
  36. * @param $limit
  37. * @return bool
  38. * @throws \think\db\exception\DataNotFoundException
  39. * @throws \think\db\exception\ModelNotFoundException
  40. * @throws \think\exception\DbException
  41. */
  42. public function getRecordList($subject_id,$user_id,$page,$limit){
  43. $examUserModel = new ExamUser();
  44. $total = $examUserModel
  45. ->alias("exam_user")
  46. ->where(['user_id'=>$user_id,'subject_id'=>$subject_id,'up_status'=>2])->count();
  47. if($page){
  48. $data = $examUserModel
  49. ->alias("exam_user")
  50. ->where(['user_id'=>$user_id,'subject_id'=>$subject_id,'up_status'=>2])
  51. ->page($page,$limit)
  52. ->order("id desc")
  53. ->select();
  54. }else{
  55. $data = $examUserModel
  56. ->alias("exam_user")
  57. ->where(['user_id'=>$user_id,'subject_id'=>$subject_id,'up_status'=>2])
  58. ->order("id desc")
  59. ->select();
  60. }
  61. return ['total'=>$total,'data'=>$data];
  62. }
  63. /**
  64. * 判断系统时间,自动交卷
  65. * @param $exam_id
  66. */
  67. private function over_time_up($exam_id,$user_id){
  68. $examUserModel = new ExamUser();
  69. $examUser = $examUserModel->where(['exam_id'=>$exam_id,'user_id'=>$user_id,'up_status'=>1,'endtime'=>['elt',time()]])->find();
  70. if($examUser){
  71. $examUserLog = new ExamUserLog();
  72. $score = $examUserLog->where(['exam_user_id'=>$examUser['id']])->sum('score');
  73. $score = empty($score)?0:$score;
  74. $examUser->save(['up_status'=>2,'up_time'=>time(),'score'=>$score]);
  75. }
  76. }
  77. /**
  78. * 开始考试
  79. */
  80. public function begin($exam_id,$user_id){
  81. $examInfo = $this->model->where(['id'=>$exam_id,'start_date'=>['elt',date("Y-m-d H:i:s",time())],'end_date'=>['egt',date("Y-m-d H:i:s",time())]])->find();
  82. if(empty($examInfo)){
  83. $this->error = "当前考试不存在或者不在考试时间范围内";
  84. return false;
  85. }
  86. if(empty($examInfo['question_ids'])){
  87. $this->error = "当前考试题目数量为0,请联系老师";
  88. return false;
  89. }
  90. //判断时间,自动交卷
  91. $this->over_time_up($exam_id,$user_id);
  92. //获取是否存在正在考试的记录
  93. $examUserModel = new ExamUser();
  94. $examUser = $examUserModel->where(['exam_id'=>$exam_id,'user_id'=>$user_id,'up_status'=>1])->find();
  95. if(empty($examUser)){
  96. if($examInfo['type'] == 1){
  97. $examUserModel = new ExamUser();
  98. $count = $examUserModel->where(['exam_id'=>$exam_id,'user_id'=>$user_id])->count();
  99. if($count){
  100. $this->error = "当前考试只能参加一次";
  101. return false;
  102. }
  103. }
  104. $endtime = time()+$examInfo['givetime']*60;
  105. $examUserModel = new ExamUser();
  106. $examUserModel->save([
  107. 'subject_id'=>$examInfo['subject_id'],
  108. 'type'=>$examInfo['type'],
  109. 'exam_id'=>$examInfo['id'],
  110. 'user_id'=>$user_id,
  111. 'exam_name'=>$examInfo['exam_name'],
  112. 'begintime'=>time(),
  113. 'endtime'=>$endtime,
  114. 'total_scode'=>$examInfo['score'],
  115. 'question_ids'=>$examInfo['question_ids']]);
  116. }else{
  117. $endtime = $examUser['endtime'];
  118. }
  119. return $endtime;
  120. }
  121. /**
  122. * 考试答题卡
  123. */
  124. public function getCard($exam_id,$user_id){
  125. $examInfo = $this->model->where(['id'=>$exam_id,'start_date'=>['elt',date("Y-m-d H:i:s",time())],'end_date'=>['egt',date("Y-m-d H:i:s",time())]])->find();
  126. if(empty($examInfo)){
  127. $this->error = "当前考试不存在或者不在考试时间范围内";
  128. return false;
  129. }
  130. //判断时间,自动交卷
  131. $this->over_time_up($exam_id,$user_id);
  132. //获取当前考试当前用户的正在考试的考试记录
  133. $examUserModel = new ExamUser();
  134. $examUserInfo = $examUserModel->where(['exam_id'=>$exam_id,'user_id'=>$user_id,'up_status'=>1])->find();
  135. if(empty($examUserInfo)){
  136. $this->error = "本次考试已结束";
  137. return false;
  138. }
  139. $questionModel = new \app\admin\model\yexam\Question();
  140. $data = $questionModel
  141. ->alias("question")
  142. ->field("question.id,question.type,(log.id IS NOT NULL) as state")
  143. ->join("yexam_exam_user_log log","question.id=log.question_id and log.exam_user_id=".$examUserInfo['id'],"left")
  144. ->where(['question.id'=>['in',$examInfo['question_ids']]])
  145. ->order("question.type asc,id asc")
  146. ->select();
  147. return $data;
  148. }
  149. private function getCurrentExam($exam_id,$user_id){
  150. //获取当前考试当前用户的正在考试的考试记录
  151. $examUserModel = new ExamUser();
  152. $examUser = $examUserModel->where(['exam_id'=>$exam_id,'user_id'=>$user_id,'up_status'=>1])->find();
  153. return $examUser;
  154. }
  155. /**
  156. * 获取上次答题定位num
  157. */
  158. public function getLastPosition($exam_id,$user_id){
  159. $examInfo = $this->model->where(['id'=>$exam_id,'start_date'=>['elt',date("Y-m-d H:i:s",time())],'end_date'=>['egt',date("Y-m-d H:i:s",time())]])->find();
  160. if(empty($examInfo)){
  161. $this->error = "当前考试不存在或者不在考试时间范围内";
  162. return false;
  163. }
  164. $examUser = $this->getCurrentExam($exam_id,$user_id); //获取当前考试当前用户的正在考试的考试记录
  165. if(empty($examUser)){
  166. $this->error = "本次考试已结束";
  167. return false;
  168. }
  169. $examUserLog = new ExamUserLog();
  170. $userlog = $examUserLog
  171. ->alias("log")
  172. ->field("log.question_id")
  173. ->join("yexam_question question","log.question_id=question.id",'inner')
  174. ->where(['exam_id'=>$exam_id,'exam_user_id'=>$examUser['id']])->order('log.lasttime desc')->find();
  175. if(empty($userlog)){
  176. return 1;
  177. }else{
  178. $questionModel = new \app\admin\model\yexam\Question();
  179. $questionList = $questionModel->field("id")->where(['id'=>['in',$examInfo['question_ids']]])->order("type asc,id asc")->select();
  180. $num = 1;
  181. foreach($questionList as $k=>$v){
  182. if($userlog['question_id'] == $v['id']){
  183. break;
  184. }
  185. $num++;
  186. }
  187. return $num;
  188. }
  189. }
  190. /**
  191. * 获取题目详情
  192. * @throws Exception
  193. */
  194. public function getQuestion($exam_id,$question_id,$user_id){
  195. $examUser = $this->getCurrentExam($exam_id,$user_id); //获取当前考试当前用户的正在考试的考试记录
  196. if(empty($examUser)){
  197. $this->error = "本次考试已结束";
  198. return false;
  199. }
  200. $questionModel = new \app\admin\model\yexam\Question();
  201. $question = $questionModel->field("id,question_name,type,right_answer,area")->where(['id'=>$question_id])->find();
  202. if(empty($question)){
  203. $this->error = "当前题目不存在";
  204. return false;
  205. }else{
  206. //获取答案
  207. if($question['type'] == 3){
  208. $question['answers'] = array(
  209. '0' => array(
  210. 'answer' => "对",
  211. 'answer_code' => "1",
  212. 'id' => 1,
  213. 'question_id' => $question['id']
  214. ),
  215. '1' => array(
  216. 'answer' => "错",
  217. 'answer_code' => "0",
  218. 'id' => 2,
  219. 'question_id' => $question['id']
  220. ),
  221. );
  222. }else{
  223. $answerModel = new Answer();
  224. $question['answers'] = $answerModel->where(['question_id'=>$question['id']])->select();
  225. }
  226. $examUserLog = new ExamUserLog();
  227. $userLog = $examUserLog->where(['exam_user_id'=>$examUser['id'],'question_id'=>$question['id']])->find();
  228. if($userLog){
  229. $question['user_answer'] = $userLog['user_answer'];
  230. }else{
  231. $question['user_answer'] = "";
  232. }
  233. }
  234. return $question;
  235. }
  236. /**
  237. * 手动交卷
  238. */
  239. public function up($exam_id,$user_id){
  240. $examInfo = $this->model->where(['id'=>$exam_id])->find();
  241. if(empty($examInfo)){
  242. $this->error = "当前考试不存在";
  243. return false;
  244. }
  245. //获取当前考试当前用户的正在考试的考试记录
  246. $examUserModel = new ExamUser();
  247. $examUser = $examUserModel->where(['exam_id'=>$exam_id,'user_id'=>$user_id,'up_status'=>1])->find();
  248. if(empty($examUser)){
  249. $this->error = "本次考试已结束";
  250. return false;
  251. }
  252. $examUserLog = new ExamUserLog();
  253. $score = $examUserLog->where(['exam_user_id'=>$examUser['id']])->sum('score');
  254. $score = empty($score)?0:$score;
  255. $up_time = time()>$examUser['endtime']?$examUser['endtime']:time();
  256. $examUser->save(['up_status'=>2,'up_time'=>$up_time,'score'=>$score]);
  257. //获取答对题目数
  258. $examUserLog = new ExamUserLog();
  259. $right_num = $examUserLog->where(['exam_user_id'=>$examUser['id'],'state'=>1])->count();
  260. //总题数
  261. $questionModel = new \app\admin\model\yexam\Question();
  262. $num = $questionModel->where(['id'=>['in',$examInfo['question_ids']]])->count();
  263. return ['score'=>$score,'up_time'=>$up_time,'usetime'=>$up_time-$examUser['begintime'],'dadui_num'=>$right_num,'total_num'=>$num];
  264. }
  265. /**
  266. * 答题
  267. */
  268. public function answer($exam_id,$user_id,$question_id,$answer){
  269. //判断时间,自动交卷
  270. $this->over_time_up($exam_id,$user_id);
  271. //获取当前考试当前用户的正在考试的考试记录
  272. $examUserModel = new ExamUser();
  273. $examUser = $examUserModel->where(['exam_id'=>$exam_id,'user_id'=>$user_id,'up_status'=>1])->find();
  274. if(empty($examUser)){
  275. $this->error = "本次考试已结束";
  276. return false;
  277. }
  278. //获取当前题目正确/错误
  279. $questionModel = new \app\admin\model\yexam\Question();
  280. $questionInfo = $questionModel->where(['id'=>$question_id])->find();
  281. if(empty($questionInfo)){
  282. $this->error = "本次题目不存在";
  283. return false;
  284. }
  285. $examConfigModel = new ExamConfig();
  286. $config = $examConfigModel->where(['exam_id'=>$exam_id,'type'=>$questionInfo['type']])->find();
  287. if(empty($config)){
  288. $_score = 0;
  289. }else{
  290. $_score = $config['score'];
  291. }
  292. if ($answer == $questionInfo['right_answer']) {
  293. $score = $_score;
  294. $state = 1;
  295. }else{
  296. $score = 0;
  297. $state = 0;
  298. }
  299. //获取题目分数
  300. $examUserLog = new ExamUserLog();
  301. if($userlog = $examUserLog->where(['exam_id'=>$exam_id,'exam_user_id'=>$examUser['id'],'question_id'=>$question_id])->find()){
  302. $userlog->save([
  303. 'user_answer' => $_POST['answer'],
  304. 'lasttime' => time(),
  305. 'state' => $state,
  306. 'score' => $score
  307. ]);
  308. }else{
  309. $examUserLog = new ExamUserLog();
  310. $examUserLog->save([
  311. 'exam_user_id' => $examUser['id'],
  312. 'user_id' => $user_id,
  313. 'exam_id' => $exam_id,
  314. 'question_id' => $question_id,
  315. 'user_answer' => $answer,
  316. 'lasttime' => time(),
  317. 'state' => $state,
  318. 'score' => $score
  319. ]);
  320. }
  321. return true;
  322. }
  323. /**
  324. * 获取考试错题列表
  325. */
  326. public function getErrorLogList($subject_id,$type,$user_id,$page,$limit){
  327. $logModel = new ExamUserLog();
  328. $count = $logModel
  329. ->alias("log")
  330. ->join("yexam_exam_user exam_user","log.exam_user_id=exam_user.id",'inner')
  331. ->join("yexam_exam exam","log.exam_id=exam.id",'inner')
  332. ->where(['exam.subject_id'=>$subject_id,'exam.type'=>$type,'log.user_id'=>$user_id,'log.state'=>0])->group("exam.id")->count();
  333. if($page){
  334. $data = $logModel
  335. ->alias("log")
  336. ->field("exam_user.id,exam_user.exam_name,count(log.id) as error_num")
  337. ->join("yexam_exam_user exam_user","log.exam_user_id=exam_user.id",'inner')
  338. ->join("yexam_exam exam","log.exam_id=exam.id",'inner')
  339. ->where(['exam.subject_id'=>$subject_id,'exam.type'=>$type,'log.user_id'=>$user_id,'log.state'=>0])
  340. ->page($page,$limit)
  341. ->group("exam_user.id")
  342. ->select();
  343. }else{
  344. $data = $logModel
  345. ->alias("log")
  346. ->field("exam_user.id,exam_user.exam_name,count(log.id) as error_num")
  347. ->join("yexam_exam_user exam_user","log.exam_user_id=exam_user.id",'inner')
  348. ->join("yexam_exam exam","log.exam_id=exam.id",'inner')
  349. ->where(['exam.subject_id'=>$subject_id,'exam.type'=>$type,'log.user_id'=>$user_id,'log.state'=>0])
  350. ->group("exam_user.id")
  351. ->select();
  352. }
  353. return ['total'=>$count,'data'=>$data];
  354. }
  355. /**
  356. * 考试错题答题卡
  357. */
  358. public function getErrorCard($exam_user_id,$user_id){
  359. //获取当前考试当前用户的正在考试的考试记录
  360. $examUserModel = new ExamUser();
  361. $examUserInfo = $examUserModel->where(['id'=>$exam_user_id,'user_id'=>$user_id])->find();
  362. if(empty($examUserInfo)){
  363. $this->error = "本次考试已结束";
  364. return false;
  365. }
  366. $questionModel = new \app\admin\model\yexam\Question();
  367. $data = $questionModel
  368. ->alias("question")
  369. ->field("question.id,log.state")
  370. ->join("yexam_exam_user_log log","question.id=log.question_id and log.exam_user_id=".$examUserInfo['id'],"inner")
  371. ->where(['question.id'=>['in',$examUserInfo['question_ids']],'log.state'=>0])
  372. ->order("question.type asc,id asc")
  373. ->select();
  374. return $data;
  375. }
  376. }